在 Django 表单中传递自定义表单值到视图

在Django中,我们可以通过表单的初始化参数initial来传递自定义的初始值给表单字段。如果我们想要在视图中设置表单的初始值,可以在视图中创建表单的实例时,传递一个字典给initial参数。

1、问题背景

我们遇到了这样一个问题:在使用 Django 表单时,我们希望将自定义表单中的值传递到视图中。然而,我们发现无法为多选选项的每个选项传递值。在渲染表单时,只有一个字符字段,而多选框中有多个选择。我们想知道是否有办法解决这个问题,以及表单集是否可以在这里提供帮助。我们对 Django 还很陌生,因此希望得到一些解释,以便更好地理解和学习。

python 复制代码
# models.py
class StateOption(models.Model):
    partstate = models.ForeignKey(State)
    partoption = models.ForeignKey(Option)
    relevantoutcome = models.ManyToManyField(Outcome, through='StateOptionOutcome')

class StateOptionOutcome(models.Model):
    stateoption = models.ForeignKey(StateOption)
    relevantoutcome = models.ForeignKey(Outcome)
    outcomevalue = models.CharField(max_length=20)

# forms.py
class UpdateStateOptionWithOutcomesForm(forms.ModelForm):
    class Meta:
        model = StateOption
        exclude = ['partstate', 'partoption']

    def __init__(self, *args, **kwargs):
        super(UpdateStateOptionWithOutcomesForm, self).__init__(*args, **kwargs)
        self.fields['relevantoutcome'] = forms.ModelMultipleChoiceField(queryset=Outcome.objects.all(), required=True, widget=forms.CheckboxSelectMultiple)
        self.fields['outcomevalue'] = forms.CharField(widget=forms.TextInput(attrs={'size': '30'}))

# views.py
stateoption = get_object_or_404(StateOption, pk=stateoption_id)

if request.method == "POST":
    form = UpdateStateOptionWithOutcomesForm(request.POST, instance=stateoption)
    if form.is_valid():
        cd = form.cleaned_data
        outcomevalue = cd['outcomevalue']

        for outcome_id in request.POST.getlist('relevantoutcome'):
            stateoption_outcome = StateOptionOutcome.objects.create(stateoption=stateoption, relevantoutcome_id=int(outcome_id), outcomevalue=outcomevalue) 

# template.html
{% for field in form %}
    {{ field.label }}:
    {{ field }}
    {% if field.errors %}
        {{ field.errors|striptags }}
    {% endif %}
{% endfor %}

2、解决方案

方法一:生成所需数量的字段

一种解决方案是编写一个循环来生成所需数量的字段。例如:

python 复制代码
# forms.py
# ...

outcome_qs = Outcome.objects.all()
self.fields['relevantoutcome'] = forms.ModelMultipleChoiceField(queryset=outcome_qs, required=True, widget=forms.CheckboxSelectMultiple)
for outcome in outcome_qs:
    # Use Outcome primary key to easily match two fields in your view.
    self.fields['outcomevalue_%s' % outcome.pk] = forms.CharField(widget=forms.TextInput(attrs={'size':'30'}) 

方法二:使用表单集

另一种解决方案是使用表单集。表单集允许我们创建一组类似的表单,每个表单都可以处理单个对象。在我们的例子中,我们可以创建一个表单集来处理每个 StateOptionOutcome 对象。

python 复制代码
# forms.py
class StateOptionOutcomeForm(forms.ModelForm):
    class Meta:
        model = StateOptionOutcome
        fields = ['relevantoutcome', 'outcomevalue']

StateOptionOutcomeFormSet = forms.formset_factory(StateOptionOutcomeForm, extra=1)

# views.py
stateoption = get_object_or_404(StateOption, pk=stateoption_id)

if request.method == "POST":
    formset = StateOptionOutcomeFormSet(request.POST)
    if formset.is_valid():
        for form in formset:
            stateoption_outcome = StateOptionOutcome.objects.create(stateoption=stateoption, **form.cleaned_data)

使用表单集的好处是,我们可以轻松地处理多个对象,而且代码也更加简洁。

相关推荐
沸材29 分钟前
Redis——实现消息队列
数据库·redis·消息队列
しかし11811439 分钟前
C语言队列的实现
c语言·开发语言·数据结构·数据库·经验分享·链表
⁤⁢初遇1 小时前
MySQL---数据库基础
数据库
wolf犭良1 小时前
27、Python 数据库操作入门(SQLite)从基础到实战精讲
数据库·python·sqlite
画扇落汗1 小时前
Python 几种将数据插入到数据库的方法(单行插入、批量插入,SQL Server、MySQL,insert into)
数据库·python·sql·mysql
银河系的一束光1 小时前
mysql的下载和安装2025.4.8
数据库·mysql
Full Stack Developme2 小时前
SQL 查询中使用 IN 导致性能问题的解决方法
数据库·sql
神经星星2 小时前
【vLLM 学习】API 客户端
数据库·人工智能·机器学习
小光学长3 小时前
基于flask+vue框架的助贫公益募捐管理系统1i6pi(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库
XiaoLeisj3 小时前
【图书管理系统】深入解析基于 MyBatis 数据持久化操作:全栈开发图书管理系统:查询图书属性接口(注解实现)、修改图书属性接口(XML 实现)
xml·java·数据库·spring boot·sql·java-ee·mybatis