带有复选框的表单集

时间:2010-11-18 19:09:03

标签: django django-forms

我希望能够在页面上显示模型列表,并允许用户同时选择其中一些模型。

例如,假设我有一个用户选择屏幕,对于模型:

class User(model):
   first = # char field
   last  = # char field
   birthdate = # date

然后我想向用户展示并让他们选择其中一个:

Please select users:
[] John  Smith   Jan, 2001
[] Mike  Davis   Feb, 2002
[] John  Doe     Dec, 2000

[Continue]

然后,此表格将被发布并处理。

我能想到的一种方法是使用ModelFormset。 问题是当我尝试使用ModelFormsets显示用户时,我无法添加复选框。

我能想到的另一种方法是创建一个表单,并在其上输出一大堆带有特定id的复选框。然后在提交 - 迭代所有选中的复选框。不确定这对Django的形式有什么作用。

欢迎任何建议。感谢。

编辑:事实证明,通过给每个复选框提供一个ID(例如患者ID)在同一个名称组中,只需在django视图中查看POST字典就可以得到我需要的内容!

2 个答案:

答案 0 :(得分:3)

我创建了一个自定义的SelectMultiple小部件,用于显示对象详细信息以及一个复选框,用于在表单仍被称为newforms时返回选择 - 它似乎仍然有效:

from django.forms import CheckboxInput, SelectMultiple
from django.utils.encoding import force_unicode
from django.utils.html import escape
from django.utils.safestring import mark_safe

class TableSelectMultiple(SelectMultiple):
    """
    Provides selection of items via checkboxes, with a table row
    being rendered for each item, the first cell in which contains the
    checkbox.

    When providing choices for this field, give the item as the second
    item in all choice tuples. For example, where you might have
    previously used::

        field.choices = [(item.id, item.name) for item in item_list]

    ...you should use::

        field.choices = [(item.id, item) for item in item_list]
    """
    def __init__(self, item_attrs, *args, **kwargs):
        """
        item_attrs
            Defines the attributes of each item which will be displayed
            as a column in each table row, in the order given.

            Any callables in item_attrs will be called with the item to be
            displayed as the sole parameter.

            Any callable attribute names specified will be called and have
            their return value used for display.

            All attribute values will be escaped.
        """
        super(TableSelectMultiple, self).__init__(*args, **kwargs)
        self.item_attrs = item_attrs

    def render(self, name, value, attrs=None, choices=()):
        if value is None: value = []
        has_id = attrs and 'id' in attrs
        final_attrs = self.build_attrs(attrs, name=name)
        output = []
        str_values = set([force_unicode(v) for v in value]) # Normalize to strings.
        for i, (option_value, item) in enumerate(self.choices):
            # If an ID attribute was given, add a numeric index as a suffix,
            # so that the checkboxes don't all have the same ID attribute.
            if has_id:
                final_attrs = dict(final_attrs, id='%s_%s' % (attrs['id'], i))
            cb = CheckboxInput(final_attrs, check_test=lambda value: value in str_values)
            option_value = force_unicode(option_value)
            rendered_cb = cb.render(name, option_value)
            output.append(u'<tr><td>%s</td>' % rendered_cb)
            for attr in self.item_attrs:
                if callable(attr):
                    content = attr(item)
                elif callable(getattr(item, attr)):
                    content = getattr(item, attr)()
                else:
                    content = getattr(item, attr)
                output.append(u'<td>%s</td>' % escape(content))
            output.append(u'</tr>')
        return mark_safe(u'\n'.join(output))

示例表单:

class JobSelectionForm(forms.Form):
    jobs = forms.MultipleChoiceField(widget=TableSelectMultiple(
               item_attrs=('formatted_number', 'name', 'client', 'get_status_display')))

    def __init__(self, accessible_jobs, *args, **kwargs):
        super(JobSelectionForm, self).__init__(*args, **kwargs)
        self.fields['jobs'].choices = [(j.id, j) \
                                       for j in accessible_jobs]

使用上面的表单,您可以在实例化表单对象时传递要显示的项目列表作为第一个参数。

模板:

{% if form.jobs.errors %}{{ form.jobs.errors }}{% endif %}
<table>
<thead>
  <tr>
    <th>&nbsp;</th>
    <th>Number</th>
    <th>Name</th>
    <th>Client</th>
    <th>Status</th>
  </tr>
</thead>
<tbody>
{{ form.jobs }}
</tbody>
</table>

答案 1 :(得分:1)

您不需要表单集,因为它用于编辑一组模型实例中的数据。您想要的是包含ModelChoiceFieldModelMultipleChoiceField的单个表单 - 您可能需要更改窗口小部件以使用CheckboxSelectMultiple