flask 利用flask_wtf扩展 创建web表单
在Flask中,为了处理web表单,我们一般使用Flask-WTF扩展,它封装了WTForms,并且它有验证表单数据的功能
- 创建语句格式: startTime = DateTimeField('计划开始时间', validators=[DataRequired('不可为空'),EqualTo('nowTime',message='两次事件必须一致')],format='%Y/%m/%d %H:%M:%S' render_kw={'placeholder':'2018-06-03 00:00:00', 'id': 'rz1','autocomplete': "off"}
- html效果:
<label>计划开始时间</label>
<input autocomplete="off" class="form-control" id="rz1" name="startTime" placeholder="2018-06-03 00:00:00" required="" type="text" value="">
3. 页面展示效果:

解析:
- DataTimeField是从wtforms包导入的字段类 from wtfms import SelectField, StringField, SubmitField, TextField, DateTimeField, FileField。。。
- '计划开始时间'是input标签的lable
- validators是验证器,验证输入值的合法性,以列表的形式封装各种验证函数,验证函数由wtforms.validators导入:from wtforms.validators import DataRequired
- format指定展示格式
- render_kw以字典的形式存放input标签的各种属性
form.py文件在项目中的位置:与逻辑层同级,即与view.py同级,以便于不同的功能块放不同的form表单
项目效果: 三个文件依次是创建蓝图 创建form表单 逻辑(视图)函数

创建表单部分代码案例:
from flask_wtf import Form
from wtforms import SelectField, StringField, SubmitField, TextField, DateTimeField, FileField, SelectMultipleField, TextAreaField, HiddenField
from wtforms.validators import DataRequired class TaskinfoAdd(Form):
taskName = StringField('任务名称:', [DataRequired('任务名称不能为空')], render_kw={'placeholder': '任务名称', 'autocomplete': 'off'})
taskInfo = TextAreaField('详情描述:', render_kw={'placeholder': '任务描述...', 'autocomplete': 'off'})
urgentId = SelectField('紧急程度:', validators=[DataRequired('请选择紧急程度')], coerce=int, render_kw={'style': 'width:110px'})
resPeople = StringField('责任人:', [DataRequired('责任人不能为空')], render_kw={'class': 'res-people', 'placeholder': '责任人', 'style': 'width:110px', 'data-toggle':"modal", 'data-target':"#myModal2"})
exePeople = StringField('参与人(多选):', render_kw={'class': 'exe-people', 'placeholder': '参与人(多选)', 'style': 'width:300px'})
sharePeople = StringField('共享人(多选):', render_kw={'placeholder': '共享人(多选)', 'style': 'width:300px'})
startTime = DateTimeField('计划开始时间:', [DataRequired('开始时间不能为空')], format='%Y-%m-%d %H:%M:%S', render_kw={'placeholder':'2018-06-03 00:00:00', 'id': 'rz1', 'class': 'form-control calendar-control', 'autocomplete': "off"})
endTime = DateTimeField('计划完成时间:', [DataRequired('完成时间不能为空')], format='%Y-%m-%d %H:%M:%S', render_kw={'placeholder':'2019-06-06 00:00:00', 'id': 'rz2', 'class': 'form-control calendar-control', 'autocomplete': 'off'})
workTime = StringField('计划工时(分钟):', [DataRequired('工时不能为空')], render_kw={'placeholder': ''}) loadFile = FileField('附件', render_kw={'multiple': 'multiple'})
submit = SubmitField('保存') class TaskinfoEdit(Form):
taskName = StringField('任务名称:', [DataRequired('任务名称不能为空')])
taskInfo = TextAreaField('详情描述:', render_kw={'placeholder': '任务描述...', 'autocomplete': 'off'})
urgentId = SelectField('紧急程度:', validators=[DataRequired('不可为空')], coerce=int, render_kw={'style': 'width:135px; display:inline-block'})
resPeople = StringField('责任人:', [DataRequired('责任人不能为空')], render_kw={'style': 'width:150px; display:inline-block'})
advanceId = SelectField('进 度:', coerce=int, render_kw={'placehoder': '50%', 'style': 'width:153px; display:inline-block'})
workTime = StringField('计划工时(分钟):', [DataRequired('工时不能为空')], render_kw={'style': 'width:100px; display: inline-block'})
exePeople = StringField('参与人:', render_kw={'style': 'width:220px; display: inline-block'})
sharePeopleTo = StringField('共享人:', render_kw={'style': 'width:220px; display: inline-block'}) loadFile = FileField('上传附件:', render_kw={'multiple': 'multiple'})
download = StringField('下载附件:', render_kw={'placehoder': '无附件', 'style': 'background: white; height:23px; display: inline-block'}, id='download')
reason = StringField('超期原因:', render_kw={'placeholder': '超期原因...', 'autocomplete': 'off'})
submit = SubmitField('提交')
form.py
flask接收后端处理:
@task_mgm.route('/taskinfo_add', methods=['GET', 'POST'])
@login_required
def taskinfo_add_fun():
form1 = TaskinfoAdd()
form1.urgentId.choices = [(urgences.id, urgences.urgentName) for urgences in Urgence.query.all()]
if form1.validate_on_submit():
task = MgmTask(taskName=form1.taskName.data, urgentId=form1.urgentId.data, resPeople=form1.resPeople.data,
exePeople=form1.exePeople.data, sharePeopleTo=form1.sharePeople.data, startTime=form1.startTime.data,
endTime=form1.endTime.data, workTime=form1.workTime.data, taskInfo=form1.taskInfo.data,
createPeople=current_user.name)
# 替换汉语逗号
task.sharePeopleTo.replace(',', ',')
task.resPeople.replace(',', ',')
task.exePeople.replace(',', ',')
# 保存上传的文档
upload(form1, task)
try:
db.session.add(task)
db.session.commit()
except Exception as e:
logging.error(e)
return redirect(url_for('task_mgm.taskinfo_add_fun'))
return render_template('/task_mgm/taskinfo_mine.html', form1=form1, name=current_user.name, grade=current_user.grade)
# 编辑任务
@task_mgm.route('/taskinfo_editID=<int:num>', methods=['GET', 'POST'])
@login_required
def taskinfo_edit_fun(num):
task = MgmTask.query.get(num)
pageType = request.values.get('pageType')
formEdit = TaskinfoEdit(obj=task)
formEdit.urgentId.choices = [(urgences.id, urgences.urgentName) for urgences in Urgence.query.all()]
formEdit.advanceId.choices = [(advances.id, advances.name) for advances in Advance.query.all()]
if task.fileName:
formEdit.download.data = task.fileName
if formEdit.validate_on_submit():
formEdit.populate_obj(task) # 重新填充表单
if formEdit.advanceId.data == 5:
task.finishTime = datetime.now()
upload(formEdit, task)
try:
db.session.add(task)
db.session.commit()
except Exception as e:
logging.error(e)
page = request.args.get('page', 1, type=int)
pagination = Comment.query.filter(Comment.taskId == num).order_by(Comment.createTime.desc()).paginate(
page, per_page=225, error_out=False)
comments = pagination.items
return render_template('/task_mgm/taskinfo_edit.html', formEdit=formEdit, header='编辑任务', comments=comments, pagination=pagination, pageType=pageType, name=current_user.name, grade=current_user.grade)
view.py
html前端可以继承bootstrap的wtf.html 也可以直接写:
- 继承:
{% import "bootstrap/wtf.html" as wtf %}
<div class="modal-body" style="margin-top: 20px">{{ wtf.quick_form(form1, form_type='horizontal', horizontal_columns=('md', 3, 8)) }}
<button type="button" class="btn btn-default" data-dismiss="modal"
style="position:relative; margin-left: 460px; bottom:49px;">关闭
</button>
</div>
- 不继承
<form method="post">
{#设置csrf_token#}
{{ form.csrf_token() }}
{{ form.username.label }}{{ form.username }}<br>
{{ form.password.label }}{{ form.password }}<br>
{{ form.password2.label }}{{ form.password2 }}<br>
...
{{ form.input }}<br>
</form>
WTForms支持的HTML标准字段:
| 字段对象 | 说明 |
|---|---|
| StringField | 文本字段 |
| TextAreaField | 多行文本字段 |
| PasswordField | 密码文本字段 |
| HiddenField | 隐藏文件字段 |
| DateField | 文本字段,值为 datetime.date 文本格式 |
| DateTimeField | 文本字段,值为 datetime.datetime 文本格式 |
| IntegerField | 文本字段,值为整数 |
| DecimalField | 文本字段,值为decimal.Decimal |
| FloatField | 文本字段,值为浮点数 |
| BooleanField | 复选框,值为True 和 False |
| RadioField | 一组单选框 |
| SelectField | 下拉列表 |
| SelectMutipleField | 下拉列表,可选择多个值 |
| FileField | 文件上传字段 |
| SubmitField | 表单提交按钮 |
| FormField | 把表单作为字段嵌入另一个表单 |
| FieldList | 一组指定类型的字段 |
WTForms常用验证函数:
| 验证函数 | 说明 |
|---|---|
| DataRequired | 确保字段中有数据 |
| EqualTo | 比较两个字段的值,常用于比较两次密码输入 |
| Length | 验证输入的字符串长度 |
| NumberRange | 验证输入的值在数字范围内 |
| URL | 验证URL |
| AnyOf | 验证输入值在可选列表中 |
| NoneOf | 验证输入值不在可选列表中 |
提示:
使用Flask-WTF需要配置参数SECRET_KEY。
CSRF_ENABLED是为了CSRF(跨站请求伪造)保护。 SECRET_KEY用来生成加密令牌,当CSRF激活的时候,该设置会根据设置的密匙生成加密令牌
flask 利用flask_wtf扩展 创建web表单的更多相关文章
- Flask之模板web表单
3.3 Web表单: web表单是web应用程序的基本功能. 它是HTML页面中负责数据采集的部件.表单有三个部分组成:表单标签.表单域.表单按钮.表单允许用户输入数据,负责HTML页面数据采集,通过 ...
- Flask 教程 第三章:Web表单
本文翻译自 The Flask Mega-Tutorial Part III: Web Forms 这是Flask Mega-Tutorial系列的第三部分,我将告诉你如何使用Web表单. 在第二章中 ...
- Flask学习 三 web表单
web表单 pip install flask-wtf 实现csrf保护 app.config['SECRET_KEY']='hard to guess string' # 可以用来存储框架,扩展,程 ...
- Flask开发系列之Web表单
Flask开发系列之Web表单 简单示例 from flask import Flask, request, render_template app = Flask(__name__) @app.ro ...
- Flask学习之三 web表单
本部分Miguel Grinberg教程的翻译地址:http://www.pythondoc.com/flask-mega-tutorial/webforms.html 开源中国的:http://ww ...
- Flask:web表单
客户端发送的所有通过POST发出的请求信息都可以通过request.form获取.但是如果我们要生成表单的HTML代码和验证提交的表单数据那么就需要采用另外的方法.Flask-WTF扩展可以把处理we ...
- Flask:处理Web表单
尽管 Flask 的请求对象提供的信息足以处理 Web 表单,但有些任务很单调,而且要重复操作.比如,生成表单的 HTML 代码和验证提交的表单数据.Flask-WTF 扩展可以把处理 Web 表单的 ...
- Flask Web Development —— Web表单(上)
Flask-WTF扩展使得处理web表单能获得更愉快的体验.该扩展是一个封装了与框架无关的WTForms包的Flask集成. Flask-WTF和它的依赖集可以通过pip来安装: (venv) $ p ...
- Flask教程 —— Web表单(上)
第二章中介绍的request对象公开了所有客户端发送的请求信息.特别是request.form可以访问POST请求提交的表单数据. 尽管Flask的request对象提供的支持足以处理web表单,但依 ...
随机推荐
- [深度概念]·K-Fold 交叉验证 (Cross-Validation)的理解与应用
K-Fold 交叉验证 (Cross-Validation)的理解与应用 我的网站 1.K-Fold 交叉验证概念 在机器学习建模过程中,通行的做法通常是将数据分为训练集和测试集.测试集是与训练独立的 ...
- Shell从入门到精通进阶之四:流程控制
流程控制是改变程序运行顺序的指令. 4.1 if语句 4.1.1 单分支 if 条件表达式; then 命令 fi 示例: #!/bin/bash N=10 if [ $N -gt 5 ]; then ...
- GoLang simple-project-demo-02
GoLang 有很多种数据类型:字符型,整型,浮点型,布尔型.下面是基础例子: package main import "fmt" func main() { fmt.Printl ...
- 兼容小程序的canvas画图组件jmGraph
基于CANVAS的简单画图组件让你用类似于dom的方式,在canvas上画图,感觉会不会很爽. 主页:http://graph.jm47.com/示例:http://graph.jm47.com/ex ...
- 初步认识Swiper_前端交互控制神器_滚动3D切换等特效简单制作
前言: 本人在项目的工作中负责研发,页面及交互基本都是交给前端去做的.以前前端写的东西大概都知道,都是一些JS,CSS和HTML等的一些基本控制,都懂!但是今天前端突然做了一个具有特殊效果的DOM:页 ...
- sql 脚本编写之路 常用语句(一) 1.用一个表中的某一列更新另外一个表的某些列:
for ACCESS 数据库: update a, b set a.name=b.name1 where a.id=b.id for SQL Server 数据库: update a set a.na ...
- Mysql、SqlServer、Oracle三大数据库的区别
一.MySQL 优点: 体积小.速度快.总体拥有成本低,开源: 支持多种操作系统: 是开源数据库,提供的接口支持多种语言连接操作 : MySQL的核心程序采用完全的多线程编程.线程是轻量级的进程,它可 ...
- 我的世界 ParaCraft 结合开源地图 OpenStreetMap 生成3D校园的方法简介
我的世界ParaCraft结合开源地图OpenStreetMap生成3D校园的方法简介 版本1.0 日期2019.2.3 作者Ray (82735589@qq.com) www.TimeGIS.com ...
- Odoo薪酬管理 公式配置
薪酬计算的一般原理是:在基本工资的基础上,加上各种津贴,减去社保.公积金.个税等各种扣除项之后,得出最终的实发工资.此外,还要计算社保.公积金等公司应该承担的部分. 在同一公司中,针对不同的地区.不同 ...
- 智能指针std::weak_ptr
std::weak_ptr 避免shared_ptr内存泄漏的利器.