模板

雖然 Flask 并不強(qiáng)迫我們使用任何一個(gè)特定的模板語(yǔ)言,它假設(shè)我們要使用 Jinja。在 Flask 社區(qū)中大部分開(kāi)發(fā)者使用 Jinja,我建議你們也這樣做。有很多的擴(kuò)展幫助我們使用其它的模板語(yǔ)言,像 Flask-Genshi 和 Flask-Mako。堅(jiān)持使用默認(rèn)的模板語(yǔ)言,除非你有更好的理由使用其它的模板語(yǔ)言。還不知道 Jinja 語(yǔ)法不是一個(gè)好的理由!你會(huì)節(jié)省大量的時(shí)間和煩惱。

<em>
當(dāng)我們提及到 “Jinja” 的時(shí)候,就是在說(shuō) Jinja2。存在 Jinja1,但是我們不會(huì)與它打交道。我們討論的是這個(gè):http://jinja.pocoo.org/。
</em>

Jinja 快速入門(mén)

Jinja 官方文檔在解釋語(yǔ)法和語(yǔ)言的功能上做出很大的工作。我不會(huì)在這里重復(fù),但是我要確保你記得這個(gè)重要的注意事項(xiàng):

<em>
有兩種分隔符。{% raw %}{% ... %}{% endraw %} 和 {% raw %}{{ ... }}{% endraw %}。第一個(gè)用于執(zhí)行類似 for 循環(huán)或者賦值的聲明,后者是用于輸出表達(dá)的結(jié)果到模板中。
</em>

如何組織模板

那么模板如何融入到我們的應(yīng)用程序?如果你一直關(guān)注 Flask 的話,你可能注意到了 Flask 是十分靈活,它并沒(méi)有對(duì)其內(nèi)容進(jìn)行一些特殊的限制。模板也不例外。你可能也注意到了通常有一個(gè)推薦的地方來(lái)放置東西(比如,模板)。對(duì)于模板而言,那個(gè)地方就是在包的目錄里。

myapp/
    __init__.py
    models.py
    views/
    templates/
    static/
run.py
requirements.txt
templates/
    layout.html
    index.html
    about.html
    profile/
        layout.html
        index.html
    photos.html
    admin/
        layout.html
        index.html
        analytics.html

templates 目錄的結(jié)構(gòu)是與我們路由結(jié)構(gòu)平行的。對(duì)于路由 myapp.com/admin/analytics 的模板就是 templates/admin/analytics.html。在目錄里面還有一些額外的模板,它們不會(huì)直接地被渲染。layout.html 文件是為了讓其它的模板繼承。

繼承

很像蝙蝠俠的背景故事一樣,一個(gè)組織優(yōu)秀的模板目錄很大程度上依靠繼承。父模板 通常定義一個(gè)通用的結(jié)構(gòu),所有 子模板 都能很好的繼承它。在我們的例子中,layout.html 就是一個(gè)父模板而其它 .html 文件就是子模板。

你通常有一個(gè)頂層的 layout.html,它定義了你的應(yīng)用程序的通用布局以及你的網(wǎng)站的每一部分。如果你看看上面的目錄的話,你會(huì)看到一個(gè)頂層的 myapp/templates/layout.html,同樣還有 myapp/templates/profile/layout.html 和 myapp/templates/admin/layout.html。最后兩個(gè)文件繼承和修改第一個(gè)文件。

{# _myapp/templates/layout.html_ #}


<!DOCTYPE html>
<html lang="en">
    <head>
        <title>{% raw %}{% block title %}{% endblock %}{% endraw %}</title>
    </head>
    <body>
    {% block body %}
        <h1>This heading is defined in the parent.</h1>
    {% endblock %}
    </body>
</html>

在子模板中,我們可以擴(kuò)展父模板并且定義這些塊的內(nèi)容。

{# _myapp/templates/index.html_ #}

{% extends "layout.html" %}
{% block title %}Hello world!{% endblock %}
{% block body %}
    {{ super() }}
    <h2>This heading is defined in the child.</h2>
{% endblock %}

super() 函數(shù)讓我們渲染父級(jí)塊的內(nèi)容。

<em>
關(guān)于繼承的更多信息,請(qǐng)參閱 Jinja 模板繼承文檔。
</em>

創(chuàng)建宏

我們可以在我們模板中堅(jiān)持 DRY(不要重復(fù)自己)的原則,通過(guò)抽象出重復(fù)出現(xiàn)的代碼片段到 宏。如果我們正工作在為我們應(yīng)用程序?qū)Ш降?HTML 上,我們需要給一個(gè) “活躍的”鏈接一個(gè) class(class=”active”)。沒(méi)有宏的話,我們要編寫(xiě)一大段 if ... else 語(yǔ)句,這些語(yǔ)句檢查每一個(gè)鏈接找到正處于活躍的一個(gè)。

宏提供了一種模塊化代碼的方式;它們像函數(shù)一樣工作。讓我們看看如何使用宏標(biāo)記一個(gè)活躍的鏈接。

{# myapp/templates/layout.html #}

{% from "macros.html" import nav_link with context %}
<!DOCTYPE html>
<html lang="en">
    <head>
    {% block head %}
        <title>My application</title>
    {% endblock %}
    </head>
    <body>
        <ul class="nav-list">
            {{ nav_link('home', 'Home') }}
            {{ nav_link('about', 'About') }}
            {{ nav_link('contact', 'Get in touch') }}
        </ul>
    {% block body %}
    {% endblock %}
    </body>
</html>

在這個(gè)模板中我們現(xiàn)在要做的就是調(diào)用一個(gè)未定義的宏 - nav_link -接著向其傳遞兩個(gè)參數(shù):目標(biāo)端點(diǎn)(例如,目標(biāo)視圖的函數(shù)名)以及我們要顯示的文本。

  • 你可能會(huì)注意到在導(dǎo)入語(yǔ)句中我們指定了 with context。Jinja 的 context 是由傳遞到 render_template() 函數(shù)的參數(shù)以及來(lái)自我們的 Python 代碼的 Jinja 環(huán)境上下文組成。對(duì)于模板來(lái)說(shuō),這些變量在模板被渲染的時(shí)候是可用的。
  • 一些變量是明顯地由我們傳入,例如,render_template("index.html", color="red"),但是還有一些變量和函數(shù)是由 Flask 自動(dòng)地包含在上下文中,例如,request, gsession。當(dāng)我們說(shuō) {% raw %}{% from ... import ... with context %}{% endraw %} 的時(shí)候,就是告訴 Jinja 這些變量對(duì)宏也可用。
  • 通過(guò) Flask 傳入到 Jinja 上下文的所有全局變量: http://flask.pocoo.org/docs/templating/#standard-context(中文翻譯:http://www.pythondoc.com/flask/templating.html#id2)。
  • 我們可以使用上下文處理器定義我們想要的并且插入到 Jinja 上下文的變量和函數(shù): http://flask.pocoo.org/docs/templating/#context-processors (中文翻譯:http://www.pythondoc.com/flask/templating.html#id6)。

現(xiàn)在是時(shí)候定義在我們模板中使用的 nav_link 宏。

{# myapp/templates/macros.html #}

{% macro nav_link(endpoint, text) %}
{% if request.endpoint.endswith(endpoint) %}
    <li class="active"><a href="{{ url_for(endpoint) }}">{{text}}</a></li>
{% else %}
    <li><a href="{{ url_for(endpoint) }}">{{text}}</a></li>
{% endif %}
{% endmacro %}

現(xiàn)在我們已經(jīng)在 myapp/templates/macros.html 中定義了宏。在這個(gè)宏中我們使用了 Flask 的 request 對(duì)象 — 默認(rèn)情況下在 Jinja 上下文中是可用的 — 用來(lái)檢查傳入到 nav_link 中的路由的端點(diǎn)是否是當(dāng)前請(qǐng)求。如果是,我們正在當(dāng)前頁(yè)面上,接著我們標(biāo)記它為活躍的。

  • 從 x 導(dǎo)入 y 語(yǔ)句采用了 x 的相對(duì)路徑。如果我們的模板是 myapp/templates/user/blog.html,我們可以在使用 from "../macros.html" 導(dǎo)入 nav_link。

自定義過(guò)濾器

Jinja 過(guò)濾器是一個(gè)函數(shù),它能夠在 {% raw %}{{ ... }}{% endraw %} 中用于處理一個(gè)表達(dá)式的結(jié)果。在表達(dá)式結(jié)果輸出到模板之前它就被調(diào)用。

<h2>{{ article.title|title }}</h2>

在這段代碼中,title 過(guò)濾器接收 article.title 作為參數(shù)并且返回一個(gè)過(guò)濾后的標(biāo)題,接著過(guò)濾后的標(biāo)題將會(huì)輸出到模板中。這就像 UNIX 的“管道化”一個(gè)程序到另一個(gè)程序的輸出。

  • 有很多像 title 一樣的內(nèi)置過(guò)濾器。請(qǐng)參閱 Jinja 文檔中的 完整列表。

我們可以在我們的 Jinja 模板中定義自己的過(guò)濾器供使用。舉例來(lái)說(shuō),我們將會(huì)實(shí)現(xiàn)一個(gè)簡(jiǎn)單 caps 過(guò)濾器用來(lái)大寫(xiě)一個(gè)字符串中所有的字母。

  • Jinja 已經(jīng)有一個(gè) upper 過(guò)濾器來(lái)做這樣的事情,并且還有一個(gè) capitalize 過(guò)濾器,它能用來(lái)大寫(xiě)第一個(gè)字母,小寫(xiě)其余的字母。這些也能處理 unicode 轉(zhuǎn)換,但是我們會(huì)繼續(xù)我們的示例,讓大家目前能夠知道如何自定義過(guò)濾器。

我們要在 myapp/util/filters.py 中定義我們的過(guò)濾器。這里給出一個(gè) util 包,它里面有各種各樣的模塊。

# myapp/util/filters.py

from .. import app

@app.template_filter()
def caps(text):
    """Convert a string to all caps."""
    return text.uppercase()

在這段代碼中我們使用 @app.template_filter() 裝飾器注冊(cè)我們的函數(shù)成一個(gè) Jinja 過(guò)濾器。默認(rèn)的過(guò)濾器名稱就是函數(shù)的名稱,但是你可以傳入一個(gè)參數(shù)到裝飾器中來(lái)改變它。

@app.template_filter('make_caps')
def caps(text):
    """Convert a string to all caps."""
    return text.uppercase()

現(xiàn)在我們可以在模板中調(diào)用 make_caps 而不是 {% raw %}caps:{{ "hello world!"|make_caps }}{% endraw %}。

為了要讓我們的過(guò)濾器在模板中可用的話,我們只需要在我們的頂層 \\init.py\\ 的中導(dǎo)入它。

# myapp/__init__.py

# Make sure app has been initialized first to prevent circular imports.
from .util import filters

摘要

  • 使用 Jinja 模板。
  • Jinja 有兩種分隔符:{% ... %}{{ ... }}。 第一個(gè)用于執(zhí)行類似 for 循環(huán)或者賦值的聲明,后者是用于輸出表達(dá)的結(jié)果到模板中。
  • 模板應(yīng)該在 myapp/templates/ 中 — 例如,在應(yīng)用程序包中的一個(gè)目錄。
  • 我建議模板/目錄結(jié)構(gòu)反映應(yīng)用程序的 URL 結(jié)構(gòu)。
  • 你應(yīng)該在 myapp/templates 中有一個(gè)頂層的 layout.html,同樣網(wǎng)站的每一部分也應(yīng)該有一個(gè)。后者繼承并且擴(kuò)展了前者。
  • 宏就像由模板代碼構(gòu)成的函數(shù)。
  • 過(guò)濾器就是有 Python 代碼組成的函數(shù)并且能在模板中使用。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 第三章 模板 序 為什么要分離 易于維護(hù)的代碼,關(guān)鍵在于保持簡(jiǎn)單的結(jié)構(gòu)。而我們之前編寫(xiě)的hello.py雖然簡(jiǎn)單,...
    科幻經(jīng)典閱讀 1,612評(píng)論 0 6
  • 默認(rèn)情況下,F(xiàn)lask 在程序文件夾中的 templates 子文件夾中尋找模板。在下一個(gè) hello.py 版本...
    改變自己_now閱讀 8,190評(píng)論 0 8
  • 渲染模板 模板文件后綴名是 .html, 這些文件就是普通的 html 文件加上一些占位符變量, 它們都默認(rèn)存放在...
    焉知非魚(yú)閱讀 835評(píng)論 0 1
  • 模板是一個(gè)包含響應(yīng)文本的文件,其中包含用占位變量表示的動(dòng)態(tài)部分,其具體值只在請(qǐng)求的上下文中才能知道。使用真實(shí)值替換...
    藕絲空間閱讀 1,362評(píng)論 0 1
  • 莫憶往昔滋滋趣事,無(wú)戀那日嘻嘻笑游。風(fēng)塵黃土本自天意,無(wú)盡征途何患幽幽。天下大事何止今日,無(wú)奈我輩幾人憂愁。難覓清...
    鄭樹(shù)宏閱讀 284評(píng)論 5 3

友情鏈接更多精彩內(nèi)容