|  |  |  | <?php
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Twig view renderer class file.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * @link http://www.yiiframework.com/
 | 
					
						
							|  |  |  |  * @copyright Copyright © 2008 Yii Software LLC
 | 
					
						
							|  |  |  |  * @license http://www.yiiframework.com/license/
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace yii\twig;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Yii;
 | 
					
						
							|  |  |  | use yii\base\View;
 | 
					
						
							|  |  |  | use yii\base\ViewRenderer as BaseViewRenderer;
 | 
					
						
							|  |  |  | use yii\helpers\Html;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * TwigViewRenderer allows you to use Twig templates in views.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * @author Alexander Makarov <sam@rmcreative.ru>
 | 
					
						
							|  |  |  |  * @since 2.0
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | class ViewRenderer extends BaseViewRenderer
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * @var string the directory or path alias pointing to where Twig cache will be stored.
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $cachePath = '@runtime/Twig/cache';
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * @var array Twig options.
 | 
					
						
							|  |  |  | 	 * @see http://twig.sensiolabs.org/doc/api.html#environment-options
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $options = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @var array Objects or static classes.
 | 
					
						
							|  |  |  |      * Keys of the array are names to call in template, values are objects or names of static classes.
 | 
					
						
							|  |  |  |      * Example: `['html' => '\yii\helpers\Html']`.
 | 
					
						
							|  |  |  |      * In the template you can use it like this: `{{ html.a('Login', 'site/login') | raw }}`.
 | 
					
						
							|  |  |  |      */
 | 
					
						
							|  |  |  |     public $globals = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @var array Custom functions.
 | 
					
						
							|  |  |  |      * Keys of the array are names to call in template, values are names of functions or static methods of some class.
 | 
					
						
							|  |  |  |      * Example: `['rot13' => 'str_rot13', 'a' => '\yii\helpers\Html::a']`.
 | 
					
						
							|  |  |  |      * In the template you can use it like this: `{{ rot13('test') }}` or `{{ a('Login', 'site/login') | raw }}`.
 | 
					
						
							|  |  |  |      */
 | 
					
						
							|  |  |  |     public $functions = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @var array Custom filters.
 | 
					
						
							|  |  |  |      * Keys of the array are names to call in template, values are names of functions or static methods of some class.
 | 
					
						
							|  |  |  |      * Example: `['rot13' => 'str_rot13', 'jsonEncode' => '\yii\helpers\Json::encode']`.
 | 
					
						
							|  |  |  |      * In the template you can use it like this: `{{ 'test'|rot13 }}` or `{{ model|jsonEncode }}`.
 | 
					
						
							|  |  |  |      */
 | 
					
						
							|  |  |  |     public $filters = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @var array Custom extensions.
 | 
					
						
							|  |  |  |      * Example: `['Twig_Extension_Sandbox', 'Twig_Extension_Text']`
 | 
					
						
							|  |  |  |      */
 | 
					
						
							|  |  |  |     public $extensions = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * @var array Twig lexer options.
 | 
					
						
							|  |  |  |      * Example: Smarty-like syntax:
 | 
					
						
							|  |  |  |      * ```php
 | 
					
						
							|  |  |  |      * [
 | 
					
						
							|  |  |  |      *     'tag_comment'  => ['{*', '*}'],
 | 
					
						
							|  |  |  |      *     'tag_block'    => ['{', '}'],
 | 
					
						
							|  |  |  |      *     'tag_variable' => ['{$', '}']
 | 
					
						
							|  |  |  |      * ]
 | 
					
						
							|  |  |  |      * ```
 | 
					
						
							|  |  |  |      * @see http://twig.sensiolabs.org/doc/recipes.html#customizing-the-syntax
 | 
					
						
							|  |  |  |      */
 | 
					
						
							|  |  |  |     public $lexerOptions = [];
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  | 	 * @var \Twig_Environment twig environment object that do all rendering twig templates
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public $twig;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	public function init()
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$this->twig = new \Twig_Environment(null, array_merge([
 | 
					
						
							|  |  |  | 			'cache' => Yii::getAlias($this->cachePath),
 | 
					
						
							|  |  |  | 			'charset' => Yii::$app->charset,
 | 
					
						
							|  |  |  | 		], $this->options));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding custom extensions
 | 
					
						
							|  |  |  | 		if (!empty($this->extensions)) {
 | 
					
						
							|  |  |  | 			foreach ($this->extensions as $extension) {
 | 
					
						
							|  |  |  | 				$this->twig->addExtension(new $extension());
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding custom globals (objects or static classes)
 | 
					
						
							|  |  |  | 		if (!empty($this->globals)) {
 | 
					
						
							|  |  |  | 			$this->addGlobals($this->globals);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding custom functions
 | 
					
						
							|  |  |  | 		if (!empty($this->functions)) {
 | 
					
						
							|  |  |  | 			$this->addFunctions($this->functions);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding custom filters
 | 
					
						
							|  |  |  | 		if (!empty($this->filters)) {
 | 
					
						
							|  |  |  | 			$this->addFilters($this->filters);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding custom extensions
 | 
					
						
							|  |  |  | 		if (!empty($this->extensions)) {
 | 
					
						
							|  |  |  | 			$this->addExtensions($this->extensions);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Change lexer syntax
 | 
					
						
							|  |  |  | 		if (!empty($this->lexerOptions)) {
 | 
					
						
							|  |  |  | 			$this->setLexerOptions($this->lexerOptions);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Adding global 'void' function (usage: {{void(App.clientScript.registerScriptFile(...))}})
 | 
					
						
							|  |  |  | 		$this->twig->addFunction('void', new \Twig_Function_Function(function($argument){
 | 
					
						
							|  |  |  | 		}));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		$this->twig->addFunction('path', new \Twig_Function_Function(function ($path, $args = []) {
 | 
					
						
							|  |  |  | 			return Html::url(array_merge([$path], $args));
 | 
					
						
							|  |  |  | 		}));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		$this->twig->addGlobal('app', \Yii::$app);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Renders a view file.
 | 
					
						
							|  |  |  | 	 *
 | 
					
						
							|  |  |  | 	 * This method is invoked by [[View]] whenever it tries to render a view.
 | 
					
						
							|  |  |  | 	 * Child classes must implement this method to render the given view file.
 | 
					
						
							|  |  |  | 	 *
 | 
					
						
							|  |  |  | 	 * @param View $view the view object used for rendering the file.
 | 
					
						
							|  |  |  | 	 * @param string $file the view file.
 | 
					
						
							|  |  |  | 	 * @param array $params the parameters to be passed to the view file.
 | 
					
						
							|  |  |  | 	 *
 | 
					
						
							|  |  |  | 	 * @return string the rendering result
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function render($view, $file, $params)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$this->twig->addGlobal('this', $view);
 | 
					
						
							|  |  |  | 		$this->twig->setLoader(new TwigSimpleFileLoader(dirname($file)));
 | 
					
						
							|  |  |  | 		return $this->twig->render(pathinfo($file, PATHINFO_BASENAME), $params);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Adds global objects or static classes
 | 
					
						
							|  |  |  | 	 * @param array $globals @see self::$globals
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function addGlobals($globals)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		foreach ($globals as $name => $value) {
 | 
					
						
							|  |  |  | 			if (!is_object($value)) {
 | 
					
						
							|  |  |  | 				$value = new ViewRendererStaticClassProxy($value);
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 			$this->twig->addGlobal($name, $value);
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  |  	 * Adds custom functions
 | 
					
						
							|  |  |  | 	 * @param array $functions @see self::$functions
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function addFunctions($functions)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$this->_addCustom('Function', $functions);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Adds custom filters
 | 
					
						
							|  |  |  | 	 * @param array $filters @see self::$filters
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function addFilters($filters)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$this->_addCustom('Filter', $filters);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Adds custom extensions
 | 
					
						
							|  |  |  | 	 * @param array $extensions @see self::$extensions
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function addExtensions($extensions)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		foreach ($extensions as $extName) {
 | 
					
						
							|  |  |  | 			$this->twig->addExtension(new $extName());
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Sets Twig lexer options to change templates syntax
 | 
					
						
							|  |  |  | 	 * @param array $options @see self::$lexerOptions
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	public function setLexerOptions($options)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$lexer = new \Twig_Lexer($this->twig, $options);
 | 
					
						
							|  |  |  | 		$this->twig->setLexer($lexer);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * Adds custom function or filter
 | 
					
						
							|  |  |  | 	 * @param string $classType 'Function' or 'Filter'
 | 
					
						
							|  |  |  | 	 * @param array $elements Parameters of elements to add
 | 
					
						
							|  |  |  | 	 * @throws \Exception
 | 
					
						
							|  |  |  | 	 */
 | 
					
						
							|  |  |  | 	private function _addCustom($classType, $elements)
 | 
					
						
							|  |  |  | 	{
 | 
					
						
							|  |  |  | 		$classFunction = 'Twig_' . $classType . '_Function';
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		foreach ($elements as $name => $func) {
 | 
					
						
							|  |  |  | 			$twigElement = null;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			switch ($func) {
 | 
					
						
							|  |  |  | 				// Just a name of function
 | 
					
						
							|  |  |  | 				case is_string($func):
 | 
					
						
							|  |  |  | 					$twigElement = new $classFunction($func);
 | 
					
						
							|  |  |  | 					break;
 | 
					
						
							|  |  |  | 				// Name of function + options array
 | 
					
						
							|  |  |  | 				case is_array($func) && is_string($func[0]) && isset($func[1]) && is_array($func[1]):
 | 
					
						
							|  |  |  | 					$twigElement = new $classFunction($func[0], $func[1]);
 | 
					
						
							|  |  |  | 					break;
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if ($twigElement !== null) {
 | 
					
						
							|  |  |  | 				$this->twig->{'add'.$classType}($name, $twigElement);
 | 
					
						
							|  |  |  | 			} else {
 | 
					
						
							|  |  |  | 				throw new \Exception("Incorrect options for \"$classType\" $name.");
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 |