mirror of
				https://github.com/phpbb/phpbb.git
				synced 2025-10-22 04:07:04 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			324 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			324 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?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\extension;
 | |
| 
 | |
| use phpbb\template\twig\environment;
 | |
| use Twig\Extension\AbstractExtension;
 | |
| 
 | |
| class icon extends AbstractExtension
 | |
| {
 | |
| 	/** @var \phpbb\user */
 | |
| 	protected $user;
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor.
 | |
| 	 *
 | |
| 	 * @param \phpbb\user	$user			User object
 | |
| 	 */
 | |
| 	public function __construct(\phpbb\user $user)
 | |
| 	{
 | |
| 		$this->user = $user;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Returns the name of this extension.
 | |
| 	 *
 | |
| 	 * @return string						The extension name
 | |
| 	 */
 | |
| 	public function getName()
 | |
| 	{
 | |
| 		return 'icon';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Returns a list of functions to add to the existing list.
 | |
| 	 *
 | |
| 	 * @return \Twig\TwigFunction[]			Array of twig functions
 | |
| 	 */
 | |
| 	public function getFunctions()
 | |
| 	{
 | |
| 		return [
 | |
| 			new \Twig\TwigFunction('Icon', [$this, 'icon'], ['needs_environment' => true]),
 | |
| 		];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Generate icon HTML for use in the template, depending on the mode.
 | |
| 	 *
 | |
| 	 * @param environment	$environment	Twig environment object
 | |
| 	 * @param string		$type			Icon type (font|iconify|png|svg)
 | |
| 	 * @param string		$icon			Icon name (eg. "bold")
 | |
| 	 * @param string		$title			Icon title
 | |
| 	 * @param bool			$hidden			Hide the icon title from view
 | |
| 	 * @param string		$classes		Additional classes (eg. "fa-fw")
 | |
| 	 * @param array			$attributes		Additional attributes for the icon, where the key is the attribute.
 | |
| 	 *                      				{'data-ajax': 'mark_forums'} results in ' data-ajax="mark_forums"'
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function icon(environment $environment, $type, $icon, $title = '', $hidden = false, $classes = '', array $attributes = [])
 | |
| 	{
 | |
| 		$type = strtolower($type);
 | |
| 		$icon = is_array($icon) ? $this->get_first_icon($icon) : $icon;
 | |
| 
 | |
| 		if (empty($icon))
 | |
| 		{
 | |
| 			return '';
 | |
| 		}
 | |
| 
 | |
| 		$not_found	= false;
 | |
| 		$source		= '';
 | |
| 		$view_box	= '';
 | |
| 
 | |
| 		switch ($type)
 | |
| 		{
 | |
| 			case 'font':
 | |
| 				// Nothing to do here..
 | |
| 			break;
 | |
| 
 | |
| 			case 'iconify':
 | |
| 				$source = explode(':', $icon);
 | |
| 				$source = $source[0];
 | |
| 			break;
 | |
| 
 | |
| 			case 'png':
 | |
| 				$filesystem	= $environment->get_filesystem();
 | |
| 				$root_path	= $environment->get_web_root_path();
 | |
| 
 | |
| 				$board_url	= defined('PHPBB_USE_BOARD_URL_PATH') && PHPBB_USE_BOARD_URL_PATH;
 | |
| 				$base_path	= $board_url ? generate_board_url() . '/' : $root_path;
 | |
| 
 | |
| 				// Iterate over the user's styles and check for icon existance
 | |
| 				foreach ($this->get_style_list() as $style_path)
 | |
| 				{
 | |
| 					if ($filesystem->exists("{$root_path}styles/{$style_path}/theme/png/{$icon}.png"))
 | |
| 					{
 | |
| 						$source = "{$base_path}styles/{$style_path}/theme/png/{$icon}.png";
 | |
| 
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// Check if the icon was found or not
 | |
| 				$not_found = empty($source);
 | |
| 			break;
 | |
| 
 | |
| 			case 'svg':
 | |
| 				try
 | |
| 				{
 | |
| 					// Try to load and prepare the SVG icon
 | |
| 					$file	= $environment->load('svg/' . $icon . '.svg');
 | |
| 					$source	= $this->prepare_svg($file, $view_box);
 | |
| 
 | |
| 					if (empty($view_box))
 | |
| 					{
 | |
| 						return '';
 | |
| 					}
 | |
| 				}
 | |
| 				catch (\Twig\Error\LoaderError $e)
 | |
| 				{
 | |
| 					// Icon was not found
 | |
| 					$not_found = true;
 | |
| 				}
 | |
| 				catch (\Twig\Error\Error $e)
 | |
| 				{
 | |
| 					return '';
 | |
| 				}
 | |
| 			break;
 | |
| 
 | |
| 			default:
 | |
| 				return '';
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		// If no PNG or SVG icon was found, display a default 404 SVG icon.
 | |
| 		if ($not_found)
 | |
| 		{
 | |
| 			try
 | |
| 			{
 | |
| 				$file	= $environment->load('svg/404.svg');
 | |
| 				$source	= $this->prepare_svg($file, $view_box);
 | |
| 			}
 | |
| 			catch (\Twig\Error\Error $e)
 | |
| 			{
 | |
| 				return '';
 | |
| 			}
 | |
| 
 | |
| 			$type = 'svg';
 | |
| 			$icon = '404';
 | |
| 		}
 | |
| 
 | |
| 		try
 | |
| 		{
 | |
| 			return $environment->render("macros/icons/{$type}.twig", [
 | |
| 				'ATTRIBUTES'	=> (string) $this->implode_attributes($attributes),
 | |
| 				'CLASSES'		=> (string) $classes,
 | |
| 				'ICON'			=> (string) $icon,
 | |
| 				'SOURCE'		=> (string) $source,
 | |
| 				'TITLE'			=> (string) $title,
 | |
| 				'TITLE_ID'		=> $title && $type === 'svg' ? unique_id() : '',
 | |
| 				'VIEW_BOX'		=> (string) $view_box,
 | |
| 				'S_HIDDEN'		=> (bool) $hidden,
 | |
| 			]);
 | |
| 		}
 | |
| 		catch (\Twig\Error\Error $e)
 | |
| 		{
 | |
| 			return '';
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Prepare an SVG for usage in the template icon.
 | |
| 	 *
 | |
| 	 * This removes any <?xml ?> and <!DOCTYPE> elements,
 | |
| 	 * aswell as the root <svg> and any <title> elements.
 | |
| 	 *
 | |
| 	 * @param \Twig\TemplateWrapper	$file		The SVG file loaded from the environment
 | |
| 	 * @param string				$view_box	The viewBox attribute value
 | |
| 	 * @return string							The cleaned SVG
 | |
| 	 */
 | |
| 	protected function prepare_svg(\Twig\TemplateWrapper $file, &$view_box = '')
 | |
| 	{
 | |
| 		$code = $file->render();
 | |
| 		$code = preg_replace( "/<\?xml.+?\?>/", '', $code);
 | |
| 
 | |
| 		$doc = new \DOMDocument();
 | |
| 		$doc->preserveWhiteSpace = false;
 | |
| 
 | |
| 		// Hide html5/svg errors
 | |
| 		libxml_use_internal_errors(true);
 | |
| 
 | |
| 		// Options parameter prevents $dom->saveHTML() from adding an <html> element.
 | |
| 		$doc->loadHTML($code, LIBXML_HTML_NOIMPLIED | LIBXML_HTML_NODEFDTD);
 | |
| 
 | |
| 		// Remove any DOCTYPE
 | |
| 		foreach ($doc->childNodes as $child)
 | |
| 		{
 | |
| 			if ($child->nodeType === XML_DOCUMENT_TYPE_NODE)
 | |
| 			{
 | |
| 				$child->parentNode->removeChild($child);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$xpath = new \DOMXPath($doc);
 | |
| 
 | |
| 		/**
 | |
| 		 * Remove the root <svg> element
 | |
| 		 * and all <title> elements.
 | |
| 		 *
 | |
| 		 * @var \DOMElement $element
 | |
| 		 */
 | |
| 		foreach ($xpath->query('/svg | //title') as $element)
 | |
| 		{
 | |
| 			if ($element->nodeName === 'svg')
 | |
| 			{
 | |
| 				// Return the viewBox attribute value of the root SVG element by reference
 | |
| 				$view_box = $element->getAttribute('viewbox');
 | |
| 
 | |
| 				$width = $element->getAttribute('width');
 | |
| 				$height = $element->getAttribute('height');
 | |
| 
 | |
| 				if (empty($view_box) && $width && $height)
 | |
| 				{
 | |
| 					$view_box = "0 0 {$width} {$height}";
 | |
| 				}
 | |
| 
 | |
| 				while (isset($element->firstChild))
 | |
| 				{
 | |
| 					$element->parentNode->insertBefore($element->firstChild, $element);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			$element->parentNode->removeChild($element);
 | |
| 		}
 | |
| 
 | |
| 		$string = $doc->saveHTML();
 | |
| 		$string = preg_replace('/\s+/', ' ', $string);
 | |
| 
 | |
| 		return $string;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Finds the first icon that has a "true" value and returns it.
 | |
| 	 *
 | |
| 	 * This allows sending an array to the Icon() function,
 | |
| 	 * where the keys are the icon names and the values are their checks.
 | |
| 	 *
 | |
| 	 * {{ Icon('font', {
 | |
| 	 * 		'bullhorn': topicrow.S_POST_GLOBAL or topicrow.S_POST_ANNOUNCE,
 | |
| 	 * 		'star': topicrow.S_POST_STICKY,
 | |
| 	 * 		'lock': topicrow.S_TOPIC_LOCKED,
 | |
| 	 * 		'fire': topicrow.S_TOPIC_HOT,
 | |
| 	 * 		'file': true,
 | |
| 	 * }, lang('MY_TITLE'), true) }}
 | |
| 	 *
 | |
| 	 * @param array		$icons			Array of icons and their booleans
 | |
| 	 * @return string					The first 'true' icon
 | |
| 	 */
 | |
| 	protected function get_first_icon(array $icons)
 | |
| 	{
 | |
| 		foreach ($icons as $icon => $boolean)
 | |
| 		{
 | |
| 			// In case the key is not a string,
 | |
| 			// this icon does not have a check
 | |
| 			// so instantly return it
 | |
| 			if (!is_string($icon))
 | |
| 			{
 | |
| 				return $boolean;
 | |
| 			}
 | |
| 
 | |
| 			if ($boolean)
 | |
| 			{
 | |
| 				return $icon;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return '';
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Implode an associated array of attributes to a string for usage in a template.
 | |
| 	 *
 | |
| 	 * @param array		$attributes		Associated array of attributes
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	protected function implode_attributes(array $attributes)
 | |
| 	{
 | |
| 		$string = '';
 | |
| 
 | |
| 		foreach ($attributes as $key => $value)
 | |
| 		{
 | |
| 			$string .= ' ' . $key . '="' . $value . '"';
 | |
| 		}
 | |
| 
 | |
| 		return $string;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the style tree of the style preferred by the current user.
 | |
| 	 *
 | |
| 	 * @return array					Style tree, most specific first
 | |
| 	 */
 | |
| 	protected function get_style_list()
 | |
| 	{
 | |
| 		$style_list = [$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;
 | |
| 	}
 | |
| }
 |