使用twigExtension实现页面区块功能

所谓区块,就是在页面中展示的一块内容,可以与当前页面内容相关或者无关。譬如常见的在网站侧边栏展示 “最新文章”区块,或者在文章内容页展示“相关文章”区块。

对于与当前页面相关的区块,我们可以在页面的控制器中通过查询数据库等方式获取数据,在模板中 渲染;而对于与当前页面内容无关的区块,我们则需要通过其他方式。

Symfony在处理这些区块时,提供两种种方式:

(来源:http://symfony.com/doc/current/book/templating.html)

include,用来渲染页面相关内容

{% raw  %}
{# app/Resources/views/article/list.html.twig #}
{% extends 'layout.html.twig' %}

{% block body %}
    <h1>Recent Articles<h1>

    {% for article in articles %}
        {{ include('article/article_details.html.twig', { 'article': article }) }}
    {% endfor %}
{% endblock %}
{% endraw %}

render,用来渲染页面无关内容

{% raw  %}

{# app/Resources/views/base.html.twig #}

{# ... #}
<div id="sidebar">
    {{ render(controller(
        'AcmeArticleBundle:Article:recentArticles',
        { 'max': 3 }
    )) }}
</div>

{% endraw %}

然而,由render的运行原理可以看出,它会发送一个子请求,调用子控制器。虽然官方说它很快,但是测试过后发现有很明显的时间消耗。

使用Twig扩展函数

Twig扩展函数是另一个很好的输出区块内容的方式(虽然Symfony官方的介绍不多)。

譬如:

<?php

namespace AppBundle\Twig\Extension;

use Symfony\Component\HttpKernel\KernelInterface;
use JMS\DiExtraBundle\Annotation as DI;

/**
 * @DI\Service("app.twig.extension", public=false)
 * @DI\Tag("twig.extension")
 */
class AppExtension extends \Twig_Extension {

    /**
     * {@inheritdoc}
     */
    public function getFunctions() {
        return array(
            'app_sidemenu' => new \Twig_Function_Method($this, 'renderSideMenu', array(
                'is_safe' => array('html'),
                'needs_environment' => true,
            )),
        );
    }


    /**
     * 侧边栏信息
     */
    public function renderSideMenu (  \Twig_Environment $twig ) {

        $categories = array();

        /* 在此处处理区块逻辑 */

        $template = $twig->loadTemplate('AppBundle:Common:sideblock.html.twig');
        $content = $template->render(array('categories' => $categories));
        return $content;
    }



    /**
     * {@inheritdoc}
     */
    public function getName() {
        return 'app_bundle';
    }
}


上面是一个简单的Twig扩展,其作用就是提供app_sidemenu函数,在页面中显示网站侧边栏分类菜单。 renderSideMenu方法从数据库中获取分类信息,用twig模板渲染成菜单html,然后缓存到redis里。

需要注意的是 getFunctions 方法中,返回数据的 needs_environment 参数,如果设为true,它会为你的回调函数 传入 \Twig_Environment $twig,使你在回调函数中方便地进行模板渲染。

使用 twigExtension 来渲染区块还有个好处就是可以很方便的管理缓存,因为我们可以将render出来的内容存在cache中,这样不同的区块可以有自己的缓存清理逻辑。

下面我们尝试在扩展中使用Redis来缓存区块内容。


namespace AppBundle\Twig\Extension;

use Symfony\Component\HttpKernel\KernelInterface;
use JMS\DiExtraBundle\Annotation as DI;

/**
 * @DI\Service("app.twig.site.extension")
 * @DI\Tag("twig.extension")
 */
class AppSiteExtension extends \Twig_Extension {

    protected $redis;

    /**
     * @DI\InjectParams({
     *     "redis" = @DI\Inject("snc_redis.cache")
     * })
     */
    public function __construct($redis){
        $this->redis = $redis;
    }

    /**
     * {@inheritdoc}
     */
    public function getFunctions() {
        return array(
            'app_header' => new \Twig_Function_Method($this, 'renderHeader', array(
                'is_safe' => array('html'),
                'needs_environment' => true,
            )),
        );
    }

    /**
     * 缓存头部信息,缓存时间 36000s
     */
    public function renderHeader( \Twig_Environment $twig){
        $key = 'app:cache:header';

        if($content = $this->redis->get($key)){
            return $content;
        }
        $template = $twig->loadTemplate('AppBundle:Common:header.html.twig');
        $content = $template->render(array());
        $this->redis->setex($key, 36000, $content);
        return $content;
    }

    /**
     * {@inheritdoc}
     */
    public function getName() {
        return 'app_bundle_site';
    }
}

上面是一个使用redis缓存区块内容的简单应用,跟yii的widget缓存类似(对yii了解不多,错了请指正)。

我们在扩展的构造函数中注入 $redis 实例,然后将渲染后的网站头部模板内容缓存到rendis中,在下一次访问中直接输出。

ps:在此只是给缓存信息一个定期的过期时间,实际应用场景中对cache的处理可能会更复杂。