Code Coverage |
||||||||||
Classes and Traits |
Functions and Methods |
Lines |
||||||||
Total | |
0.00% |
0 / 1 |
|
22.22% |
2 / 9 |
CRAP | |
64.29% |
63 / 98 |
twig | |
0.00% |
0 / 1 |
|
22.22% |
2 / 9 |
103.78 | |
64.29% |
63 / 98 |
__construct | |
0.00% |
0 / 1 |
4.01 | |
92.86% |
13 / 14 |
|||
clear_cache | |
0.00% |
0 / 1 |
2.15 | |
66.67% |
2 / 3 |
|||
get_user_style | |
0.00% |
0 / 1 |
12.00 | |
0.00% |
0 / 6 |
|||
set_style | |
0.00% |
0 / 1 |
110.00 | |
0.00% |
0 / 23 |
|||
set_custom_style | |
0.00% |
0 / 1 |
11.03 | |
93.55% |
29 / 31 |
|||
display | |
0.00% |
0 / 1 |
2.03 | |
80.00% |
4 / 5 |
|||
assign_display | |
100.00% |
1 / 1 |
2 | |
100.00% |
4 / 4 |
|||
get_template_vars | |
100.00% |
1 / 1 |
3 | |
100.00% |
11 / 11 |
|||
get_source_file_for_handle | |
0.00% |
0 / 1 |
2.00 | |
0.00% |
0 / 1 |
<?php | |
/** | |
* | |
* This file is part of the phpBB Forum Software package. | |
* | |
* @copyright (c) phpBB Limited <https://www.phpbb.com> | |
* @license GNU General Public License, version 2 (GPL-2.0) | |
* | |
* For full copyright and license information, please see | |
* the docs/CREDITS.txt file. | |
* | |
*/ | |
namespace phpbb\template\twig; | |
use phpbb\template\exception\user_object_not_available; | |
/** | |
* Twig Template class. | |
*/ | |
class twig extends \phpbb\template\base | |
{ | |
/** | |
* Path of the cache directory for the template | |
* | |
* Cannot be changed during runtime. | |
* | |
* @var string | |
*/ | |
private $cachepath = ''; | |
/** | |
* phpBB path helper | |
* @var \phpbb\path_helper | |
*/ | |
protected $path_helper; | |
/** | |
* phpBB root path | |
* @var string | |
*/ | |
protected $phpbb_root_path; | |
/** | |
* PHP file extension | |
* @var string | |
*/ | |
protected $php_ext; | |
/** | |
* phpBB config instance | |
* @var \phpbb\config\config | |
*/ | |
protected $config; | |
/** | |
* Current user | |
* @var \phpbb\user | |
*/ | |
protected $user; | |
/** | |
* Extension manager. | |
* | |
* @var \phpbb\extension\manager | |
*/ | |
protected $extension_manager; | |
/** | |
* Twig Environment | |
* | |
* @var \Twig\Environment | |
*/ | |
protected $twig; | |
/** | |
* Constructor. | |
* | |
* @param \phpbb\path_helper $path_helper | |
* @param \phpbb\config\config $config | |
* @param \phpbb\template\context $context template context | |
* @param \phpbb\template\twig\environment $twig_environment | |
* @param string $cache_path | |
* @param \phpbb\user|null $user | |
* @param array|\ArrayAccess $extensions | |
* @param \phpbb\extension\manager $extension_manager extension manager, if null then template events will not be invoked | |
*/ | |
public function __construct(\phpbb\path_helper $path_helper, $config, \phpbb\template\context $context, \phpbb\template\twig\environment $twig_environment, $cache_path, \phpbb\user $user = null, $extensions = array(), \phpbb\extension\manager $extension_manager = null) | |
{ | |
$this->path_helper = $path_helper; | |
$this->phpbb_root_path = $path_helper->get_phpbb_root_path(); | |
$this->php_ext = $path_helper->get_php_ext(); | |
$this->config = $config; | |
$this->user = $user; | |
$this->context = $context; | |
$this->extension_manager = $extension_manager; | |
$this->cachepath = $cache_path; | |
$this->twig = $twig_environment; | |
foreach ($extensions as $extension) | |
{ | |
$this->twig->addExtension($extension); | |
} | |
// Add admin namespace | |
if ($this->path_helper->get_adm_relative_path() !== null && is_dir($this->phpbb_root_path . $this->path_helper->get_adm_relative_path() . 'style/')) | |
{ | |
$this->twig->getLoader()->setPaths($this->phpbb_root_path . $this->path_helper->get_adm_relative_path() . 'style/', 'admin'); | |
} | |
} | |
/** | |
* Clear the cache | |
* | |
* @return \phpbb\template\template | |
*/ | |
public function clear_cache() | |
{ | |
if (is_dir($this->cachepath)) | |
{ | |
$this->twig->clearCacheFiles(); | |
} | |
return $this; | |
} | |
/** | |
* Get the style tree of the style preferred by the current user | |
* | |
* @return array Style tree, most specific first | |
* | |
* @throws user_object_not_available When user service was not set | |
*/ | |
public function get_user_style() | |
{ | |
if ($this->user === null) | |
{ | |
throw new user_object_not_available(); | |
} | |
$style_list = array( | |
$this->user->style['style_path'], | |
); | |
if ($this->user->style['style_parent_id']) | |
{ | |
$style_list = array_merge($style_list, array_reverse(explode('/', $this->user->style['style_parent_tree']))); | |
} | |
return $style_list; | |
} | |
/** | |
* Set style location based on (current) user's chosen style. | |
* | |
* @param array $style_directories The directories to add style paths for | |
* E.g. array('ext/foo/bar/styles', 'styles') | |
* Default: array('styles') (phpBB's style directory) | |
* @return \phpbb\template\template $this | |
*/ | |
public function set_style($style_directories = array('styles')) | |
{ | |
if ($style_directories !== array('styles') && $this->twig->getLoader()->getPaths('core') === array()) | |
{ | |
// We should set up the core styles path since not already setup | |
$this->set_style(); | |
} | |
$names = $this->get_user_style(); | |
// Add 'all' folder to $names array | |
// It allows extensions to load a template file from 'all' folder, | |
// if a style doesn't include it. | |
$names[] = 'all'; | |
$paths = array(); | |
foreach ($style_directories as $directory) | |
{ | |
foreach ($names as $name) | |
{ | |
$path = $this->phpbb_root_path . trim($directory, '/') . "/{$name}/"; | |
$template_path = $path . 'template/'; | |
$theme_path = $path . 'theme/'; | |
$is_valid_dir = false; | |
if (is_dir($template_path)) | |
{ | |
$is_valid_dir = true; | |
$paths[] = $template_path; | |
} | |
if (is_dir($theme_path)) | |
{ | |
$is_valid_dir = true; | |
$paths[] = $theme_path; | |
} | |
if ($is_valid_dir) | |
{ | |
// Add the base style directory as a safe directory | |
$this->twig->getLoader()->addSafeDirectory($path); | |
} | |
} | |
} | |
// If we're setting up the main phpBB styles directory and the core | |
// namespace isn't setup yet, we will set it up now | |
if ($style_directories === array('styles') && $this->twig->getLoader()->getPaths('core') === array()) | |
{ | |
// Set up the core style paths namespace | |
$this->twig->getLoader()->setPaths($paths, 'core'); | |
} | |
$this->set_custom_style($names, $paths); | |
return $this; | |
} | |
/** | |
* Set custom style location (able to use directory outside of phpBB). | |
* | |
* Note: Templates are still compiled to phpBB's cache directory. | |
* | |
* @param string|array $names Array of names (or detailed names) or string of name of template(s) in inheritance tree order, used by extensions. | |
* E.g. array( | |
* 'name' => 'adm', | |
* 'ext_path' => 'adm/style/', | |
* ) | |
* @param string|array $paths Array of style paths, relative to current root directory | |
* @return \phpbb\template\template $this | |
*/ | |
public function set_custom_style($names, $paths) | |
{ | |
$paths = (is_string($paths)) ? array($paths) : $paths; | |
$names = (is_string($names)) ? array($names) : $names; | |
// Set as __main__ namespace | |
$this->twig->getLoader()->setPaths($paths); | |
// Add all namespaces for all extensions | |
if ($this->extension_manager instanceof \phpbb\extension\manager) | |
{ | |
$names[] = 'all'; | |
foreach ($this->extension_manager->all_enabled() as $ext_namespace => $ext_path) | |
{ | |
// namespaces cannot contain / | |
$namespace = str_replace('/', '_', $ext_namespace); | |
$paths = array(); | |
foreach ($names as $template_dir) | |
{ | |
if (is_array($template_dir)) | |
{ | |
if (isset($template_dir['ext_path'])) | |
{ | |
$ext_style_template_path = $ext_path . $template_dir['ext_path']; | |
$ext_style_path = dirname($ext_style_template_path); | |
$ext_style_theme_path = $ext_style_path . 'theme/'; | |
} | |
else | |
{ | |
$ext_style_path = $ext_path . 'styles/' . $template_dir['name'] . '/'; | |
$ext_style_template_path = $ext_style_path . 'template/'; | |
$ext_style_theme_path = $ext_style_path . 'theme/'; | |
} | |
} | |
else | |
{ | |
$ext_style_path = $ext_path . 'styles/' . $template_dir . '/'; | |
$ext_style_template_path = $ext_style_path . 'template/'; | |
$ext_style_theme_path = $ext_style_path . 'theme/'; | |
} | |
$is_valid_dir = false; | |
if (is_dir($ext_style_template_path)) | |
{ | |
$is_valid_dir = true; | |
$paths[] = $ext_style_template_path; | |
} | |
if (is_dir($ext_style_theme_path)) | |
{ | |
$is_valid_dir = true; | |
$paths[] = $ext_style_theme_path; | |
} | |
if ($is_valid_dir) | |
{ | |
// Add the base style directory as a safe directory | |
$this->twig->getLoader()->addSafeDirectory($ext_style_path); | |
} | |
} | |
$this->twig->getLoader()->setPaths($paths, $namespace); | |
} | |
} | |
return $this; | |
} | |
/** | |
* Display a template for provided handle. | |
* | |
* The template will be loaded and compiled, if necessary, first. | |
* | |
* This function calls hooks. | |
* | |
* @param string $handle Handle to display | |
* @return \phpbb\template\template $this | |
*/ | |
public function display($handle) | |
{ | |
$result = $this->call_hook($handle, __FUNCTION__); | |
if ($result !== false) | |
{ | |
return $result[0]; | |
} | |
$this->twig->display($this->get_filename_from_handle($handle), $this->get_template_vars()); | |
return $this; | |
} | |
/** | |
* Display the handle and assign the output to a template variable | |
* or return the compiled result. | |
* | |
* @param string $handle Handle to operate on | |
* @param string $template_var Template variable to assign compiled handle to | |
* @param bool $return_content If true return compiled handle, otherwise assign to $template_var | |
* @return \phpbb\template\template|string if $return_content is true return string of the compiled handle, otherwise return $this | |
*/ | |
public function assign_display($handle, $template_var = '', $return_content = true) | |
{ | |
if ($return_content) | |
{ | |
return $this->twig->render($this->get_filename_from_handle($handle), $this->get_template_vars()); | |
} | |
$this->assign_var($template_var, $this->twig->render($this->get_filename_from_handle($handle), $this->get_template_vars())); | |
return $this; | |
} | |
/** | |
* Get template vars in a format Twig will use (from the context) | |
* | |
* @return array | |
*/ | |
protected function get_template_vars() | |
{ | |
$context_vars = $this->context->get_data_ref(); | |
$vars = array_merge( | |
$context_vars['.'][0], // To get normal vars | |
array( | |
'definition' => new \phpbb\template\twig\definition(), | |
'loops' => $context_vars, // To get loops | |
) | |
); | |
if ($this->user instanceof \phpbb\user) | |
{ | |
$vars['user'] = $this->user; | |
} | |
// cleanup | |
unset($vars['loops']['.']); | |
// Inject in the main context the value added by assign_block_vars() to be able to use directly the Twig loops. | |
foreach ($vars['loops'] as $key => &$value) | |
{ | |
$vars[$key] = $value; | |
} | |
return $vars; | |
} | |
/** | |
* {@inheritdoc} | |
*/ | |
public function get_source_file_for_handle($handle) | |
{ | |
return $this->twig->getLoader()->getCacheKey($this->get_filename_from_handle($handle)); | |
} | |
} |