You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							232 lines
						
					
					
						
							6.4 KiB
						
					
					
				
			
		
		
	
	
							232 lines
						
					
					
						
							6.4 KiB
						
					
					
				<?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."); | 
						|
			} | 
						|
		} | 
						|
	} | 
						|
}
 | 
						|
 |