我正在使用Flask,Jinja,WTForms,SqlAlchemy等为学习语言创建类似测验的Web应用程序。一旦用户通过成功浏览存储在JSON文件中的所有级别完成这样的语言课程,我希望该应用程序为他提供练习模式,用户将回答随机选择的级别。
当我运行应用程序时,我可以看到随机级别生成的单选按钮,但是当我选择任何答案并提交时, form.validate_on_submit()返回False 和表单.errors返回 {' practiceform':[u'不是有效的选择']} 。当我将值硬编码为currentLevel变量时,它可以正常工作。
views.py
@user_blueprint.route('/courses/<course>/quiz/practice',methods=['GET','POST'])
@login_required
def practice(course):
courseClass = class_for_name("project.models", course.capitalize())
courses = courseClass.query.filter_by(email=current_user.email).first()
maxLevel = courseClass.query.filter_by(email=current_user.email).first().get_maxLevel()
currentLevel = randint(0, maxLevel-1) # If this value is hard-coded or fetched from db, it works correctly
dic = generateQuestion(course, currentLevel)
display = dic["display"]
correct = dic["correct"]
options = dic["options"]
form = PracticeForm(request.form)
form.practiceform.choices = [(option, option) for option in options]
if form.validate_on_submit():
practiceForm = form.practiceform.data
if ((practiceForm == correct) and courses):
# Do something
flash("Nice job", 'success')
return redirect(url_for('user.practice', course=course))
else:
# Do something else
flash("Wrong answer", 'danger')
return redirect(url_for('user.practice', course=course))
return render_template('courses/practice.html', form=form, display=display)
forms.py
class PracticeForm(Form):
practiceform = RadioField('practice')
practice.html
{% extends "_base.html" %}
{% block content %}
<form action='' method='POST' role='form'>
<p>
<!-- Tried put form.csrf, form.csrf_token, form.hidden_tag() here -->
{{ form.practiceform() }}
</p>
<input type="submit" value="submit" />
</form>
{% endblock %}
那我在那里错过了什么?是什么区别让我们说硬编码25级,哪个正常工作,或者数字25是在randint中随机生成的?
答案 0 :(得分:2)
我的猜测是option
是int
,WT表格会从str
获得request.form
。
当数据从请求返回时,除非您使用string
构造函数的coerce
kwarg明确指定类型,否则它将被WTForms视为wtforms.fields.*Field
:
practiceform = RadioField('practice', coerce=int)
答案 1 :(得分:0)
所以我发现randint()导致了这个问题,因为在GET和POST动作上调用了练习(课程)方法,这导致了两个不同的整数 - &gt;大多数时候有两种不同的形式。所以我重构了代码。保留了GET动作的练习(课程)方法,并创建了一个处理POST动作的新方法,这解决了这个问题。