django形式:绑定表单的默认值

时间:2013-04-15 23:47:24

标签: django forms

使用此表格:

class Form(forms.Form):
    name = forms.CharField(required=False, initial='Hello world')

如果我在视图中做了类似的事情:

form = Form(request.GET)
if form.is_valid():
    name = form.cleaned_data['name']

即使request.GET不包含name作为键,名称的初始值也会丢失。有没有解决方法?我希望初始值可以将表单绑定为“默认值”。

7 个答案:

答案 0 :(得分:7)

通过略微修改Gonzalo的解决方案,这是正确的方法:

class Form(forms.Form):
    name = forms.CharField(required=False, initial='Hello world')

    def clean_name(self):
        if not self['name'].html_name in self.data:
            return self.fields['name'].initial
        return self.cleaned_data['name']

如果您需要,可以查看django-filter app。我最近才发现它。

答案 1 :(得分:5)

initial并不真正用于为表单字段设置默认值。 相反,在向用户显示表单时,它实际上更像是一个占位符实用程序,如果不需要该字段,则无法正常工作(如示例所示)。

您可以做的是define a clean_<fieldname>方法,检查该字段是否为空值并返回默认值:

class Form(forms.Form):
    name = forms.CharField(required=False, initial='Hello world')

    def clean_name(self):
        name = self.cleaned_data['name']
        if name is None:
            return self.fields['name'].initial
        return name

答案 2 :(得分:4)

我使用以下模式将默认值设置为为表单提供的初始值 -

class InitialDefaultForm(forms.Form):
    def clean(self):
        cleaned_data = super(InitialDefaultForm, self).clean()
        # if data is not provided for some fields and those fields have an
        # initial value, then set the values to initial value
        for name in self.fields:
            if not self[name].html_name in self.data and self.fields[name].initial is not None:
                cleaned_data[name] = self.fields[name].initial
        return cleaned_data

这样可以确保所有具有初始值且不从用户获取值的字段都会按其初始值填充。

答案 3 :(得分:2)

这会起作用吗?

initial_form_data = {'name': 'Hello World'}   #put all the initial for fields in this dict
initial_form_data.update(request.GET)  #any field available in request.GET will override that field in initial_form_data
form = Form(initial_form_data)
if form.is_valid():
    name = form.cleaned_data['name']

答案 4 :(得分:2)

建议的解决方案要么对我不起作用,要么似乎不是很优雅。文档指定initial不适用于绑定表单,这似乎是原始提问者(和我的)用例:

  

这就是仅为未绑定表单显示初始值的原因。对于绑定表单,HTML输出将使用绑定数据。

     

https://docs.djangoproject.com/en/1.10/ref/forms/fields/#initial

我的解决方案是查看表单是否应该绑定:

initial = {'status': [Listing.ACTIVE], 'min_price': 123}     # Create default options

if request.method == 'GET':
    # create a form instance and populate it with data from the request:
    if len(request.GET):
        form = ListingSearchForm(request.GET)       # bind the form
    else:
        form = ListingSearchForm(initial=initial)   # if GET is empty, use default form

您还可以使用其他方式初始化表单(如上所述)。

答案 5 :(得分:1)

request.GET是一个类似于对象的字典。

initial仅适用于未绑定表单的情况。

表单有一个名为data的属性。在表单初始化期间,此属性作为第一个位置参数或data关键字参数提供。

绑定表单是指您将一些数据作为表单的第一个参数提供的表单,未绑定表单将data属性设置为None。

在表单form=Form(request.GET)的初始化中,您提供了第一个位置参数,因此在表单上设置了data属性,它成为绑定表单。即使request.GET是一个空字典,也会发生这种情况。由于您的表单成为绑定表单,因此initial name字段对其没有任何影响。

所以,在你GET请求中你应该这样做:

form = Form()

您的initial name字段将被尊重。

或者,如果您想从request.GET中读取name,如果它想要使用它而不是字段的首字母,那么请在您的视图中进行跟踪。

name = request.GET.get(name)
form_level_initial = {}
if name:
    form_level_initial['name'] = name
form = Form(initial=form_level_initial)

答案 6 :(得分:0)

这些答案实际上并没有完全符合clime的要求。所以这是我对同一问题的解决方案:

class LeadsFiltersForm(forms.Form):
    TYPE_CHOICES = Lead.TYPES
    SITE_CHOICES = [(site.id, site.name) for site in Site.objects.all()]

    type = forms.MultipleChoiceField(
        choices=TYPE_CHOICES, widget=forms.CheckboxSelectMultiple(),
        required=False
    )
    site = forms.MultipleChoiceField(
        widget=forms.CheckboxSelectMultiple(), required=False,
        choices=SITE_CHOICES
    )
    date_from = forms.DateField(input_formats=['%m-%d-%Y',], required=False,
                                widget=forms.TextInput(attrs={'placeholder': 'Date From'}),
                                initial=timezone.now() - datetime.timedelta(days=30))
    date_to = forms.DateField(input_formats=['%m-%d-%Y',], required=False,
                                widget=forms.TextInput(attrs={'placeholder': 'Date To'}))

    defaults = {
        'type': [val[0] for val in TYPE_CHOICES],
        'site': [val[0] for val in SITE_CHOICES],
        'date_from': (timezone.now() - datetime.timedelta(days=30)).strftime('%m-%d-%Y'),
        'date_to': timezone.now().strftime('%m-%d-%Y')
    }

    def __init__(self, data, *args, **kwargs):
        super(LeadsFiltersForm, self).__init__(data, *args, **kwargs)
        self.data = self.defaults.copy()
        for key, val in data.iteritems():
            if not data.get(key):
                continue
            field = self.fields.get(key)
            if field and getattr(field.widget, 'allow_multiple_selected', False):
                self.data[key] = data.getlist(key)
            else:
                self.data[key] = data.get(key)