资讯专栏INFORMATION COLUMN

Flask-WTF进阶和WTForms扩展

zone / 1692人阅读

摘要:和都是很好用的插件,然而当它们结合到一起后,就不是那么美妙了。当然了写一个完整的扩展还是很麻烦的。可以看官方扩展的源码。。。

Flask-WTFFlask-SQLAlchemy都是很好用的插件,然而当它们结合到一起后,就不是那么美妙了。

问题的提出

models.py中定义了一个ArticleCategoryTag类:

class Article(db.Model):
    """定义文章"""

    __tablename__ = "articles"
    id = db.Column(db.Integer, primary_key=True)
    title = db.Column(db.String(128), unique=True, index=True)
    # 保存md格式的文本
    content = db.Column(db.Text)
    # 保存html格式的文本
    content_html = db.Column(db.Text)
    # 文章分类
    category_id = db.Column(db.Integer, db.ForeignKey("categories.id"))
    # 文章标签
    tags = db.relationship(
        "Tag", secondary="article_tag_ref", backref="articles")
class Category(db.Model):
    """文章分类"""

    __tablename__ = "categories"
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(128), unique=True)
    articles = db.relationship("Article", backref="category", lazy="dynamic")

class Tag(db.Model):
    """文章标签"""

    __tablename__ = "tags"
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String(128), unique=True)

# 文章和标签的映射表 ,多对多关系
article_tag_ref = db.Table("article_tag_ref",
                           db.Column("article_id", db.Integer,
                                     db.ForeignKey("articles.id")),
                           db.Column("tag_id",  db.Integer,
                                     db.ForeignKey("tags.id"))
                           )

然后在forms.py中定义一个ArticleForm表单

class ArticleForm(Form):

    title = StringField(u"标题", validators=[Required()])
    category = QuerySelectField(u"分类", query_factory=getUserFactory(["id", "name"]), get_label="name")
    tags = StringField(u"标签", validators=[Required()])
    content = PageDownField(u"正文", validators=[Required()])
    submit = SubmitField(u"发布")

此时在处理表单的时候可以这样:

form = ArticleForm()
if form.validate_on_submit():
    article = Article(title=from.data.title, content=form.data.content,category=form.category.data)
    ...

等等,这样怎么处理form.data.tags?只有像下面这样写了:

"""
:param tags:
    标签列表,如[u"测试",u"Flask"]
"""
def str_to_obj(tags):
    r = []
    for tag in tags:
        tag_obj = Tag.query.filter_by(name=tag).first()
        if tag_obj is None:
            tag_obj = Tag(name=tag)
        r.append(tag_obj)
    return r

然后在上面的代码中加入:

 form = ArticleForm()
if form.validate_on_submit():
    article = Article(title=from.data.title, content=form.data.content, category=form.category.data, tags=str_to_obj(form.data.tags))   

这样是不是很难看,像form.data.category就是一个对象,为撒到form.data.tags了就不是了,还要专门写一个函数来坐一个转换?这个时候就有必要扩展WTForms中的表单了。
 

WTForms入门

阅读WTForms文档,关于如何创建一个TagListField,贴一下代码:

class TagListField(Field):
    widget = TextInput()

    def _value(self):
        if self.data:
            return u", ".join(self.data)
        else:
            return u""

    def process_formdata(self, valuelist):
        if valuelist:
            self.data = [x.strip() for x in valuelist[0].split(",")]
        else:
            self.data = []

简单了看了一下WTForms源码,大致搞清楚了上面代码两个方法的作用:

_value The _value method is called by the TextInput widget to provide the value that is displayed in the form. 在初始化表单的时候,就是调用这个方法在表单中渲染数据

process_formdata 表单提交时,处理该字段的数据。

编写WTForm扩展

根据上面的代码,将TagListField中的字符串转为models.py中定义的Tag对象即可:

class TagListField(Field):
    widget = TextInput()

    def __init__(self, label=None, validators=None,
                 **kwargs):
        super(TagListField, self).__init__(label, validators, **kwargs)

    def _value(self):
        if self.data:
            r = u""
            for obj in self.data:
                r += self.obj_to_str(obj)
            return u""
        else:
            return u""

    def process_formdata(self, valuelist):
        print "process_formdata.."
        print valuelist
        if valuelist:
            tags = self._remove_duplicates([x.strip() for x in valuelist[0].split(",")])
            self.data = [self.str_to_obj(tag) for tag in tags]
        else:
            self.data = None

    def pre_validate(self, form):
        pass

    @classmethod
    def _remove_duplicates(cls, seq):
        """去重"""
        d = {}
        for item in seq:
            if item.lower() not in d:
                d[item.lower()] = True
                yield item

    @classmethod
    def str_to_obj(cls, tag):
        """将字符串转换位obj对象"""
        tag_obj = Tag.query.filter_by(name=tag).first()
        if tag_obj is None:
            tag_obj = Tag(name=tag)
        return tag_obj

    @classmethod
    def obj_to_str(cls, obj):
        """将对象转换为字符串"""
        if obj:
            return obj.name
        else:
            return u""

主要就是在process_formdata这一步处理表单的数据,将字符串转换为需要的数据。最终就可以在forms.py中这样定义表单了:

...
class ArticleForm(Form):
    """编辑文章表单"""

    title = StringField(u"标题", validators=[Required()])
    category = QuerySelectField(u"分类", query_factory=get_category_factory(["id", "name"]), get_label="name")
    tags = TagListField(u"标签", validators=[Required()])
    content = PageDownField(u"正文", validators=[Required()])
    submit = SubmitField(u"发布")
...

views.py中处理表单就很方便了:

def edit_article():
    """编辑文章"""

    form = ArticleForm()
    if form.validate_on_submit():
        article = Article(title=form.title.data, content=form.content.data)
        article.tags = form.tags.data
        article.category = form.category.data
        try:
            db.session.add(article)
            db.session.commit()
        except:
            db.session.rollback()
    return render_template("dashboard/edit.html", form=form)

代码是不是很简洁了?^_^。。。

当然了写一个完整的WTForms扩展还是很麻烦的。这里只是刚刚入门。可以看官方扩展QuerySelectField的源码。。。

最终效果

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/37788.html

相关文章

  • Flask Web Development —— Web表单(上)

    摘要:每个表单域都可以连接到一个或多个是一个用于检查用户提交的输入是否合法的函数。表单域构造函数的第一个参数是一个,在渲染表单到时会使用。验证确保提交的表单域不为空。表单域验证都是直接从包中导入。表格展示了一组支持的标准表单域。 第二章中介绍的request对象公开了所有客户端发送的请求信息。特别是request.form可以访问POST请求提交的表单数据。 尽管Flask的request...

    CODING 评论0 收藏0
  • Flask五之表单

    摘要:通过的核心特性,函数可实现这种效果仅调用函数并不能把消息显示出来,程序使用的模板要渲染这些消息。注意在模板中使用循环是因为在之前的请求循环中每次调用函数时都会生成一个消息,所以可能有多个消息在排队等待显示。 五、表单 1、Flask-WTF 扩展 Flask-WTF 及其依赖可使用 pip 安装: (venv) $ pip install flask-wtf 2、跨站请求伪造保护 【设...

    justCoding 评论0 收藏0
  • 使用 Flask AngularJS 构建博客 - 1

    摘要:注原文作者,原文地址为在这个教程中,我们将使用和构建一个博客。在开发期间,这将允许我们把它们运行在不同的端口例如和。现在我们将进入目录并使用运行这个脚本。示例创建一篇文章为了创建一篇文章,你需要发送一个请求给。 注:原文作者 John Kevin M. Basco,原文地址为 Building a blog using Flask and AngularJS Part 1 在...

    刘玉平 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<