ThinkPHP的模板引擎內(nèi)置了布局模板功能支持,可以方便的實現(xiàn)模板布局以及布局嵌套功能。

有三種布局模板的支持方式:

第一種方式:全局配置方式

這種方式僅需在項目配置文件中添加相關(guān)的布局模板配置,就可以簡單實現(xiàn)模板布局功能,比較適用于全站使用相同布局的情況,需要配置開啟layout_on 參數(shù)(默認(rèn)不開啟),并且設(shè)置布局入口文件名layout_name(默認(rèn)為layout)。

'template'  =>  [
    'layout_on'     =>  true,
    'layout_name'   =>  'layout',
]

開啟layout_on后,我們的模板渲染流程就有所變化,例如:

namespace app\index\controller;

use think\Controller;

Class User extends Controller
{
     public function add() 
     {
         return $this->fetch('add');
     }
}

在不開啟layout_on布局模板之前,會直接渲染 application/index/view/user/add.html 模板文件,開啟之后,首先會渲染application/index/view/layout.html 模板,布局模板的寫法和其他模板的寫法類似,本身也可以支持所有的模板標(biāo)簽以及包含文件,區(qū)別在于有一個特定的輸出替換變量{__CONTENT__},例如,下面是一個典型的layout.html模板的寫法:

{include file="public/header" /}
 {__CONTENT__}
{include file="public/footer" /}

讀取layout模板之后,會再解析user/add.html 模板文件,并把解析后的內(nèi)容替換到layout布局模板文件的{CONTENT} 特定字符串。

當(dāng)然可以通過設(shè)置來改變這個特定的替換字符串,例如:

'template'  =>  [
    'layout_on'     =>  true,
    'layout_name'   =>  'layout',
    'layout_item'   =>  '{__REPLACE__}'
]

一個布局模板同時只能有一個特定替換字符串。

采用這種布局方式的情況下,一旦user/add.html 模板文件或者layout.html布局模板文件發(fā)生修改,都會導(dǎo)致模板重新編譯。

如果需要指定其他位置的布局模板,可以使用:

'template'  =>  [
    'layout_on'     =>  true,
    'layout_name'   =>  'layout/layoutname',
    'layout_item'   =>  '{__REPLACE__}'
]

就表示采用application/index/view/layout/layoutname.html作為布局模板。

如果某些頁面不需要使用布局模板功能,可以在模板文件開頭加上 {__NOLAYOUT__} 字符串。

如果上面的user/add.html 模板文件里面包含有{__NOLAYOUT__},則即使當(dāng)前開啟布局模板,也不會進(jìn)行布局模板解析。

第二種方式:模板標(biāo)簽方式

這種布局模板不需要在配置文件中設(shè)置任何參數(shù),也不需要開啟layout_on,直接在模板文件中指定布局模板即可,相關(guān)的布局模板調(diào)整也在模板中進(jìn)行。

以前面的輸出模板為例,這種方式的入口還是在user/add.html 模板,但是我們可以修改下add模板文件的內(nèi)容,在頭部增加下面的布局標(biāo)簽(記得首先關(guān)閉前面的layout_on設(shè)置,否則可能出現(xiàn)布局循環(huán)):

{layout name="layout" /}

表示當(dāng)前模板文件需要使用layout.html 布局模板文件,而布局模板文件的寫法和上面第一種方式是一樣的。當(dāng)渲染user/add.html 模板文件的時候,如果讀取到layout標(biāo)簽,則會把當(dāng)前模板的解析內(nèi)容替換到layout布局模板的{CONTENT} 特定字符串。

一個模板文件中只能使用一個布局模板,如果模板文件中沒有使用任何layout標(biāo)簽則表示當(dāng)前模板不使用任何布局。

如果需要使用其他的布局模板,可以改變layout的name屬性,例如:

{layout name="newlayout" /}

還可以在layout標(biāo)簽里面指定要替換的特定字符串:

{layout name="Layout/newlayout" replace="[__REPLACE__]" /}

第三種方式:使用layout控制模板布局

使用內(nèi)置的layout方法可以更靈活的在程序中控制模板輸出的布局功能,尤其適用于局部需要布局或者關(guān)閉布局的情況,這種方式也不需要在配置文件中開啟layout_on。例如:

namespace app\index\controller;

use think\Controller;

class User extends Controller
{
     public function add() 
     {
         $this->view->engine->layout(true);
         return $this->fetch('add');
     }
}

表示當(dāng)前的模板輸出啟用了布局模板,并且采用默認(rèn)的layout布局模板。

如果當(dāng)前輸出需要使用不同的布局模板,可以動態(tài)的指定布局模板名稱,例如:

namespace app\index\controller;

use think\Controller;

class User extends Controller
{
     public function add() 
     {
         $this->view->engine->layout('Layout/newlayout');
         return $this->display('add');
     }
}

或者使用layout方法動態(tài)關(guān)閉當(dāng)前模板的布局功能(這種用法可以配合第一種布局方式,例如全局配置已經(jīng)開啟了布局,可以在某個頁面單獨(dú)關(guān)閉):

namespace app\index\controller;

use think\Controller;

class User extends Controller
{
     public function add() 
     {
        // 臨時關(guān)閉當(dāng)前模板的布局功能
         $this->view->engine->layout(false); 
         return $this->display('add');
     }
}

三種模板布局方式中,第一種和第三種是在程序中配置實現(xiàn)模板布局,第二種方式則是單純通過模板標(biāo)簽在模板中使用布局。具體選擇什么方式,需要根據(jù)項目的實際情況來了。