使用此表格:
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
作为键,名称的初始值也会丢失。有什么解决方法吗?我希望初始值能够将表单绑定为“默认值”。
稍微修改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 应用程序。我最近才发现的。
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
我使用以下模式将默认值设置为表单给出的初始值-
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
这确保所有具有初始值且不从用户获取值的字段都由其初始值填充。
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)
这行得通吗:
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']
所提出的解决方案要么对我不起作用,要么看起来不太优雅。 该文档指定初始不适用于绑定表单,这似乎是原始提问者(和我)的用例:
这就是为什么仅针对未绑定表单显示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
您还可以使用其他方式初始化表单(如上所述)。
没有一个答案实际上完全符合气候的要求。所以这是我针对同一问题的解决方案:
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)
我解决问题的方法:
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)