成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Flask-WTF進(jìn)階和WTForms擴(kuò)展

zone / 1701人閱讀

摘要:和都是很好用的插件,然而當(dāng)它們結(jié)合到一起后,就不是那么美妙了。當(dāng)然了寫一個(gè)完整的擴(kuò)展還是很麻煩的??梢钥垂俜綌U(kuò)展的源碼。。。

Flask-WTFFlask-SQLAlchemy都是很好用的插件,然而當(dāng)它們結(jié)合到一起后,就不是那么美妙了。

問題的提出

models.py中定義了一個(gè)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"))
    # 文章標(biāo)簽
    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):
    """文章標(biāo)簽"""

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

# 文章和標(biāo)簽的映射表 ,多對(duì)多關(guān)系
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中定義一個(gè)ArticleForm表單

class ArticleForm(Form):

    title = StringField(u"標(biāo)題", validators=[Required()])
    category = QuerySelectField(u"分類", query_factory=getUserFactory(["id", "name"]), get_label="name")
    tags = StringField(u"標(biāo)簽", validators=[Required()])
    content = PageDownField(u"正文", validators=[Required()])
    submit = SubmitField(u"發(fā)布")

此時(shí)在處理表單的時(shí)候可以這樣:

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:
    標(biāo)簽列表,如[u"測(cè)試",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就是一個(gè)對(duì)象,為撒到form.data.tags了就不是了,還要專門寫一個(gè)函數(shù)來坐一個(gè)轉(zhuǎn)換?這個(gè)時(shí)候就有必要擴(kuò)展WTForms中的表單了。
 

WTForms入門

閱讀WTForms文檔,關(guān)于如何創(chuàng)建一個(gè)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 = []

簡(jiǎn)單了看了一下WTForms源碼,大致搞清楚了上面代碼兩個(gè)方法的作用:

_value The _value method is called by the TextInput widget to provide the value that is displayed in the form. 在初始化表單的時(shí)候,就是調(diào)用這個(gè)方法在表單中渲染數(shù)據(jù)

process_formdata 表單提交時(shí),處理該字段的數(shù)據(jù)。

編寫WTForm擴(kuò)展

根據(jù)上面的代碼,將TagListField中的字符串轉(zhuǎn)為models.py中定義的Tag對(duì)象即可:

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):
        """將字符串轉(zhuǎn)換位obj對(duì)象"""
        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):
        """將對(duì)象轉(zhuǎn)換為字符串"""
        if obj:
            return obj.name
        else:
            return u""

主要就是在process_formdata這一步處理表單的數(shù)據(jù),將字符串轉(zhuǎn)換為需要的數(shù)據(jù)。最終就可以在forms.py中這樣定義表單了:

...
class ArticleForm(Form):
    """編輯文章表單"""

    title = StringField(u"標(biāo)題", validators=[Required()])
    category = QuerySelectField(u"分類", query_factory=get_category_factory(["id", "name"]), get_label="name")
    tags = TagListField(u"標(biāo)簽", validators=[Required()])
    content = PageDownField(u"正文", validators=[Required()])
    submit = SubmitField(u"發(fā)布")
...

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)

代碼是不是很簡(jiǎn)潔了?^_^。。。

當(dāng)然了寫一個(gè)完整的WTForms擴(kuò)展還是很麻煩的。這里只是剛剛?cè)腴T??梢钥垂俜綌U(kuò)展QuerySelectField的源碼。。。

最終效果

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/37788.html

相關(guān)文章

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

    摘要:每個(gè)表單域都可以連接到一個(gè)或多個(gè)是一個(gè)用于檢查用戶提交的輸入是否合法的函數(shù)。表單域構(gòu)造函數(shù)的第一個(gè)參數(shù)是一個(gè),在渲染表單到時(shí)會(huì)使用。驗(yàn)證確保提交的表單域不為空。表單域驗(yàn)證都是直接從包中導(dǎo)入。表格展示了一組支持的標(biāo)準(zhǔn)表單域。 第二章中介紹的request對(duì)象公開了所有客戶端發(fā)送的請(qǐng)求信息。特別是request.form可以訪問POST請(qǐng)求提交的表單數(shù)據(jù)。 盡管Flask的request...

    CODING 評(píng)論0 收藏0
  • Flask五之表單

    摘要:通過的核心特性,函數(shù)可實(shí)現(xiàn)這種效果僅調(diào)用函數(shù)并不能把消息顯示出來,程序使用的模板要渲染這些消息。注意在模板中使用循環(huán)是因?yàn)樵谥暗恼?qǐng)求循環(huán)中每次調(diào)用函數(shù)時(shí)都會(huì)生成一個(gè)消息,所以可能有多個(gè)消息在排隊(duì)等待顯示。 五、表單 1、Flask-WTF 擴(kuò)展 Flask-WTF 及其依賴可使用 pip 安裝: (venv) $ pip install flask-wtf 2、跨站請(qǐng)求偽造保護(hù) 【設(shè)...

    justCoding 評(píng)論0 收藏0
  • 使用 Flask AngularJS 構(gòu)建博客 - 1

    摘要:注原文作者,原文地址為在這個(gè)教程中,我們將使用和構(gòu)建一個(gè)博客。在開發(fā)期間,這將允許我們把它們運(yùn)行在不同的端口例如和?,F(xiàn)在我們將進(jìn)入目錄并使用運(yùn)行這個(gè)腳本。示例創(chuàng)建一篇文章為了創(chuàng)建一篇文章,你需要發(fā)送一個(gè)請(qǐng)求給。 注:原文作者 John Kevin M. Basco,原文地址為 Building a blog using Flask and AngularJS Part 1 在...

    劉玉平 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<