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

问题描述 投票:0回答:8

使用此表格:

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
作为键,名称的初始值也会丢失。有什么解决方法吗?我希望初始值能够将表单绑定为“默认值”。

django forms
8个回答
12
投票

稍微修改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 应用程序。我最近才发现的。


5
投票

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

您可以做的是定义一个

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

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

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


2
投票

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)

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']

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

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


0
投票

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

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)

0
投票

我解决问题的方法:

class SomeForm(forms.Form):
    # ...
    def __init__(self, data, *args, **kwargs):
        # get a mutable version of data
        super().__init__(data.copy(), *args, **kwargs)
        for name, field in self.fields.items():
            if field.initial:
                self.data.setdefault(name, field.initial)
© www.soinside.com 2019 - 2024. All rights reserved.