在Yii框架中使用PHP模板引擎Twig的例子

Twig是一款快速、安全、灵活的PHP模板引擎,它内置了许多filter和tags,并且支持模板继承,能让你用最简洁的代码来描述你的模板。他的语法和Python下的模板引擎Jinjia以及Django的模板语法都非常像。 比如我们在PHP中需要输出变量并且将其进行转义时,语法比较累赘:

代码如下:


但是在Twig中可以这样写:
代码如下:
{{ var }}
{{ var|escape }}
{{ var|e }} {# shortcut to escape a variable #}

遍历数组:
代码如下:
{% for user in users %}
* {{ user.name }}
{% else %}
No user has been found.
{% endfor %}

但是要在Yii Framework集成Twig就会遇到点麻烦了,官方网站中已经有能够集成Twig的方案,所以这里我也不再赘述。但是由于Twig中是不支持PHP语法的,所以在有些表达上会遇到困难,比如我们在写Form的视图时,经常会这么写:

代码如下:
beginWidget('CActiveForm'); ?>
Login


  • label($model,'username'); ?>
    textField($model,'username'); ?>


  • label($model,'password'); ?>
    passwordField($model,'password'); ?>


error($model,'password'); ?>
endWidget(); ?>

但是这样的语法是没法在twig中表达的,所以想去扩展下Twig的功能,让他能够支持我们自定义的widget标签,然后自动解析成我们需要的代码。 总共需要两个类:TokenParser和Node,下面直接上代码
代码如下:
PHP
/*
* This file is an extension of Twig.
*
* (c) 2010 lfyzjck
*/

/**
* parser widget tag in Yii framework
*
* {% beginwidget 'CActiveForm' as form %}
* content of form
* {% endwidget %}
*
*/
class Yii_WidgetBlock_TokenParser extends Twig_TokenParser
{
/**
* Parses a token and returns a node.
*
* @param Twig_Token $token A Twig_Token instance
*
* @return Twig_NodeInterface A Twig_NodeInterface instance
*/
public function parse(Twig_Token $token)
{
$lineno = $token->getLine();
$stream = $this->parser->getStream();

$name = $stream->expect(Twig_Token::STRING_TYPE);
if($stream->test(Twig_Token::PUNCTUATION_TYPE)){
$args = $this->parser->getExpressionParser()->parseHashExpression();
}
else{
$args = new Twig_Node_Expression_Array(array(),$lineno);
}

$stream->expect(Twig_Token::NAME_TYPE);
$assign = $stream->expect(Twig_Token::NAME_TYPE);
$stream->expect(Twig_Token::BLOCK_END_TYPE);

$body = $this->parser->subparse(array($this,'decideBlockEnd'),true);
$stream->expect(Twig_Token::BLOCK_END_TYPE);

return new Yii_Node_WidgetBlock(array(
'alias' => $name->getValue(),
'assign' => $assign,
),$body,$args,$lineno,$this->getTag());
}

/**
* Gets the tag name associated with this token parser.
*
* @param string The tag name
*/
public function getTag()
{
return 'beginwidget';
}

public function decideBlockEnd(Twig_Token $token)
{
return $token->test('endwidget');
}
}

class Yii_Node_WidgetBlock extends Twig_Node
{
public function __construct($attrs,Twig_NodeInterface $body,Twig_Node_Expression_Array $args = NULL,$tag)
{
$attrs = array_merge(array('value' => false),$attrs);
$nodes = array('args' => $args,'body' => $body);
parent::__construct($nodes,$attrs,$tag);
}

public function compile(Twig_Compiler $compiler)
{
$compiler->addDebugInfo($this);
$compiler->write('$context["'.$this->getAttribute('assign')->getValue().'"] = $context["this"]->beginWidget("'.$this->getAttribute('alias').'",');
$argNode = $this->getNode('args');
$compiler->subcompile($argNode)
->raw(');')
->raw("\n");

$compiler->indent()->subcompile($this->getNode('body'));

$compiler->raw('$context["this"]->endWidget();');
}
}
?>


然后在Twig初始化的地方增加我们的语法解析类:
代码如下:
$twig->addTokenParser(new Yii_WidgetBlock_TokenParser);

然后我们就可以在twig的模板里这么写了:
代码如下:
{% beginwidget 'CActiveForm' as form %}


  • {{ form.label(model,'username') }}
    {{ form.textField(model,'username') }}


  • {{ form.label(model,'password') }}
    {{ form.passwordField(model,'password') }}


{% endwidget %}

相关文章

统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...
统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...
前言 之前做了微信登录,所以总结一下微信授权登录并获取用户...
FastAdmin是我第一个接触的后台管理系统框架。FastAdmin是一...
之前公司需要一个内部的通讯软件,就叫我做一个。通讯软件嘛...
统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...