从用户界面动态添加新的 WTForms FieldList 条目

Dynamically add new WTForms FieldList entries from user interface

我有一个 flask + wtforms 应用程序,我希望用户能够在其中输入父对象和任意数量的子对象。我不确定从用户界面动态创建新子表单输入字段的最佳方式是什么。

到目前为止我得到了什么

下面是一个完整的工作示例。 (注意:这是一个人工示例,用于突出显示单个 .py 文件中的所有工作部分,这会导致一些非常混乱的代码。抱歉。)

from flask import Flask, render_template_string
from flask_wtf import FlaskForm
from wtforms import FieldList, FormField, StringField, SubmitField
from wtforms.validators import InputRequired


# Here I'm defining a parent form, AuthorForm, and a child form, BookForm.
# I'm using the FieldList and FormField features of WTForms to allow for multiple
# nested child forms (BookForms) to be attached to the parent (AuthorForm).
class BookForm(FlaskForm):
    title = StringField('title', validators=[InputRequired()])
    genre = StringField('genre', validators=[InputRequired()])

# I'm defining a min_entry of 1 so that new forms contain a blank BookForm entry
class AuthorForm(FlaskForm):
    name = StringField('name', validators=[InputRequired()])
    books = FieldList(FormField(BookForm), min_entries=1)
    submit = SubmitField('Save')


# Here's my Jinja template
html_template_string = """
<html>
    <head><title>Whosebug example</title></head>
    <body>
        <form action="" method="post" role="form">
            {{ form.hidden_tag() }}
            {{ form.name.label }} {{ form.name() }}
            {% for book in form.books %}
                <p>
                {{ book.title.label }} {{ book.title() }}
                {{ book.genre.label }} {{ book.genre() }}
                {{ book.hidden_tag() }}
                </p>
            {% endfor %}
            {{ form.submit() }}
        </form>
    </body>
</html>
"""

# Alright, let's start the app and try out the forms
app = Flask(__name__)
app.config['SECRET_KEY'] = 'secret'

@app.route('/', methods=['GET', 'POST'])
def index():
    form = AuthorForm()
    if form.validate_on_submit():
        for book in form.books.data:
            print(book)
    return render_template_string(html_template_string, form=form)

if __name__ == '__main__':
    app.run()

我卡在哪里

我知道如何在服务器端创建新的子条目(BookForm 条目)。我可以将空字典传递给我的表单,WTForms 会为我生成输入:

...
form = AuthorForm()
form.books.append_child({})
form.books.append_child({})
form.books.append_child({})
...

Bam,现在我的页面有三本书的输入字段,我可以预先填充它们的数据。 WTForms 在呈现页面之前生成表单输入内容。它计算出每个输入所需的所有 ID,等等。

如果我希望用户能够单击一个按钮并从用户端添加新的 BookForm 输入实例,在页面呈现后……我该怎么做?我是否必须自己在 JavaScript 中手动构建输入字段,使用 WTForms 生成的内容作为参考?这看起来很乱而且容易损坏,或者至少是丑陋的代码。

WTForms 是否有办法根据需要为新输入呈现 HTML,管理输入标签 ID 等的唯一性?我可以 post 返回服务器以将空白条目附加到表单并重新呈现页面,但这会使我失去所有现有的用户输入,因此它实际上不起作用。 Whosebug问题在评论中提出了同样的事情,并提出了同样的反对意见。

如果这必须以丑陋的手动方式完成,那么我可以做到。当有更好的(甚至可能是官方的)解决方案时,我只是不想采取糟糕的方法。

我完全理解你的问题,使用纯 Flask 来实现这个。 但几个月前我遇到过同样的情况。经过大量的研究和奇怪而多余的方法,我发现,实施它不仅是艰苦的,而且是漫长而艰难的,任何小的改变或调试工作都会破坏整个事情。

我知道这不是你问的。但是,如果您希望通过 Flask+ jQuery 来做到这一点。我将向您展示我通过视图从 HTML 前端动态加载数据所做的一些工作,希望这对您有所帮助,如果您改变主意并决定合并一些 jQuery .

这是我的表格,使用 HTML 和 Jinja2:

<form method="POST" action="">
            {{ form.hidden_tag() }}
            <fieldset class="form-group">
                <legend class="border-bottom mb-4">XYZ</legend>

                <div>
                    {{ form.standard.label(class="form-control-label") }}
                    {{ form.standard(class="form-control form-control-lg") }}
                </div>

                <div>
                    {{ form.wps.label(class="form-control-label") }}
                    {{ form.wps(class="form-control form-control-lg") }}
                </div>
            ... 
            </fieldset>
            <div class="form-group">
                {{ form.submit(class='btn btn-outline-success') }}
            </div>

        </form>

这是调用此表单的视图:

@app.route("/newqualification/<welder_id>", methods=['GET', 'POST'])
def newqualification(welder_id=None):
    form = #passformhere

        #writemethod.

    return render_template('xyz.html', title='xyz', form=form)

这是我编写的有用的 jQuery 脚本,您可以通过 <script> 标记将其包含在 HTML 中。

<script>
 $(document).ready(function(){

            $("#standard").change(function(){
                var std = $(this).find('option:selected').val(); //capture value from form.

                $.getJSON("{{url_for('modifywps')}}",{'std':std}, function(result){ //use captured value to sent to view via a dictionary {'key':val}, via url_for('view_name')

                    $.each(result, function(i, field){ //value you want will be back here and gone through..
                        $.each(field, function(j,k){
                            $("#wps").append('<option value="'+j+'">'+k+'</option>'); // use jQuery to insert it back into the form, via the form name you gave in jinja templating
                        });
                    });
                });
            });


            $("#wps").change(function(){

                    var std = $('#wps').find('option:selected').val(); // capture value from form.
                    $.getJSON("{{url_for('modifyprocess')}}",{'std':std}, function(result)

                        $.each(result, function(i, field){
                            $.each(field, function(j,k){
                                $("#processes").append('<option value="'+j+'">'+k+'</option>');
                            });
                        });
                    });
            });
</script>

使用 jQuery 和 getJSON() 方法,我可以向此视图发送请求:

@app.route("/modifywps", methods=['POST', 'GET'])
def modifywps():
    application_standard_id = request.args.get('std') # gets value from the getJson()

   # process it however you want.. 

    return #whatever you want.

所以每次发生变化时,数据都会从站点动态读取,发送到 Flask 视图,在那里进行任何你想要的处理,发回相同的 getJSON() 并直接插入到表单中!! 瞧,你已经动态地使用 jQuery+Flask 来操纵你想做的任何事情,使用你最好的新朋友 JSON。

经过大量研究,我发现最简单的方法就是通过 jQuery 来完成上述操作。 学习 jQuery 中的 getJSON()post() 方法,将其融入您的 HTML 中,您就会得到答案。 还有 jsonify(),我相信你可以通过 import json.

导入 Python

jsonify() 提供一个列表,您可以将其用于 return 作为用于发送数据的同一 getJSON() 中的视图。