Ham*_*eem 18 python flask flask-sqlalchemy flask-wtforms
我对于flask框架相当新,并且正在为webportal创建一个编辑配置文件页面.我陷入困境,无法自动填写表格.
这是我的表单类:
class EditProfile(Form):
username = TextField('Username', [Required()])
email = TextField('Email', [Required()])
about = TextAreaField('About', [Required()])
website = TextField('Website', [Required()])
Run Code Online (Sandbox Code Playgroud)
这是我评估表单的函数.
def editprofile(nickname = None):
if g.fas_user['username'] == nickname or request.method == 'POST':
form = EditProfile()
form_action = url_for('profile.editprofile')
if request.method == 'POST' and form.validate():
if form.username.data == nickname :
query = EditProfile(form.username.data,
form.email.data,
form.about.data,
form.website.data,
)
print query #debug
db.session.add(query)
db.session.commit()
flash('User Updated')
print "added"
return(url_for('profile.editprofile'))
return render_template('profile/add.html', form=form,
form_action=form_action, title="Update Profile")
else:
return "Unauthorised"
Run Code Online (Sandbox Code Playgroud)
我的表单形式的html模板是:
{% extends "base.html" %}
{% block title %}
{{ title }}
{% endblock %}
{% block content %}
{% from "_formhelpers.html" import render_field %}
<div id="Edit Profile">
<h2>{{ title }}</h2>
<form method="post" action="{{ form_action }}">
<fieldset>
<legend></legend>
{{ render_field(form.username) }}
{{ render_field(form.email)}}
{{ render_field(form.about )}}
{{ render_field(form.website) }}
</fieldset>
<input type="submit" class="button" value="Save"/>
</form>
</div>
{% endblock %}
Run Code Online (Sandbox Code Playgroud)
我有一个用户类的对象.从该对象我想预先填写此表单.如何预填充表单中的值.我想在这里实现编辑配置文件功能.
dir*_*irn 20
您需要在创建表单时将对象传递给表单.
form = EditProfile(obj=user) # or whatever your object is called
Run Code Online (Sandbox Code Playgroud)
你会遇到一些麻烦
query = EditProfile(form.username.data,
form.email.data,
form.about.data,
form.website.data,
)
db.session.add(query)
Run Code Online (Sandbox Code Playgroud)
它会创建EditProfile表单的新实例.然后,您尝试将其添加到会话中.会话需要模型,而不是表单.
相反,在验证表单后,您可以将其值与对象相关联.
form.populate_obj(user) # or whatever your object is called
Run Code Online (Sandbox Code Playgroud)
由于您的对象已加载,因此您无需将其添加到会话中.你可以删除db.session.add(query)并只是打电话db.session.commit().
我发现这样做的最简单方法是在get请求中填写表单字段.
@decorator_authorized_user # This decorator should make sure the user is authorized, like @login_required from flask-login
def editprofile(nickname = None):
# Prepare the form and user
form = EditProfile()
form_action = url_for('profile.editprofile')
my_user = Users.get(...) # get your user object or whatever you need
if request.method == 'GET':
form.username.data = my_user.username
form.email.data = my_user.email
# and on
if form.validate_on_submit():
# This section needs to be reworked.
# You'll want to take the user object and set the appropriate attributes
# to the appropriate values from the form.
if form.username.data == nickname:
query = EditProfile(form.username.data,
form.email.data,
form.about.data,
form.website.data,
)
print query #debug
db.session.add(query)
db.session.commit()
flash('User Updated')
print "added"
return(url_for('profile.editprofile'))
return render_template('profile/add.html', form=form,
form_action=form_action, title="Update Profile")
Run Code Online (Sandbox Code Playgroud)
这将设置函数以在get请求中返回预填充表单.你必须重做下面的部分form.validate_on_submit.Dirn的回答表明了一些正确的事情要做.
小智 6
要使用SQLAlchemy对象填充表单,请使用:
form = EditProfile(obj=<SQLAlchemy_object>)
Run Code Online (Sandbox Code Playgroud)
如果您的表单字段由于某种原因(它们应该)与模型中的数据库列不匹配,那么表单类将采用kwargs:
**kwargs - 如果formdata或obj都不包含字段的值,则表单将为字段分配匹配关键字参数的值(如果提供).
我发现一个用例就是如果你有一个包含引用多个模型的字段的表单(例如通过关系); 通过一个obj是不够的.
因此,让我们以您的示例为例,说明您在数据库模型(for user)中使用site而不是website(表单字段名称).您这样做是为了从SQLAlchemy对象填充您的表单:
form = EditProfile(obj=user, website=user.site)
Run Code Online (Sandbox Code Playgroud)
然后在POST中你必须这样做以从你的表单填充你的SQLAchemy对象:
form.populate_obj(user)
user.site = form.website.data
db.session.commit()
Run Code Online (Sandbox Code Playgroud)