views:

316

answers:

5

I'm wrestling with how to best create HTML pages in Django that can either be used for displaying or editing data. That is, I'd like the field's values to appear as text in display mode, but in their widgets when in edit/add mode. It appears that Django wasn't designed to do this: the fields always appear in their widgets (eg, text input, text area, etc).

Is there a common technique for handling this, short of using forms for one, and not the other?

I was thinking of a custom templatetag filter that could be used for every form field, like:

{{ form.field_name|render_field:mode }}

where render_field would either return the field's HTML widget, or just the value as text, based on the mode.

Have I missed something, or is this a viable solution?

A: 

Since you are saving the data, you must have a model attached to the form somehow, a modelform or not. So you can just use that model directly to get the values and render it in a template like you want.

The above suggestion would be possible, but since forms can be rather complex, it's probably not an easy task or worth the bother. Depends on how often you want to do this. But then it would probably be easier to create a filter for the model instead of the form.

googletorp
Yes, I can just use the model, but that doesn't take advantage of anything that forms can do. What would a filter have to do with this? I have a lot of display/edit views, so a good choice could save us from having to do each one twice (once for display, once for edit).
D.S. Blank
A: 

in a main template do

{% if form %}
    {% include 'form.html' %}
{% else %}
    {% include 'display.html' %}
{% endif %}

then those two templates would be just the part that renders a form, or a nice organized div. this way, the edit view would have a form defined, and the display view would not. this will be a lot easier to keep track of than what you're trying to do (even though what you're trying should be possible, i believe your method will suffer in performance).

i do this in a blog app so that the edit page looks just like the display page, therefore making it extremely intuitive to edit/write blog entries. (not to mention this allows me to very easily put some ajax in to edit entries within any view with some nice fancy effects)

Brandon H
This is exactly what I don't want to do. I don't want to have two forms, but one.
D.S. Blank
A: 

Have I missed something

Forms not only display the field widgets but also handle the post data. A post sent would cause it to process the data cleanup, form and field error handling, etc.

It's kind of breaking the pattern - why create and render a form object only to tweak it not to look like a form?

If you are worried about too much work on templates, try to solve it with template inheritance in a best possible way. If you are very sure you want to change only the field tag, you can make sth like

{% if form %}
    {% for error in form.field.errors %}
        {{ error|escape }}
    {% endfor %}
    {{ form.field }}
{% else %}
    {{ object.field }}
{% endif %}

for every field, but IMO that's not the point, YMMV.

Also what comes to mind (thinking of your solution) is dynamically attach widgets to form fields, but that'd be overengineering.

zalew
A: 

I have the same problem. Right now I have separate templates for display and editing; the former renders object fields, the latter form fields (but sometimes also object fields, for things which are not editable). The HTML structure can be quite complex: for example, on some pages I have large tables representing a multi-level hierarchy of objects. As a result, I end up with a large amount of duplicated code in the two templates, which is the opposite of DRY.

I have used template filters with form fields before, to save code when displaying errors together with the field. I think it would be a viable solution in this case. Another possibility may be to use a ModelForm subclass which can be told to render uneditable versions of the fields. This would help keep the templates simple. You could even render both the static and the editable version of a field, and use JavaScript to switch between them, activating and deactivating an editing mode.

LaC
A: 

Answering my own question, apparently. I ended up with a three-part solution:

  1. attach the model to the form, so I'll have the default display widgets for each field
  2. write the form using a template tag, passing it the form.field, user, and action
  3. write a render template tag to handle #2

Step one:

form.model = Model(...)

Step two:


 {{form.field1.label}}
 {% render form.field1 user action %}


 {{form.field2.label}}
 {% render form.field2 user action %}

Step three:

Something like:

def render(formfield, user, action, default_text="Private"):
    if not user.is_authenticated():
        action = "view"
    if action == "view":
        if user.is_authenticated():
            fieldname = formfield.name 
            retval = str(getattr(formfield.form.model, fieldname))
        else:
            retval = default_text
    else:
        retval = formfield.as_widget()
    return retval
D.S. Blank