# theoretical code, not tested
class PortStatus(models.Model):
status = models.CharField(max_length=100)
def __unicode__(self):
return self.status
# I prefer singular model names
class Result(models.Model):
status = models.ForeignKey(PortStatus, blank=True)
class ResultsForm(forms.ModelForm):
class Meta:
model = Result
extra_choice = forms.CharField(max_length=100, required=False)
def __init__(self, *args, **kwargs):
super(ResultsForm, self).__init__(*args, **kwargs)
self.save_new_status = False
def clean(self):
cleaned_data = self.cleaned_data()
status = cleaned_data.get('status')
extra_choice = cleaned_data.get('extra_choice')
if status and extra_choice:
raise forms.ValidationError("Can't specify both")
if not status and not extra_choice:
raise forms.ValidationError('Make a selection or add status')
if not status and extra_choice:
# make sure extra_choice isn't already in choices
if PostStatus.objects.filter(
status__iexact=extra_choice).count() > 0:
raise forms.ValidationError('Status present, etc')
else:
self.save_new_status = True
return cleaned_data
def save(self, commit=True)
instance = super(ResultsForm, self).save(commit=False)
if self.save_new_status:
new_status = PostStatus.objects.create(
status=self.cleaned_data.get('extra_choice'))
self.status = new_status
if commit:
instance.save()
return instance
Django表单在谈到什么是有效的选择时非常挑剔,这是一件好事。它可以防止恶意值通过操纵的POST数据传入。在
因此,您将无法以编程方式添加一个选项,一个是因为您的选择是一个硬编码的元组;第二个是因为您添加的任何选项都不会是运行
clean
时窗体认为有效的选项的一部分,因为初始化窗体时该值不存在。在相反,请考虑将这些选择作为一个单独的模型,并提供一个单独的表单字段来为该模型添加一个记录,这样下次就可以看到选择了。在
Django表单可以混合使用模型字段和非模型字段,因此使用它来添加输入字段。您还可以通过JavaScript控制额外字段的可见性。可能可以将所有这些组合到一个自定义字段的多个小部件中。在
相关问题 更多 >
编程相关推荐