gpt4 book ai didi

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

转载 作者:太空狗 更新时间:2023-10-29 18:33:14 27 4
gpt4 key购买 nike

我有一个 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>stackoverflow 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 输入实例,在页面呈现之后……我该怎么做呢?我是否必须使用 WTForms 生成的内容作为引用,自己在 JavaScript 中手动构建输入字段?这看起来很乱而且容易损坏,或者至少是难看的代码。

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

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

最佳答案

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

我知道这不完全是你问的。但是,如果您希望通过 Flask+ jQuery 来做到这一点。我将向您展示我通过 View 从 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>

这是调用此表单的 View :

@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()方法,我可以向这个 View 发送请求:

@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 View ,在那里进行任何你想要的处理,然后发回相同的 getJSON()并直接插入表格!瞧,你已经动态地使用 jQuery+Flask 来操纵你想做的任何事情,使用你最好的新 friend JSON。

经过大量研究,我发现最简单的方法就是通过 jQuery。学习getJSON()post() jQuery 中的方法,将其合并到您的 HTML 中,您就有了答案。还有jsonify()我相信您可以通过 import json 在 Python 中导入.

将列表提供给 jsonify()您可以使用它作为相同的 View 返回 getJSON()你曾经发送数据。

关于javascript - 从用户界面动态添加新的 WTForms FieldList 条目,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51817148/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com