Джанго: Как сделать форму с пользовательскими шаблонами? - PullRequest
20 голосов
/ 04 февраля 2011

У меня есть модель:

class Setting(models.Model):

    class Meta:
        abstract = True

    name = models.CharField(max_length=120, primary_key=True)
    description = models.CharField(max_length=300, blank=True)

class IntegerSetting(Setting):
    value = models.IntegerField()

Я хотел бы создать форму, которая будет выглядеть примерно так:

<form method="POST" action="">
     {% for model in models %}
     <label>{{model.name}}</label> <input value='{{model.value}}' />
     <p>{{model.description}}</p>
     {% endfor %}
</form>

Я не совсем уверен, как это сделать. Возможно, мне нужно использовать formset?

from django.forms.models import modelformset_factory
from apps.about.models import Setting, IntegerSetting

def site_settings(request):
    formset = modelformset_factory(IntegerSetting)()

    return render_to_response("about/admin/site-settings.html", {'formset': formset}, RequestContext(request, {}))

Тогда в шаблоне я бы хотел отобразить форму иначе, чем по умолчанию. Однако я не совсем уверен, как получить доступ к свойствам модели. Это правильный подход, или есть другой способ, которым я должен идти об этом?

Обновление : Это то, чем я сейчас занимаюсь. Он воспроизводит именно так, как я хотел бы, кроме стиля. Тем не менее, я чувствую, что это глубоко взломано:

class SettingsForm(ModelForm):
    class Meta:
        model = IntegerSetting

    def as_table(self):
        bound_field = BoundField(self, self.fields['value'], 'value')
        return mark_safe("<tr><td><label>%s</label></td><td>%s\n<p class='help'>%s</p></td></tr>" % (self.instance.name,
                                                                       self.instance.description, 
                                                                       bound_field.__unicode__())) 

def edit_settings(request):
    forms = [SettingsForm(instance=intSetting) for intSetting in IntegerSetting.objects.all()]

    return render_to_response("admin/edit-settings.html", {'forms': forms}, RequestContext(request, {}))

редактирование-settings.html

{% extends "admin/base_site.html" %}
{% block title %}System Settings{% endblock %}

{% block content %}
    <form method="post" action="">
        <table>
        {% for form in forms %}
        {{form}}
        {% endfor %}
        </table>
    </form>
{% endblock %}

Есть ли лучший подход к этому?

Кроме того, я не уверен, возникнут ли у меня проблемы при отправке формы или нет.

Ответы [ 4 ]

19 голосов
/ 05 февраля 2011
<form action="/contact/" method="post">
    {% for field in form %}
        <div class="fieldWrapper">
            {{ field.errors }}
            {{ field.label_tag }}: {{ field }}
        </div>
    {% endfor %}
    <p><input type="submit" value="Send message" /></p>
</form>

Вы можете найти полную документацию здесь: http://docs.djangoproject.com/en/dev/topics/forms/#customizing-the-form-template

9 голосов
/ 04 февраля 2011

Я не думаю, что вам нужен набор форм здесь. Посмотрите здесь , если вы хотите собственный шаблон для одного представления. Если вы хотите создать свой собственный {{form.as_foobar}}, просто создайте подкласс form.Form, примерно так:

class MyForm(forms.Form):
  def as_foobar(self):
    return self._html_output(
      normal_row = u'%(label)s %(field)s%(help_text)s',
      error_row = u'%s',
      row_ender = '',
      help_text_html = u' %s',
      errors_on_separate_row = False)

и просто используйте его в вашем forms.py:

class ContactForm(MyForm):
  # ..
3 голосов
/ 14 ноября 2013

Для тех, кому нужна <table> версия ответа Джбкуртина:

<form method="post">{% csrf_token %}
  <table>
    {% for field in form %}
      <tr>
        <th>{{field.label_tag}}</th>
        <td>
          {{ field.errors }}
          {{ field }}
        </td>
      </tr>
    {% endfor %}
  </table>
  <hr/>
  <input type="submit" value="Conferma" />
</form>
1 голос
/ 05 февраля 2011

Похоже, вас могут заинтересовать django-floppyforms ( docs ), который дает вам гораздо больший контроль над рендерингом поля.

...