如何在validate_on_submit()块之后填充WTForms FieldList?

Cho*_*key 6 forms fieldlist flask wtforms

关于如何使用WTForms的FieldList,确实缺乏文档.所以,多亏了互联网,我已经能够将以下内容整合在一起:

形成:

class BranchForm(Form):
    name = StringField('Name', validators = [Required()])
    equipment = FieldList(SelectField('Equipment', validators=[Required()], coerce=int,
        choices = [(x.id, x.name) for x in Equipment.query.all()]))
    mod = FieldList(StringField('Method of Delivery', validators = [Optional()]))
Run Code Online (Sandbox Code Playgroud)

视图:

def edit_branch(id):
    branch = Branch.query.filter_by(id=id).first()

    #populate data_in to be used by BranchForm
    data_in = []
    for eq_obj in branch.equipment_assoc:
        data_in.append(('equipment', eq_obj.equipment.id))
        data_in.append(('mod', eq_obj.mod))

    editform = BranchForm(data=MultiDict(data_in))

    if editform.validate_on_submit():
        branch.name = editform.name.data

        db.session.add(branch)
        db.session.commit()

        return redirect('/admin/branches/' + str(branch.id))

    editform.name.data = branch.name

    return render_template("branch_edit.html",
        title="Edit Branch",
        branch = branch,
        editform = editform)
Run Code Online (Sandbox Code Playgroud)

什么让我失望的是,在其他地方,我使用了WTForm表单并使用我的数据库中的数据填充了字段(比如编辑表单),我必须在form.validate_on_submit()块之后填充这些表单字段,因为如果没有,那么表格永远不会更新,因为提交的内容会立即被覆盖.

请参阅"editform.name.data = branch.name"(这是我一直以来的做法)

从我在网上找到关于填充FieldList的每个例子,它显然必须在实例化期间完成,但是必须在validate_on_submit()之前实例化表单,因为validate_on_submit()是表单对象的方法.

请参阅"editform = BranchForm(data = MultiDict(data_in))"(这是我在所见的所有示例中看到的填充FieldLists的方法.)

如何使用字段列表填充表单?

Cho*_*key 2

好吧,有朋友帮我解决了这个问题。这就是我最终得到的结果:

形式:

class BranchForm(Form):
    name = StringField('Name', validators = [Required()])
    equipment = FieldList(SelectField('Equipment', validators=[Required()], coerce=int,
        choices = [(x.id, x.name) for x in Equipment.query.all()]))
    mod = FieldList(StringField('Method of Delivery', validators = [Optional()]))

    def populate_assoc(self, branch_obj):
        i = 0
        branch_obj.name = self.name.data
        for assoc_obj in branch_obj.equipment_assoc:
            assoc_obj.equipment_id = self.equipment[i].data
            assoc_obj.mod = self.mod[i].data
            i += 1
Run Code Online (Sandbox Code Playgroud)

看法:

def edit_branch(id):
    branch = Branch.query.filter_by(id=id).first()

    if request.method == 'POST':
        editform = BranchForm()

        if editform.validate_on_submit():
            editform.populate_assoc(branch)

            db.session.add(branch)
            db.session.commit()

            return redirect('/admin/branches/' + str(branch.id))

    #populate data_in to be used 
    data_in = []
    for eq_obj in branch.equipment_assoc:
        data_in.append(('equipment', eq_obj.equipment.id))
        data_in.append(('mod', eq_obj.mod))

    editform = BranchForm(data=MultiDict(data_in))
    editform.name.data = branch.name

    return render_template("branch_edit.html",
        title="Edit Branch",
        branch = branch,
        editform = editform)
Run Code Online (Sandbox Code Playgroud)

诀窍实际上是放弃使用form.validate_on_submit()作为我的逻辑分隔符,因为它依赖于表单对象。他的想法是使用if request.method == 'POST':来达到此目的。这样我就可以用两种不同的方式实例化我的表单。一个会填充以供显示,另一个仅在请求方法为 POST 时才会实例化,从而保留在表单中提交的信息。

为了完成这项工作,我将 populate_assoc 方法添加到我的表单类中,以便我可以轻松地将表单中的信息放入我的关联模型中。