Flask-WTF SelectMultipleField 使用数据库 ID 而不是名称填充
Flask-WTF SelectMultipleField populates with database IDs instead of names
我想先声明英语不是我的母语,如果我的任何解释含糊不清或没有意义,请告诉我,我会尽力他们更清楚。
我在使用 Flask-WTF 中的正确数据填充 SelectMultipleField
时遇到问题。从下面的代码中可以看出,我将用户对象传递给了表单。这可以完美地工作,因为它会在呈现页面时填充所有字段。我 运行 遇到的问题是 SelectMultipleField
填充了角色 ID 而不是角色名称。
在表单部分,我生成了一个列表,choices
,其中包含应显示的所有可用选项。我可以将其反转为 choices.append((role.id, role.name))
,它将使用名称而不是 ID 填充该字段。但是,如果我这样做,分配给用户对象(来自数据库)的角色将不会显示为预选。
我怎样才能使该字段预先填充名称而不是 ID,并保持预选分配的角色?
任何正确方向的指导或推动将不胜感激。如果缺少任何信息,我不想添加,请告诉我,我会添加。
# Models
class Role(db.Document, RoleMixin):
meta = {'collection': 'role'}
name = db.StringField(max_length=80, unique=True)
description = db.StringField(max_length=255)
def __repr__(self):
return '<Role {}>'.format(self.name)
def __str__(self):
return self.name
class User(db.Document, UserMixin):
meta = {'collection': 'User'}
first_name = db.StringField(max_length=30)
last_name = db.StringField(max_length=30)
email = db.StringField(max_length=120)
password_hash = db.StringField(max_length=255)
active = db.BooleanField(default=True)
roles = db.ListField(db.ReferenceField(Role), default=[])
def __repr__(self):
return '<User {}>'.format(self.email)
def __str__(self):
return self.email
# Forms
class EditUserForm(FlaskForm):
choices = []
for role in Role.objects:
choices.append((role.name, role.id))
first_name = StringField('First name', validators=[DataRequired()])
last_name = StringField('Last name', validators=[DataRequired()])
email = StringField('Email', validators=[DataRequired(), Email()])
password = PasswordField('Password')
password2 = PasswordField('Repeat password', validators=[EqualTo('password')])
roles = SelectMultipleField('User roles', choices=choices)
submit = SubmitField('Save')
# Routes
@app.route('/users/edit/<id>', methods=['GET', 'POST'])
@roles_required('admin')
def edit_user(id):
user = User.objects(id=id).first()
form = EditUserForm(obj=user)
if form.validate_on_submit() and request.method == 'POST':
user.first_name=form.first_name.data
user.last_name=form.last_name.data
user.email=form.email.data
user.roles = []
for role in form.roles.data:
r = Role.objects(name=role).first()
user.set_role(r.id)
if form.password.data:
user.set_password(form.password.data)
user.save()
return redirect(url_for('users'))
return render_template('edit_user.html', title='Edit user', user=user, form=form)
模板:
{% extends "layout.html" %}
{% set active_page = "edit_user" %}
{% block jumbotron %}
<h1>{{ title }}</h1>
<form action="" method="post">
{{ form.hidden_tag() }}
<p>
{{ form.first_name.label }}<br>
{{ form.first_name(size=32) }}<br>
{% for error in form.first_name.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.last_name.label }}<br>
{{ form.last_name(size=32) }}<br>
{% for error in form.last_name.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.email.label }}<br>
{{ form.email(size=32) }}<br>
{% for error in form.email.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.password.label }}<br>
{{ form.password(size=32) }}<br>
{% for error in form.password.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.password2.label }}<br>
{{ form.password2(size=32) }}<br>
{% for error in form.password2.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.roles.label }}<br>
{{ form.roles(size=32) }}<br>
{% for error in form.roles.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>{{ form.submit() }}</p>
</form>
{% endblock %}
我试图用下面的代码创建我自己的对象来传递给表单,但是我没有成功。将该对象传递给表单时,未填充任何字段。
user = User.objects(id=id).first()
temps = []
u = user.to_mongo()
u['_id'] = {'$oid': str(u['_id'])}
temp_roles = []
for role in user.roles:
r = role.to_mongo()
r['_id'] = {'$oid': str(r['_id'])}
temp_roles.append(r)
u['roles'] = temp_roles
我觉得 PRMoureu 是对的。
你可以choices.append((role.name, role.name))
仅供参考。
SelectField 的选项是元组列表 (value, label) 对,请转到此link 了解更多详情。
Select fields keep a choices property which is a sequence of (value, label) pairs. The value portion can be any type in theory, but as form data is sent by the browser as strings, you will need to provide a function which can coerce the string representation back to a comparable object.
默认的coerce是unicode,如果你使用(id,name)int .
我想先声明英语不是我的母语,如果我的任何解释含糊不清或没有意义,请告诉我,我会尽力他们更清楚。
我在使用 Flask-WTF 中的正确数据填充 SelectMultipleField
时遇到问题。从下面的代码中可以看出,我将用户对象传递给了表单。这可以完美地工作,因为它会在呈现页面时填充所有字段。我 运行 遇到的问题是 SelectMultipleField
填充了角色 ID 而不是角色名称。
在表单部分,我生成了一个列表,choices
,其中包含应显示的所有可用选项。我可以将其反转为 choices.append((role.id, role.name))
,它将使用名称而不是 ID 填充该字段。但是,如果我这样做,分配给用户对象(来自数据库)的角色将不会显示为预选。
我怎样才能使该字段预先填充名称而不是 ID,并保持预选分配的角色?
任何正确方向的指导或推动将不胜感激。如果缺少任何信息,我不想添加,请告诉我,我会添加。
# Models
class Role(db.Document, RoleMixin):
meta = {'collection': 'role'}
name = db.StringField(max_length=80, unique=True)
description = db.StringField(max_length=255)
def __repr__(self):
return '<Role {}>'.format(self.name)
def __str__(self):
return self.name
class User(db.Document, UserMixin):
meta = {'collection': 'User'}
first_name = db.StringField(max_length=30)
last_name = db.StringField(max_length=30)
email = db.StringField(max_length=120)
password_hash = db.StringField(max_length=255)
active = db.BooleanField(default=True)
roles = db.ListField(db.ReferenceField(Role), default=[])
def __repr__(self):
return '<User {}>'.format(self.email)
def __str__(self):
return self.email
# Forms
class EditUserForm(FlaskForm):
choices = []
for role in Role.objects:
choices.append((role.name, role.id))
first_name = StringField('First name', validators=[DataRequired()])
last_name = StringField('Last name', validators=[DataRequired()])
email = StringField('Email', validators=[DataRequired(), Email()])
password = PasswordField('Password')
password2 = PasswordField('Repeat password', validators=[EqualTo('password')])
roles = SelectMultipleField('User roles', choices=choices)
submit = SubmitField('Save')
# Routes
@app.route('/users/edit/<id>', methods=['GET', 'POST'])
@roles_required('admin')
def edit_user(id):
user = User.objects(id=id).first()
form = EditUserForm(obj=user)
if form.validate_on_submit() and request.method == 'POST':
user.first_name=form.first_name.data
user.last_name=form.last_name.data
user.email=form.email.data
user.roles = []
for role in form.roles.data:
r = Role.objects(name=role).first()
user.set_role(r.id)
if form.password.data:
user.set_password(form.password.data)
user.save()
return redirect(url_for('users'))
return render_template('edit_user.html', title='Edit user', user=user, form=form)
模板:
{% extends "layout.html" %}
{% set active_page = "edit_user" %}
{% block jumbotron %}
<h1>{{ title }}</h1>
<form action="" method="post">
{{ form.hidden_tag() }}
<p>
{{ form.first_name.label }}<br>
{{ form.first_name(size=32) }}<br>
{% for error in form.first_name.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.last_name.label }}<br>
{{ form.last_name(size=32) }}<br>
{% for error in form.last_name.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.email.label }}<br>
{{ form.email(size=32) }}<br>
{% for error in form.email.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.password.label }}<br>
{{ form.password(size=32) }}<br>
{% for error in form.password.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.password2.label }}<br>
{{ form.password2(size=32) }}<br>
{% for error in form.password2.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>
{{ form.roles.label }}<br>
{{ form.roles(size=32) }}<br>
{% for error in form.roles.errors %}
<span style="color: red;">[{{ error }}]</span>
{% endfor %}
</p>
<p>{{ form.submit() }}</p>
</form>
{% endblock %}
我试图用下面的代码创建我自己的对象来传递给表单,但是我没有成功。将该对象传递给表单时,未填充任何字段。
user = User.objects(id=id).first()
temps = []
u = user.to_mongo()
u['_id'] = {'$oid': str(u['_id'])}
temp_roles = []
for role in user.roles:
r = role.to_mongo()
r['_id'] = {'$oid': str(r['_id'])}
temp_roles.append(r)
u['roles'] = temp_roles
我觉得 PRMoureu 是对的。
你可以choices.append((role.name, role.name))
仅供参考。
SelectField 的选项是元组列表 (value, label) 对,请转到此link 了解更多详情。
Select fields keep a choices property which is a sequence of (value, label) pairs. The value portion can be any type in theory, but as form data is sent by the browser as strings, you will need to provide a function which can coerce the string representation back to a comparable object.
默认的coerce是unicode,如果你使用(id,name)int .