View file PF.Base/include/library/phpfox/template/template.class.php

File size: 97.7Kb
<?php
/**
 * [PHPFOX_HEADER]
 */

defined('PHPFOX') or exit('NO DICE!');

/**
 * Template
 * Loads all templates and converts it into PHP code and then caches it.
 *
 * Class is also able to:
 * - Assign variables to templates.
 * - Identify a pages title.
 * - Identify a pages breadcrumb structure.
 * - Create meta tags.
 * - Load CSS and JavaScript files.
 *
 * @copyright		[PHPFOX_COPYRIGHT]
 * @author			Raymond Benc
 * @package 		Phpfox
 * @version 		$Id: template.class.php 7317 2014-05-09 17:38:54Z Fern $
 */

class Phpfox_Template
{

    const BUNDLE_CSS_FILES =  'PHPFOX_BUNDLE_CSS_FILES';

	/**
	 * Default template name.
	 *
	 * @var string
	 */
	public $sDisplayLayout = 'template';

	/**
	 * Check to see if we are displaying a sample page.
	 *
	 * @var bool
	 */
	public $bIsSample = false;

	/**
	 * Theme ID#
	 *
	 * @var int
	 */
	public $iThemeId = 0;

	/**
	 * Reserved variable name. Which is $phpfox.
	 *
	 * @var string
	 */
	protected $sReservedVarname = 'phpfox';

	/**
	 * Left delimiter for custom functions. It is: {
	 *
	 * @var string
	 */
	protected $sLeftDelim = '{';

	/**
	 * Right delimiter for custom functions. It is: }
	 *
	 * @var string
	 */
	protected $sRightDelim = '}';

	/**
	 * List of plugins.
	 *
	 * @var array
	 */
	protected $_aPlugins = array();

	/**
	 * List of sections.
	 *
	 * @var array
	 */
	private $_aSections = array();

	/**
	 * List of all the variables assigned to templates.
	 *
	 * @var array
	 */
	private $_aVars = array('bUseFullSite' => false);

	/**
	 * List of titles assigned to a page.
	 *
	 * @var array
	 */
	private $_aTitles = array();

	/**
	 * List of data to add within the templates HTML <head></head>.
	 *
	 * @var array
	 */
	private $_aHeaders = array();

	/**
	 * List of breadcrumbs.
	 *
	 * @var array
	 */
	private $_aBreadCrumbs = array();

	/**
	 * Information about the title of the current page, which is part of the breadcrumb.
	 *
	 * @var array
	 */
	private $_aBreadCrumbTitle = array();

	/**
	 * Default file cache time.
	 *
	 * @var int
	 */
	private $_iCacheTime = 60;

	/**
	 * Override the layout of the current theme being used.
	 *
	 * @var bool
	 */
	private $_sSetLayout = false;

	/**
	 * Check to see if a template is part of the AdminCP.
	 *
	 * @var bool
	 */
	private $_bIsAdminCp = false;

	/**
	 * Folder of the theme being used.
	 *
	 * @var string
	 */
	private $_sThemeFolder;

	/**
	 * Theme layout to load.
	 *
	 * @var string
	 */
	private $_sThemeLayout;

	/**
	 * Folder of the style being used.
	 *
	 * @var string
	 */
	private $_sStyleFolder;

	/**
	 * List of meta data.
	 *
	 * @var array
	 */
	private $_aMeta = array();

	/**
	 * List of phrases to load and create JavaScript variables for.
	 *
	 * @var array
	 */
	private $_aPhrases = array();

	/**
	 * Information about the text editor.
	 *
	 * @var array
	 */
	private $_aEditor = array();

	/**
	 * URL of the current page we are on.
	 *
	 * @var string
	 */
	private $_sUrl = null;

	/**
	 * Information about the current theme we are using.
	 *
	 * @var array
	 */
	private $_aTheme = array(
			'theme_parent_id' => 0
	);

	/**
	 * Rebuild URL brought from cache.
	 *
	 * @var array
	 */
	private $_aNewUrl = array();

	/**
	 * Remove URL brought from cache.
	 *
	 * @var array
	 */
	private $_aRemoveUrl = array();

	/**
	 * List of images to be loaded and converted into a JavaScript object.
	 *
	 * @var array
	 */
	private $_aImages = array();

	/**
	 * Cache of all the <head></head> content being loaded.
	 *
	 * @var array
	 */
	private $_aCacheHeaders = array();

	/**
	 * Check to see if we are currently in test mode.
	 *
	 * @var bool
	 */
	private $_bIsTestMode = false;

	/**
	 * Mobile headers.
	 *
	 * @var array
	 */
	private $_aMobileHeaders = array();

	/**
	 * Holds section menu information
	 *
	 * @var array
	 */
	private $_aSectionMenu = array();


	private $_sFooter = '';

	/**
	 * @var array
	 */
	private $_aDirectoryNames = [];

	/**
	 * Static variable of the current theme folder.
	 *
	 * @static
	 * @var string
	 */
	protected static $_sStaticThemeFolder = null;

	private $_theme;
	private $_meta;
	private $_keepBody = false;
	private $_subMenu = [];
	private $_loadedHeader = false;

	public $delayedHeaders = [];

	/**
	 * Class constructor we use to build the current theme and style
	 * we are using.
	 *
	 */
	public function __construct()
	{
		$this->_sThemeLayout = 'frontend';
		$this->_sThemeFolder = 'default';
		$this->_sStyleFolder = 'default';

		if (defined('PHPFOX_INSTALLER'))
		{
			$this->_sThemeLayout = 'install';
		}
		else
		{
			$this->_theme = new Core\Theme();
			$this->_bIsAdminCp = (strtolower(Phpfox_Request::instance()->get('req1')) == Phpfox::getParam('admincp.admin_cp'));

			$theme =  $this->_theme->get();

			if(!empty($theme->folder)){
				$this->_sThemeFolder =  $theme->folder;
			}

			if ($this->_bIsAdminCp)
			{
				$this->_sThemeLayout = 'adminpanel';
				$this->_sThemeFolder = Phpfox::getParam('core.default_theme_name');
				$this->_sStyleFolder = Phpfox::getParam('core.default_style_name');
			}
		}

		self::$_sStaticThemeFolder = $this->_sThemeFolder;
	}

	/**
	 * @return Phpfox_Template
	 */
	public static function instance()
	{
		return Phpfox::getLib('template');
	}

	/**
	 * @param array $dirs add dirs
	 *
	 * @return $this
	 */
	public function addDirectoryNames($dirs)
	{
		foreach($dirs as $name=>$dir){
			$this->_aDirectoryNames[$name]= $dir;
		}

		return $this;
	}

    public function getDirectoryNames()
    {
        return $this->_aDirectoryNames;
	}

	/**
	 * Sets all the images we plan on using within JavaScript.
	 *
	 * PHP usage:
	 * <code>
	 * Phpfox_Template::instance()->setImage(array('layout_sample_image', 'layout/sample.png'));
	 * </code>
	 *
	 * In JavaScript the above image can be accessed by:
	 * <code>
	 * oJsImages['layout_sample_image'];
	 * </code>
	 *
	 * @param array $aImages
	 * @return object
	 */
	public function setImage($aImages)
	{
		foreach ($aImages as $sKey => $sImage)
		{
			$this->_aImages[$sKey] = $this->getStyle('image', $sImage);
		}

		return $this;
	}

    /**
     * @return array
     */
    public function getImages()
    {
        return $this->_aImages;
    }


	/**
	 * @return \Core\Theme
	 */
	public function theme() {
		return $this->_theme;
	}

	/**
	 * Get the current theme we are using.
	 *
	 * @return string
	 */
	public function getThemeLayout()
	{
		return $this->_sThemeLayout;
	}

	/**
	 * Get the cached information about the theme we are using.
	 *
	 * @return array
	 */
	public function getThemeCache()
	{
		return $this->_aTheme;
	}

	/**
	 * Override the current theme.
	 *
	 * @param array $aTheme ARRAY of values to override.
	 */
	public function setStyle($aTheme)
	{
		$this->_sThemeFolder = $aTheme['theme_folder_name'];
		$this->_sStyleFolder = $aTheme['style_folder_name'];
		$this->_aTheme = $aTheme;

		self::$_sStaticThemeFolder = $this->_sThemeFolder;
	}

	/**
	 * Get all the information regarding the theme/style we are using.
	 *
	 * @return array
	 */
	public function getStyleInUse()
	{
		return $this->_aTheme;
	}

	/**
	 * Get the total number of columns this template supports.
	 *
	 * @return int
	 */
	public function columns()
	{
		return (int) (isset($this->_aTheme['total_column']) ? $this->_aTheme['total_column'] : 3);
	}

	/**
	 * Test a style by attempting to load and display it for the user.
	 * This is used when a user is trying to demo a style.
	 *
	 * @param int $iId ID of the style.
	 * @return bool TRUE if style can be loaded, FALSE if not.
	 */
	public function testStyle($iId = null)
	{
		$sWhere = '';
		if ($iId === null)
		{
			$sWhere = 't.is_default = 1 AND s.is_default = 1';
		}
		else
		{
			$sWhere = 's.style_id = ' . (int) $iId;
		}
		$aTheme = Phpfox_Database::instance()->select('s.style_id, s.parent_id AS style_parent_id, s.folder AS style_folder_name, t.folder AS theme_folder_name, t.parent_id AS theme_parent_id')
				->from(Phpfox::getT('theme_style'), 's')
				->join(Phpfox::getT('theme'), 't', 't.theme_id = s.theme_id')
				->where($sWhere)
				->execute('getRow');

		if (!isset($aTheme['style_id']))
		{
			return false;
		}

		$this->_sThemeFolder = $aTheme['theme_folder_name'];
		$this->_sStyleFolder = $aTheme['style_folder_name'];

		if ($aTheme['style_parent_id'] > 0)
		{
			$aStyleExtend = Phpfox_Database::instance()->select('folder AS parent_style_folder')
					->from(Phpfox::getT('theme_style'))
					->where('style_id = ' . $aTheme['style_parent_id'])
					->execute('getRow');

			if (isset($aStyleExtend['parent_style_folder']))
			{
				$aTheme['parent_style_folder'] = $aStyleExtend['parent_style_folder'];
			}
		}

		if ($aTheme['theme_parent_id'] > 0)
		{
			$aThemeExtend = Phpfox_Database::instance()->select('folder AS parent_theme_folder')
					->from(Phpfox::getT('theme'))
					->where('theme_id = ' . $aTheme['theme_parent_id'])
					->execute('getRow');

			if (isset($aThemeExtend['parent_theme_folder']))
			{
				$aTheme['parent_theme_folder'] = $aThemeExtend['parent_theme_folder'];
			}
		}

		$this->_aTheme = $aTheme;
		$this->_bIsTestMode = true;

		self::$_sStaticThemeFolder = $this->_sThemeFolder;

		return true;
	}

	/**
	 * Get the theme folder being used.
	 *
	 * @return string
	 */
	public function getThemeFolder()
	{
		return $this->_sThemeFolder;
	}

	/**
	 * Get the parent theme folder being used.
	 * Issue:  http://www.phpfox.com/tracker/view/15384/
	 *
	 * @return string
	 */
	public function getParentThemeFolder()
	{
		if(isset($this->_aTheme['parent_theme_folder']))
		{
			return $this->_aTheme['parent_theme_folder'];
		}
		else
		{
			return $this->_sThemeFolder;
		}
	}

	/**
	 * Get the style folder being used.
	 *
	 * @return string
	 */
	public function getStyleFolder()
	{
		return $this->_sStyleFolder;
	}

	/**
	 * Get the logo for the site based on the style being used.
	 *
	 * @return string
	 */
	public function getStyleLogo()
	{
		return '';
	}

	/**
	 * Override the layout of the site.
	 *
	 * @param string $sName Layout we should load.
	 */
	public function setLayout($sName)
	{
		$this->_sSetLayout = $sName;
	}

	/**
     * @deprecated
	 * Force the page to use its full width and not display anything within the sidepanel.
	 *
	 * @return $this
	 */
	public function setFullSite()
	{
		$this->assign(array('bUseFullSite' => true));

		return $this;
	}

	/**
	 * Sets phrases we can later use in JavaScript.
	 *
	 * @param array $mPhrases ARRAY of phrase to build.
	 * @return $this
	 */
	public function setPhrase($mPhrases)
	{
		foreach ($mPhrases as $sVar)
		{
		    //Support for setPhrase in old way.
            $aVarName = explode('.', $sVar);
            if (isset($aVarName[1])){
                $sVarGet = $aVarName[1];
            } else {
                $sVarGet = $sVar;
            }
            
			$sPhrase = _p($sVarGet);
			$sPhrase = str_replace("'", '&#039;', $sPhrase);
			if (preg_match("/\n/i", $sPhrase))
			{
				$aParts = explode("\n", $sPhrase);
				$sPhrase = '';
				foreach ($aParts as $sPart)
				{
					$sPart = trim($sPart);
					if (empty($sPart))
					{
						$sPhrase .= '\n ';
						continue;
					}
					$sPhrase .= $sPart . ' ';
				}
				$sPhrase = trim($sPhrase);
			}
			$this->_aPhrases[$sVar] = $sPhrase;
		}

		return $this;
	}

	/**
	 * Get all the phrases set by a controller
	 *
	 * @return array ARRAY of phrases
	 */
	public function getPhrases()
	{
		return (array) $this->_aPhrases;
	}

	/**
	 * Sets the breadcrumb structure for the site.
	 *
	 * @param string $sPhrase Breadcrumb title.
	 * @param string $sLink Breadcrumb link.
	 * @param bool $bIsTitle TRUE if this is the title breadcrumb for the page.
	 * @return $this
	 */
	public function setBreadCrumb($sPhrase, $sLink = '', $bIsTitle = false)
	{
		(($sPlugin = Phpfox_Plugin::get('template_template_setbreadcrump')) ? eval($sPlugin) : false);

		if (is_array($sPhrase)) {
			foreach ($sPhrase as $aPhrase) {
                if (isset($aPhrase[0])){
                    $aPhrase[0] = Phpfox::getSoftPhrase($aPhrase[0]);
                }
				((isset($aPhrase[2]) && $aPhrase[2]) ? $this->_aBreadCrumbTitle = array($aPhrase[0], $aPhrase[1]) : $this->_aBreadCrumbs[$aPhrase[1]] = $aPhrase[0]);
			}
			return $this;
		}

		if ($bIsTitle === true) {
            $sPhrase = Phpfox::getSoftPhrase($sPhrase);
			$this->_aBreadCrumbTitle = array(Phpfox_Locale::instance()->convert($sPhrase), $sLink);
			if (!empty($sLink)) {
				$this->setMeta('og:url', $sLink);
			}
		}

		if (!defined('PHPFOX_INSTALLER')) {
			$this->_aBreadCrumbs[$sLink] = Phpfox_Locale::instance()->convert($sPhrase);
		}
		return $this;
	}

	/**
	 * Get all the breadcrumbs we have loaded so far.
	 *
	 * @return array
	 */
	public function getBreadCrumb()
	{
		if (count($this->_aBreadCrumbTitle))
		{
			foreach ($this->_aBreadCrumbs as $sKey => $mValue)
			{
				if ($sKey === $this->_aBreadCrumbTitle[1])
				{
					unset($this->_aBreadCrumbs[$sKey]);
				}
			}

			if (isset($this->_aBreadCrumbTitle[1]))
			{
				$this->setMeta('canonical', $this->_aBreadCrumbTitle[1]);
			}
		}

		if (count($this->_aBreadCrumbs) === 1 && !count($this->_aBreadCrumbTitle))
		{
			$sKey = array_keys($this->_aBreadCrumbs);
			$this->setMeta('canonical', $sKey[0]);
		}
        if (Phpfox::isAdminPanel()){
            $this->assign([
                'aAdmincpBreadCrumb' => $this->_aBreadCrumbs
            ]);
        }
		return array($this->_aBreadCrumbs, $this->_aBreadCrumbTitle);
	}

	/**
	 * Clear the breadcrumb information.
	 * @return $this
	 */
	public function clearBreadCrumb()
	{
		$this->_aBreadCrumbs = array();
		$this->_aBreadCrumbTitle = array();
        return $this;
	}

	public function errorClearAll()
	{
		$this->clearBreadCrumb();
		$this->_aTitles = array();
	}

	/**
	 * Set the page title in a public array so we can get it later
	 * and display within the template.
	 *
	 * @see getTitle()
	 * @param string $sTitle Title to display on a specific page
	 * @return Phpfox_Template
	 */
	public function setTitle($sTitle)
	{
		$this->_aTitles[] = $sTitle;

		$this->setMeta('og:site_name', Phpfox::getParam('core.site_title'));
		$this->setMeta('og:title', $sTitle);

		return $this;
	}

	public function setSectionTitle($sSectionTitle) {
		$this->assign('sSectionTitle', $sSectionTitle);

		return $this;
	}

	public function setSectionMenu($aMenu) {
		$this->assign('aSectionMenu', $aMenu);

		return $this;
	}

	public function setActionMenu($aMenu) {
		$this->assign('aActionMenu', $aMenu);

		return $this;
	}

	/**
	 * Set the current template for the site.
	 *
	 * @param string $sLayout Template name.
	 * @return $this
	 */
	public function setTemplate($sLayout)
	{
		$this->sDisplayLayout = $sLayout;

		return $this;
	}

	/**
	 * All data placed between the HTML tags <head></head> can be added with this method.
	 * Since we rely on custom templates we need the header data to be custom as well. Current
	 * support is for: css & JavaScript
	 * All HTML added here is coded under XHTML standards.
	 *
	 * @access public
	 * @param array|string $mHeaders
	 * @return $this
	 */
	public function setHeader($mHeaders, $mValue = null)
	{
		if(!$this->_loadedHeader){
			$this->_loadedHeader = true;
			$this->setHeader('cache', Phpfox::getMasterFiles());
		}
		
		if ($mHeaders == 'cache')
		{
			if (Phpfox::getParam('core.cache_js_css') && !PHPFOX_IS_AJAX)
			{
				foreach ($mValue as $sKey => $mNewValue)
				{
					$this->_aCacheHeaders[$sKey][$mNewValue] = true;
				}
			}


			$this->_aHeaders[] = $mValue;
		}
		else
		{
			if ($mValue !== null)
			{
				if ($mHeaders == 'head')
				{
					$mHeaders = array($mValue);
				}
				else
				{
					$mHeaders = array($mHeaders => $mValue);
				}
			}

			$this->_aHeaders[] = $mHeaders;
		}

		return $this;
	}

	/**
	 * All data placed between the HTML tags <head></head> can be added with this method for mobile devices.
	 * Since we rely on custom templates we need the header data to be custom as well. Current
	 * support is for: css & JavaScript
	 * All HTML added here is coded under XHTML standards.
	 *
	 * @access public
	 * @param string|array $mHeaders
	 * @return $this
	 */
	public function setMobileHeader($mHeaders, $mValue = null)
	{
		if ($mValue !== null)
		{
			$mHeaders = array($mHeaders => $mValue);
		}

		$this->_aMobileHeaders[] = $mHeaders;

		return $this;
	}

	/**
	 * Set settings for the text editor in use.
	 *
	 * @param array $aParams ARRAY of settings.
	 * @return $this
	 */
	public function setEditor($aParams = array())
	{
		if (count($aParams))
		{
			$this->_aEditor = $aParams;
		}

		$this->_aEditor['active'] = true;
		$this->_aEditor['toggle_image'] = $this->getStyle('image', 'editor/fullscreen.png');
		$this->_aEditor['toggle_phrase'] = _p('toggle_fullscreen');

        (($sPlugin = Phpfox_Plugin::get('set_editor_end')) ? eval($sPlugin) : false);

		$this->setHeader('cache', array(
						'editor.js' => 'static_script',
                        'wysiwyg/default/core.js' => 'static_script'
				)
		);

		return $this;
	}

	public function getActualTitle() {
		$title = '';
		foreach ($this->_aTitles as $sTitle) {
			$title .= Phpfox_Parse_Output::instance()->clean($sTitle) . ' ' . Phpfox::getParam('core.title_delim') . ' ';
		}
		$title = trim($title);
		$title = rtrim($title, ' ' . Phpfox::getParam('core.title_delim') . '');

		return $title;
	}

	/**
	 * Get the title for the current page behind displayed.
	 * All titles are added earlier in the script using self::setTitle().
	 * Each title is split with a delimiter specified from the Admin CP.
	 *
	 * @see setTitle()
	 * @return string $sData Full page title including delimiter
	 */
	public function getTitle()
	{
		$oFilterOutput = Phpfox::getLib('parse.output');

		(($sPlugin = Phpfox_Plugin::get('template_gettitle')) ? eval($sPlugin) : false);

		$sData = '';
		foreach ($this->_aTitles as $sTitle)
		{
			$sData .= $oFilterOutput->clean($sTitle) . ' ' . Phpfox::getParam('core.title_delim') . ' ';
		}

		if (!Phpfox::getParam('core.include_site_title_all_pages'))
		{
			$sData .= (defined('PHPFOX_INSTALLER') ? Phpfox::getParam('core.global_site_title') : Phpfox_Locale::instance()->convert(Phpfox::getParam('core.global_site_title')));
		}
		else
		{
			$sData = trim(rtrim(trim($sData), Phpfox::getParam('core.title_delim')));
			if (empty($sData))
			{
				$sData = (defined('PHPFOX_INSTALLER') ? Phpfox::getParam('core.global_site_title') : Phpfox_Locale::instance()->convert(Phpfox::getParam('core.global_site_title')));
			}
		}

		$sSort = Phpfox_Request::instance()->get('sort');
		if (!empty($sSort))
		{
			$mSortName = Phpfox_Search::instance()->getPhrase('sort', $sSort);
			if ($mSortName !== false)
			{
				$sData .= ' ' . Phpfox::getParam('core.title_delim') . ' ' . $mSortName[1];
			}
		}

		if (!Phpfox::getParam('core.branding'))
		{
			$sData .= ' - ' . Phpfox::link(false, false) . '';
		}

		return $sData;
	}

	/**
	 * Gets all the keywords from a string.
	 *
	 * @param string $sTitle Title to parse.
	 * @return string Splits all the keywords from a title.
	 */
	public function getKeywords($sTitle)
	{
		$aWords = explode(' ', $sTitle);
		$sKeywords = '';
		foreach ($aWords as $sWord)
		{
			if (empty($sWord))
			{
				continue;
			}

			if (strlen($sWord) < 2)
			{
				continue;
			}

			$sKeywords .= $sWord . ',';
		}
		$sKeywords = rtrim($sKeywords, ',');

		return $sKeywords;
	}

	/**
	 * Set all the meta tags to be used on the site.
	 *
	 * @param array|string $mMeta ARRAY of meta tags.
	 * @param string $sValue Value of meta tags in case the 1st argument is a string.
	 * @return $this
	 */
	public function setMeta($mMeta, $sValue = null)
	{
		if (!is_array($mMeta))
		{
			$mMeta = array($mMeta => $sValue);
		}

		if(isset($mMeta['keywords']))
		{
			// get custom metas added through the AdminCP -> Tools -> SEO
			$aSiteMetas = Admincp_Service_Seo_Seo::instance()->getSiteMetas();

			$sThisController = Admincp_Service_Seo_Seo::instance()->getUrl( (isset($_GET[PHPFOX_GET_METHOD]) ? $_GET[PHPFOX_GET_METHOD] : '/') );

			if(is_array($aSiteMetas) && !empty($aSiteMetas))
			{
				// make sure this is the right controller
				foreach ($aSiteMetas as $iKey => $aSiteMeta)
				{
					if (empty($aSiteMeta['url']) || strpos($sThisController, $aSiteMeta['url']) === false)
					{
						unset($aSiteMetas[$iKey]);
					}
				}

				if(!empty($sThisController))
				{
					// remove the general keywords
					$mMeta['keywords'] = array(str_replace(Phpfox::getParam('core.keywords'), '', $mMeta['keywords']));
					// check each new custom keywords
					foreach ($aSiteMetas as $aSiteMeta)
					{
						// keywords
						if($aSiteMeta['type_id'] == 0)
						{
							$mMeta['keywords'][] = $aSiteMeta['content'];
						}
					}
					$mMeta['keywords'] = join(',', $mMeta['keywords']);
				}
			}
		}
		// end

		foreach ($mMeta as $sKey => $sValue)
		{
			if ($sKey == 'og:url')
			{
				$this->_aMeta[$sKey] = $sValue;
				return $this;
			}

			if (isset($this->_aMeta[$sKey]))
			{
				$this->_aMeta[$sKey] .= ($sKey == 'keywords' ? ', ' : ' ') . $sValue;
			}
			else
			{
				$this->_aMeta[$sKey] = $sValue;
			}
			$this->_aMeta[$sKey] = ltrim($this->_aMeta[$sKey], ', ');
		}

		return $this;
	}

	/**
	 * Gets any data we plan to place within the HTML tags <head></head> for mobile devices.
	 * This method also groups the data to give the template a nice clean look.
	 *
	 * @return string $sData Returns the HTML data to be placed within <head></head>
	 */
	public function getMobileHeader()
	{
		return $this->getHeader();
	}

	/**
	 * Gets a 32 string character of the version of the static files
	 * on the site.
	 *
	 * @return string 32 character MD5 sum
	 */
	public function getStaticVersion()
	{
		$sVersion = md5((((defined('PHPFOX_NO_CSS_CACHE') && PHPFOX_NO_CSS_CACHE) || $this->_bIsTestMode === true) ? PHPFOX_TIME : Phpfox::getId() . Phpfox::getBuild()) . (defined('PHPFOX_INSTALLER') ? '' : '-' . Phpfox::getParam('core.css_edit_id') . Phpfox::getBuild() . '-' . $this->_sThemeFolder . '-' . $this->_sStyleFolder));

		(($sPlugin = Phpfox_Plugin::get('template_getstaticversion')) ? eval($sPlugin) : false);

		return $sVersion;
	}

	public function setPageMeta($meta) {
		$this->_meta = $meta;
	}

	public function getPageMeta() {
		return $this->_meta;
	}

	public function keepBody($param = null) {
		if ($param === null) {
			return $this->_keepBody;
		}

		$this->_keepBody = $param;

		return $this;
	}

	/**
	 * Gets any data we plan to place within the HTML tags <head></head>.
	 * This method also groups the data to give the template a nice clean look.
	 *
	 * @return string|array $sData Returns the HTML data to be placed within <head></head>
	 */
	public function getHeader($bReturnArray = false) {

		$bundle = $this->_bundle();

		if (Phpfox::isAdminPanel()) {
			$this->setHeader(array('custom.css' => 'style_css'));
		}

		if ($this->delayedHeaders) {
			foreach ($this->delayedHeaders as $header) {
				$this->setHeader('cache', $header);
			}
		}

		if (!defined('PHPFOX_INSTALLER')) {
			Core\Event::trigger('lib_phpfox_template_getheader', $this);
			foreach ((new Core\App())->all() as $App) {
				if ($App->head && is_array($App->head)) {
					foreach ($App->head as $head) {
						$this->setHeader($head);
					}
				}

				if ($App->js_phrases) {
					$phrases = [];
					foreach ($App->js_phrases as $key => $phrase) {
						$phrases[$key] = _p($phrase);
					}
					$this->setHeader('<script>var ' . str_replace('-' , '_',$App->apps_dir) . '_Phrases = ' . json_encode($phrases) . ';</script>');
				}

				if ($App->settings) {
					$Setting = new Core\Setting();
					foreach ($App->settings as $key => $setting) {
						if (isset($setting->js_variable)) {
							$this->setHeader('<script>var ' . $key . ' = "' . $Setting->get($key) . '";</script>');
						}
					}
				}
			}
		}

		$aArrayData = array();
		$sData = '';
		$sJs = '';
		$iVersion = $this->getStaticVersion();
		$oUrl = Phpfox_Url::instance();
		if (!defined('PHPFOX_DESIGN_DND'))
		{
			define('PHPFOX_DESIGN_DND', false);
		}

		if (!PHPFOX_IS_AJAX_PAGE)
		{
			(($sPlugin = Phpfox_Plugin::get('template_getheader')) ? eval($sPlugin) : false);

			$sJs .= "\t\t\tvar oCore = {'core.is_admincp': " . (Phpfox::isAdminPanel() ? 'true' : 'false') . ", 'core.section_module': '" . Phpfox_Module::instance()->getModuleName() . "', 'profile.is_user_profile': " . (defined('PHPFOX_IS_USER_PROFILE') && PHPFOX_IS_USER_PROFILE ? 'true' : 'false') . ", 'log.security_token': '" . Log_Service_Session::instance()->getToken() . "', 'core.url_rewrite': '" . Phpfox::getParam('core.url_rewrite') . "', 'core.country_iso': '" . (Phpfox::isUser() ? Phpfox::getUserBy('country_iso') : '') . "', 'core.can_move_on_a_y_and_x_axis' : " . ((!defined('PHPFOX_INSTALLER') && Phpfox::getParam('core.can_move_on_a_y_and_x_axis')) ? 'true' : 'false') . ", 'core.default_currency': '" . (defined('PHPFOX_INSTALLER') ? 'USD' : Core_Service_Currency_Currency::instance()->getDefault()) . "', 'core.enabled_edit_area': " . (Phpfox::getParam('core.enabled_edit_area') ? 'true' : 'false') . ", 'core.disable_hash_bang_support': " . (Phpfox::getParam('core.disable_hash_bang_support') ? 'true' : 'false') . ", 'core.site_wide_ajax_browsing': " . ((!defined('PHPFOX_IN_DESIGN_MODE') && Phpfox::getParam('core.site_wide_ajax_browsing') && !Phpfox::isAdminPanel() && Phpfox::isUser()) ? 'true' : 'false') . ", 'profile.user_id': " . (defined('PHPFOX_IS_USER_PROFILE') && PHPFOX_IS_USER_PROFILE ? Profile_Service_Profile::instance()->getProfileUserId() : 0) . "};\n";
// You are filtering out the controllers which should not load 'content' ajaxly, finding a way for pages.view/1/info and like that
			$sProgressCssFile = $this->getStyle('css', 'progress.css');
			$sStylePath = str_replace(Phpfox::getParam('core.path_file'), '', str_replace('progress.css', '', $sProgressCssFile));

			$aJsVars = array(
                    'sBaseURL' => Phpfox::getParam('core.path'),
					'sJsHome' => Phpfox::getParam('core.path_file'),
					'sJsHostname' => isset($_SERVER['HTTP_HOST'])?$_SERVER['HTTP_HOST']:'localhost',
					'sSiteName' => Phpfox::getParam('core.site_title'),
					'sJsStatic' => $oUrl->getDomain() . PHPFOX_STATIC,
					'sJsStaticImage' => Phpfox::getParam('core.url_static_image'),
					'sImagePath' => $this->getStyle('image'),
					'sStylePath' => $this->getStyle('css'),
					'sVersion' => Phpfox::getId(),
					'sJsAjax' => Phpfox::getParam('core.path') . '_ajax/',
					'sStaticVersion' => $iVersion,
					'sGetMethod' => PHPFOX_GET_METHOD,
					'sDateFormat' => (defined('PHPFOX_INSTALLER') ? '' : Phpfox::getParam('core.date_field_order')),
					'sGlobalTokenName' => Phpfox::getTokenName(),
					'sController' => Phpfox_Module::instance()->getFullControllerName(),
					'bJsIsMobile' => false,
					'sProgressCssFile' => $sProgressCssFile,
					'sHostedVersionId' => (defined('PHPFOX_IS_HOSTED_VERSION') ? PHPFOX_IS_HOSTED_VERSION : '')
			);

			if (!defined('PHPFOX_INSTALLER'))
			{
				$aJsVars['sJsCookiePath'] = Phpfox::getParam('core.cookie_path');
				$aJsVars['sJsCookieDomain'] = Phpfox::getParam('core.cookie_domain');
				$aJsVars['sJsCookiePrefix'] = Phpfox::getParam('core.session_prefix');
				$aJsVars['bPhotoTheaterMode'] = false;
				$aJsVars['bUseHTML5Video'] = false;
				$aJsVars['bIsAdminCP'] = Phpfox::isAdminPanel();
                $aJsVars['bIsUserLogin'] = Phpfox::isUser();
				if (Phpfox::isAdmin())
				{
					$aJsVars['sAdminCPLocation'] = Phpfox::getParam('admincp.admin_cp');
				}
				else
				{
					$aJsVars['sAdminCPLocation'] = '';
				}
				if (Phpfox::isModule('notification'))
				{
					$aJsVars['notification.notify_ajax_refresh'] = Phpfox::getParam('notification.notify_ajax_refresh');
				}

				$sLocalDatepicker = 'PF.Base/'.PHPFOX_STATIC .'jscript/jquery/locale/jquery.ui.datepicker-' . strtolower(Phpfox_Locale::instance()->getLangId()) . '.js';

				if (file_exists($sLocalDatepicker))
				{
					$sFile = str_replace('PF.Base/'.PHPFOX_STATIC . 'jscript/', '', $sLocalDatepicker);
					$this->setHeader(array($sFile => 'static_script'));
				}

                $bOffFullAjaxMode = Phpfox::getParam('core.turn_off_full_ajax_mode');
				$aJsVars['bOffFullAjaxMode'] = $bOffFullAjaxMode ? true : false;
			}

			(($sPlugin = Phpfox_Plugin::get('template_getheader_setting')) ? eval($sPlugin) : false);

			$sJs .= "\t\t\tvar oParams = {";
			$iCnt = 0;
			foreach ($aJsVars as $sVar => $sValue)
			{
				$iCnt++;
				if ($iCnt != 1)
				{
					$sJs .= ",";
				}

				if (is_bool($sValue))
				{
					$sJs .= "'{$sVar}': " . ($sValue ? 'true' : 'false');
				}
				elseif (is_numeric($sValue))
				{
					$sJs .= "'{$sVar}': " . $sValue;
				}
				else
				{
					$sJs .= "'{$sVar}': '" . str_replace("'", "\'", $sValue) . "'";
				}
			}
			$sJs .= "};\n";

			if (!defined('PHPFOX_INSTALLER'))
			{
				$aLocaleVars = array(
						'core.are_you_sure',
						'core.yes',
						'core.no',
						'core.save',
						'core.cancel',
						'core.go_advanced',
						'core.processing',
						'attachment.attach_files',
						'core.close',
						'core.language_packages',
						'core.move_this_block',
						'core.uploading',
						'language.loading',
						'core.saving',
						'core.loading_text_editor',
						'core.quote',
						'core.loading',
						'core.confirm'
				);

				if (Phpfox::isModule('im') && Phpfox::getParam('im.enable_im_in_footer_bar'))
				{
					$aLocaleVars[] = 'im.find_your_friends';
				}

				(($sPlugin = Phpfox_Plugin::get('template_getheader_language')) ? eval($sPlugin) : false);

				$sJs .= "\t\t\tvar oTranslations = {";
				$iCnt = 0;
				foreach ($aLocaleVars as $sValue)
				{
					$aParts = explode('.', $sValue);

					if (isset($aParts[1])){
                        $sValue = $aParts[1];
					}

					$iCnt++;
					if ($iCnt != 1)
					{
						$sJs .= ",";
					}

					$sJs .= "'{$sValue}': '" . html_entity_decode(str_replace("'", "\'", _p($sValue)), null, 'UTF-8') . "'";
				}
				$sJs .= "};\n";

				$aModules = Phpfox_Module::instance()->getModules();
				$sJs .= "\t\t\tvar oModules = {";
				$iCnt = 0;
				foreach ($aModules as $sModule => $iModuleId)
				{
					$iCnt++;
					if ($iCnt != 1)
					{
						$sJs .= ",";
					}
					$sJs .= "'{$sModule}': true";
				}
				$sJs .= "};\n";
			}

			if (count($this->_aImages))
			{
				$sJs .= "\t\t\tvar oJsImages = {";
				foreach ($this->_aImages as $sKey => $sImage)
				{
					$sJs .= $sKey . ': \'' . $sImage. '\',';
				}
				$sJs = rtrim($sJs, ',');
				$sJs .= "};\n";
			}

			$aEditorButtons = Phpfox::getLib('editor')->getButtons();

			$iCnt = 0;
			$sJs .= "\t\t\tvar oEditor = {";

			if (count($this->_aEditor) && isset($this->_aEditor['active']) && $this->_aEditor['active'])
			{
				foreach ($this->_aEditor as $sVar => $mValue)
				{
					$iCnt++;
					if ($iCnt != 1)
					{
						$sJs .= ",";
					}
					$sJs .= "'{$sVar}': " . (is_bool($mValue) ? ($mValue === true ? 'true' : 'false') : "'{$mValue}'") . "";
				}

				$sJs .= ", ";
			}
			$sJs .= "images:[";
			foreach ($aEditorButtons as $mEditorButtonKey => $aEditorButton)
			{
				$sJs .= "{";
				foreach ($aEditorButton as $sEditorButtonKey => $sEditorButtonValue)
				{
					$sJs .= "" . $sEditorButtonKey . ": '" . $sEditorButtonValue . "',";
				}
				$sJs = rtrim($sJs, ',') . "},";
			}
			$sJs = rtrim($sJs, ',') . "]";

			$sJs .= "};\n";
			if (!defined('PHPFOX_INSTALLER') && Phpfox::getParam('core.load_jquery_from_google_cdn'))
			{
				$sData .= "\t\t" . '<script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/' . Phpfox::getParam('core.jquery_google_cdn_version') . '/jquery.min.js"></script>' . "\n";
			}
		}

		if (PHPFOX_IS_AJAX_PAGE)
		{
			$this->_aCacheHeaders = array();
		}

		$bIsHttpsPage = false;
		if (!defined('PHPFOX_INSTALLER') && Phpfox::getParam('core.force_https_secure_pages'))
		{
			if (in_array(str_replace('mobile.', '', Phpfox_Module::instance()->getFullControllerName()), Core_Service_Core::instance()->getSecurePages())
					&& (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on')
			)
			{
				$bIsHttpsPage = true;
			}
		}

		$aSubCache = array();
		$sStyleCacheData = '';
		$sJsCacheData = '';
		$aCacheJs = array();
		$aCacheCSS = array();
		$sQmark = '?';
		$this->_sFooter = '';
		if (!defined('PHPFOX_INSTALLER')) {
			$this->_sFooter .= '<script src="' . url('/user/boot') . '"></script>';
		}
		$sJs .= "\t\t\t" . 'var $Behavior = {}, $Ready = $Ready = function(callback) {$Behavior[callback.toString().length] = callback;}, $Events = {}, $Event = function(callback) {$Events[callback.toString().length] = callback;};' . "\n";
		$sJs .= "\t\t\t" .'var $Core = {};' . "\n";
		$aCustomCssFile = array();
		foreach ($this->_aHeaders as $aHeaders)
		{
			if (!is_array($aHeaders))
			{
				$aHeaders = array($aHeaders);
			}

			foreach ($aHeaders as $mKey => $mValue)
			{
				$sQmark = (strpos($mKey, '?') ? '&amp;' : '?');

				if (is_numeric($mKey))
				{
					if ($mValue === null)
					{
						continue;
					}

					if ($bReturnArray)
					{
						$aArrayData[] = $mValue;
					}
					else
					{
						if (is_string($mValue) && (strpos($mValue, '.js') !== false || strpos($mValue, 'javascript') !== false))
						{
							if (strpos($mValue, 'RecaptchaOptions'))
							{
								$sData .= "\t\t" . $mValue . "\n";
							}
							else
							{
								$this->_sFooter .= "\t\t". $mValue;
							}
						}
						else if (is_string($mValue))
						{
							$sData .= "\t\t" . $mValue . "\n";
						}
						else
						{
							$sData .= "\t\t" . implode($mValue) . "\n";
						}
					}

				} elseif (filter_var($mKey, FILTER_VALIDATE_URL) !== FALSE) {
                    if (strpos($mKey, '.css') !== false) {
                        $aCacheCSS[] = $mKey;
                    } elseif (strpos($mKey, '.js') !== false) {
                        $aCacheJs[] = $mKey;
                    }
                } else if ($mKey == 'master')
				{
					$aMaster = array('css' => array(), 'jscript' => array());
					foreach ($mValue as $sValKey => $sValVal)
					{
						if (strpos($sValKey, '.css') !== false)
						{
							if ($sValVal == 'style_css')
							{
								$aMaster['css'][] = 'theme/frontend/' . $this->getThemeFolder() . '/style/' . $this->getStyleFolder() . '/css/' . $sValKey;
							}
							else if (strpos($sValVal, 'module_') !== false)
							{
								$aMaster['css'][] = 'module/' . (str_replace('module_','',$sValVal)) . '/static/css/' . $this->getThemeFolder() . '/' . $this->getStyleFolder() . '/' . $sValKey;
							}
						}
						else if (strpos($sValKey, '.js') !== false)
						{
							if ($sValVal == 'static_script')
							{
								$aMaster['jscript'][] = 'static/jscript/' . $sValKey;
							}
							else if (strpos($sValVal, 'module_') !== false)
							{
								$aMaster['jscript'][] = 'module/' . (str_replace('module_', '', $sValVal)) . '/static/jscript/' . $sValKey;
							}
						}
					}
					unset($this->_aHeaders[$mKey]); // just to avoid confusions
					$this->_aHeaders['master'] = $aMaster;
				}
				else
				{
					$bToHead = false;
					// This happens when the developer needs something to go to <head>
					if (is_array($mValue))
					{
						$aKeyHead = array_keys($mValue);
						$aKeyValue = array_values($mValue);
						$bToHead = ($mKey == 'head');
						$mKey = array_pop($aKeyHead);
						$mValue = array_pop($aKeyValue);
					}

					switch ($mValue)
					{
						case 'style_script':

							if (isset($aSubCache[$mKey][$mValue]))
							{
								continue;
							}

							if ($bReturnArray)
							{
								$aArrayData[] = $this->getStyle('jscript', $mKey);
							}
							else
							{
								if ($bToHead == 'head')
								{
									$aCacheCSS[] = str_replace(Phpfox::getParam('core.path_file'), '', $this->getStyle('jscript', $mKey));
								}
								else
								{
									$aCacheJs[] = str_replace(Phpfox::getParam('core.path_file'), '', $this->getStyle('jscript', $mKey));
								}

							}
							break;
						case 'style_css':
							$bCustomStyle = false;
							if ($bCustomStyle === false)
							{
								if ($bReturnArray)
								{
									$aArrayData[] = $this->getStyle('css', $mKey);
								}
								else
								{
									$aCacheCSS[] = str_replace(Phpfox::getParam('core.path_file'), '', $this->getStyle('css', $mKey));
								}
							}
							else
							{
								if (defined('PHPFOX_IS_HOSTED_SCRIPT'))
								{
									$bLoadCustomThemeOverwrite = true;
								}
								else
								{
									if ($bReturnArray)
									{
										$aArrayData[] = Phpfox::getParam('core.url_file') . 'static/' . $this->_aTheme['style_id'] . '_' . $mKey;
									}
									else
									{
										if (isset($this->_aCacheHeaders[$mKey]))
										{
											$aCustomCssFile[] = Phpfox::getParam('core.url_file') . 'static/' . $this->_aTheme['style_id'] . '_' . $mKey . '';
										}
										else
										{
											if ($bIsHttpsPage)
											{
												$aCustomCssFile[] = Phpfox::getParam('core.url_file') . 'static/' . $this->_aTheme['style_id'] . '_secure_' . $mKey;
											}
											else
											{
												$aCustomCssFile[] = Phpfox::getParam('core.url_file') . 'static/' . $this->_aTheme['style_id'] . '_' . $mKey;
											}

										}
									}
								}
							}

							break;
						case 'static_script':
							if (isset($aSubCache[$mKey][$mValue]))
							{
								continue;
							}

							if (!defined('PHPFOX_INSTALLER') && Phpfox::getParam('core.load_jquery_from_google_cdn'))
							{
								if ($mKey == 'jquery/ui.js' || $mKey == 'jquery/jquery.js')
								{
									if ($mKey == 'jquery/ui.js')
									{
										$this->_sFooter .= "\t\t" . '<script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jqueryui/' . Phpfox::getParam('core.jquery_ui_google_cdn_version') . '/jquery-ui.min.js"></script>' . "\n";
									}
									continue;
								}
							}

							if ($bReturnArray)
							{
								$aArrayData[] = Phpfox::getParam('core.url_static_script') . $mKey;
							}
							else
							{
								if (isset($this->_aCacheHeaders[$mKey]))
								{
									if ($bToHead == 'head')
									{
										$aCacheCSS[] = 'static/jscript/' . $mKey;
									}
									else
									{
										$aCacheJs[] =  'static/jscript/' . $mKey;
									}
								}
								else
								{
									if ($bToHead == 'head')
									{
										$aCacheCSS[] = 'static/jscript/' . $mKey;
									}
									else
									{
										$aCacheJs[] = 'static/jscript/' . $mKey;
									}
								}
							}
							break;

						default:
						    if(preg_match('/app_/i', $mValue)){
                                $aParts = explode('_', $mValue, 2);
                                $basePath =  dirname(Phpfox::getParam('core.path_file'));
                                if ('.js' == substr($mKey, -3)){
                                    if ($bReturnArray)
                                    {
                                        $aArrayData[] = $basePath  . '/PF.Site/Apps/'. $aParts[1] . '/assets/' . $mKey;
                                    }
                                    else
                                    {
                                        if (isset($this->_aCacheHeaders[$mKey]))
                                        {
                                            $aCacheJs[] =  $basePath . '/PF.Site/Apps/'. $aParts[1] . '/assets/' . $mKey;
                                        }
                                        else
                                        {
                                            $aCacheJs[] =  $basePath . '/PF.Site/Apps/'. $aParts[1] . '/assets/' . $mKey;
                                        }
                                    }

                                }else if('.css' == substr($mKey, -4)){
                                    $Theme = $this->theme()->get();
                                    $sFileName =  trim(str_replace(dirname(Phpfox::getParam('core.path_file')),'',$this->getStyle('css', $mKey, $aParts[1],true)),'/');
                                    $sFileName = $Theme->getCssFileName($sFileName,'app');
                                    $aCacheCSS[] = $sFileName;
                                    $bCustomStyle = false;
                                    if ($bCustomStyle === false)
                                    {
                                        if ($bReturnArray)
                                        {
                                            $aCachesCSS[] = $sFileName;
                                        }
                                        else
                                        {
                                            $aCachesCSS[] = $sFileName;
                                        }
                                    }
                                    else
                                    {
                                        $aCachesCSS[] = $sFileName;
                                    }
                                }
                            }else if (preg_match('/module_/i', $mValue))
							{
								$aParts = explode('_', $mValue);
								if (isset($aParts[1]) && Phpfox::isModule($aParts[1]))
								{
									if (substr($mKey, -3) == '.js')
									{
										if ($bReturnArray)
										{
											$aArrayData[] = Phpfox::getParam('core.path_file') . 'module/' . $aParts[1] . '/static/jscript/' . $mKey;
										}
										else
										{
											if (isset($this->_aCacheHeaders[$mKey]))
											{
												$aCacheJs[] = 'module/' . $aParts[1] . '/static/jscript/' . $mKey;
											}
											else
											{
												$aCacheJs[] = 'module/' . $aParts[1] . '/static/jscript/' . $mKey;
											}
										}
									}
									elseif (substr($mKey, -4) == '.css')
									{
										$Theme = $this->theme()->get();
										$sFileName = $Theme->getCssFileName(str_replace(Phpfox::getParam('core.path_file'),'',$this->getStyle('css', $mKey, $aParts[1])));
										$aCacheCSS[] = $sFileName;
										$bCustomStyle = false;
										if ($bCustomStyle === false)
										{
											if ($bReturnArray)
											{
												$aCachesCSS[] = $sFileName;
											}
											else
											{
												$aCachesCSS[] = $sFileName;
											}
										}
										else
										{
											$aCachesCSS[] = $sFileName;
										}
									}
								}
							}
							break;
					}

					$aSubCache[$mKey][$mValue] = true;
				}
			}
		}

		$sCacheData = '';
		$sCacheData .= "\n\t\t<script type=\"text/javascript\">\n";
		$sCacheData .= $sJs;
		$sCacheData .= "\t\t</script>";

		if (!empty($sStyleCacheData))
		{
			$sCacheData .= "\n\t\t" . '<link rel="stylesheet" type="text/css" href="' . Phpfox::getParam('core.url_static') . 'gzip.php?t=css&amp;s=' . $sStylePath . '&amp;f=' . rtrim($sStyleCacheData, ',') . '&amp;v=' . $iVersion . '" />';
		}

		if (!empty($sJsCacheData))
		{
			$sCacheData .= "\n\t\t" . '<script type="text/javascript" src="' . Phpfox::getParam('core.url_static') . 'gzip.php?t=js&amp;f=' . rtrim($sJsCacheData, ',') . '&amp;v=' . $iVersion . '"></script>';
		}

		if (!empty($sCacheData))
		{
			$sData = preg_replace('/<link rel="shortcut icon" type="image\/x-icon" href="(.*?)" \/>/i', '<link rel="shortcut icon" type="image/x-icon" href="\\1" />' . $sCacheData, $sData);
		}

		if ($bReturnArray)
		{
			$sData = '';
		}
		$aCacheJs = array_unique($aCacheJs);

		$aSubCacheCheck = array();

		$sBaseUrl = Phpfox::getLib('cdn')->getUrl(Phpfox::getBaseUrl());
		$sPathActual = Phpfox::getLib('cdn')->getUrl(Phpfox::getParam('core.path_actual'));
		$sCorePath = Phpfox::getLib('cdn')->getUrl(Phpfox::getParam('core.path'));
		foreach ($aCacheCSS as $sFile)
		{
			if (defined('PHPFOX_INSTALLER'))
			{
				// $sData .= "\t\t" . '<link rel="stylesheet" type="text/css" href="' . $sFile . $sQmark .'v=' . $iVersion . '" />' . "\n";
			}
			else
			{
				if (isset($aSubCacheCheck[$sFile]))
				{
					continue;
				}

				$aSubCacheCheck[$sFile] = true;
                if (filter_var($sFile, FILTER_VALIDATE_URL) !== FALSE) {
                    $sData .= "\t\t" . '<link href="' . $sFile . $sQmark .'v=' . $iVersion . '" rel="stylesheet" type="text/css" />' . "\n";
                } else if(strpos($sFile,'PF.Site') === false){
					$sData .= "\t\t" . '<link href="' . $sBaseUrl . 'PF.Base/'. $sFile . $sQmark .'v=' . $iVersion . '" rel="stylesheet" type="text/css" />' . "\n";
				}else{
					$sData .= "\t\t" . '<link href="' . $sBaseUrl . $sFile . $sQmark .'v=' . $iVersion . '" rel="stylesheet" type="text/css" />' . "\n";
				}


			}
		}

		if ($bundle) {
			$this->_sFooter .= '<!-- autoload -->';
		}


		$aExcludeBundles = [];

		(($sPlugin = Phpfox_Plugin::get('template_getheader_exclude_bundle_js')) ? eval($sPlugin) : false);


        foreach ($aCacheJs as $sFile) {
            if (!defined('PHPFOX_INSTALLER')) {

                $async = '';

                foreach($aExcludeBundles as $tmp){
                    if(false !== strpos($sFile,$tmp)){
                        $async = 'type="text/javascript" ';
                    }
                }

                if (filter_var($sFile, FILTER_VALIDATE_URL) !== FALSE) {
                    $this->_sFooter .= "\t\t" . '<script '.$async.'src="' . $sFile . $sQmark . 'v=' . $iVersion . '"></script>' . "\n";
                }else{
                    $this->_sFooter .= "\t\t" . '<script '.$async.'src="' . $sPathActual . 'PF.Base/' . $sFile . $sQmark . 'v=' . $iVersion . '"></script>' . "\n";
                }
            }
        }

		if (count($this->_aPhrases))
		{
			$sData .= "\n\t\t<script type=\"text/javascript\">\n\t\t";
			foreach ($this->_aPhrases as $sVar => $sPhrase)
			{
				$sPhrase = html_entity_decode($sPhrase, null, 'UTF-8');

				$sData .= "\t\t\toTranslations['{$sVar}'] = '" . str_replace("'", "\'", $sPhrase) . "';\n";
			}
			$sData .= "\t\t</script>\n";
		}

		if (!defined('PHPFOX_INSTALLER') && !Phpfox::isAdminPanel()) {
			$Request = \Phpfox_Request::instance();
			if ($Request->segment(1) == 'theme' && $Request->segment(2) == 'demo') {
				$sData .= '<link href="' . $sBaseUrl . 'PF.Base/theme/default/flavor/default.css?v=' . Phpfox::internalVersion() . '" rel="stylesheet">';
			}
			else {
				$Theme = $this->_theme->get();
				(($sPlugin = Phpfox_Plugin::get('template_getheader_css')) ? eval($sPlugin) : false);
				if (request()->get('force-flavor') && !file_exists(PHPFOX_DIR_SITE . 'flavors/' . flavor()->active->id . '/flavor/' . $Theme->flavor_folder . '.css')) {
					$sData .= '<link href="' . $sBaseUrl . 'PF.Base/theme/bootstrap/flavor/default.css?v=' . Phpfox::internalVersion() . '" rel="stylesheet">';
				}
				else {
					$sData .= '<link href="' . $sBaseUrl . 'PF.Site/flavors/' . flavor()->active->id . '/flavor/' . $Theme->flavor_folder . '.css?v=' . Phpfox::internalVersion() . '" rel="stylesheet">';
				}

				$sData .= '<link href="' . $sBaseUrl . 'PF.Base/less/version/boot.css?v=' . Phpfox::internalVersion() . '" rel="stylesheet">';
			}
		}

		if (!defined('PHPFOX_INSTALLER')) {
			$Apps = new Core\App();
			foreach ($Apps->all() as $App) {
				$product_version = Phpfox::internalVersion() . '-' . str_replace('.', '', $App->version);

				$assets = $App->path . 'assets/';
				if (file_exists($assets . 'autoload.js')) {
					$url = str_replace(PHPFOX_DIR_SITE, $sPathActual . 'PF.Site/', $assets) . 'autoload.js';
					$this->_sFooter .= '<script src="' . $url . '?v=' . $product_version . '"></script>';
				}

				/*
				 * Neil: All autoload.css will load and compile to 1 file when rebuild css
				 * Ray: I've added this back because in 4.4 we have a feature to compile all CSS files
				 */
				 if (file_exists($assets . 'autoload.css')) {
                    $url = str_replace(PHPFOX_DIR_SITE, $sPathActual . 'PF.Site/', $assets) . 'autoload.css';
                    $sData .= '<link href="' . $url . '?v=' . $product_version . '" rel="stylesheet">';
                }
			}

            if (CSS_DEVELOPMENT_MODE) {
                $Theme = $this->_theme->get();
                $css = new Core\Theme\CSS($Theme);
                $css->buildFile('developing.less', 'developing');
                $sData .= '<link href="' . $sBaseUrl . 'PF.Site/flavors/' . flavor()->active->id . '/flavor/developing.css?v=' . Phpfox::internalVersion() . '" rel="stylesheet">';
            }

			if (!Phpfox::isAdminPanel() && is_object($this->_theme)) {
				$asset = $this->_theme->get()->getPath() . 'assets/autoload.js';
				if (file_exists($asset)) {
					$url = str_replace(PHPFOX_PARENT_DIR, home(), $asset);
					$this->_sFooter .= '<script src="' . $url . '?v=' . Phpfox::internalVersion() . '"></script>';
				}
			}
		}

		if (!defined('PHPFOX_INSTALLER') && !$bundle) {
			$this->_sFooter .= "\t\t" . '<script type="text/javascript"> $Core.init(); </script>' . "\n";
		}

		if (isset($this->_meta['head'])) {
			$sData .= $this->_meta['head'];
			if (Phpfox::isAdmin()) {
				$this->_sFooter .= '<script>var page_editor_meta = ' . json_encode(['head' => $this->_meta['head']]) . ';</script>';
			}
		}

		(($sPlugin = Phpfox_Plugin::get('template_getheader_end')) ? eval($sPlugin) : false);

		if ($bReturnArray)
		{
			$aArrayData[] = $sData;

			return $aArrayData;
		}

		// Convert meta data
		$bHasNoDescription = false;
		if (count($this->_aMeta) && !PHPFOX_IS_AJAX_PAGE && !defined('PHPFOX_INSTALLER'))
		{
			$oPhpfoxParseOutput = Phpfox::getLib('parse.output');
			$aFind = array(
					'&lt;',
					'&gt;',
					'$'
			);
			$aReplace = array(
					'<',
					'>',
					'&36;'
			);

			foreach ($this->_aMeta as $sMeta => $sMetaValue)
			{
				$sMetaValue = str_replace($aFind, $aReplace, $sMetaValue);
				$sMetaValue = strip_tags($sMetaValue);
				$sMetaValue = str_replace(array("\n", "\r"), "", $sMetaValue);
				$bIsCustomMeta = false;

				switch ($sMeta)
				{
					case 'keywords':
						if (isset($this->_meta['keywords'])) {
							$sMetaValue = $this->_meta['keywords'];
							continue;
						}

						$sKeywordSearch = Phpfox::getParam('core.words_remove_in_keywords');
						if (!empty($sKeywordSearch))
						{
							$aKeywordsSearch = array_map('trim', explode(',', $sKeywordSearch));
						}
						$sMetaValue = $oPhpfoxParseOutput->shorten($oPhpfoxParseOutput->clean($sMetaValue), Phpfox::getParam('core.meta_keyword_limit'));

						$sMetaValue = trim(rtrim(trim($sMetaValue), ','));
						$aParts = explode(',', $sMetaValue);
						$sMetaValue = '';
						$aKeywordCache = array();
						foreach ($aParts as $sPart)
						{
							$sPart = trim($sPart);

							if (isset($aKeywordCache[$sPart]))
							{
								continue;
							}

							if (isset($aKeywordsSearch) && in_array(strtolower($sPart), array_map('strtolower', $aKeywordsSearch)))
							{
								continue;
							}

							$sMetaValue .= $sPart . ', ';

							$aKeywordCache[$sPart] = true;
						}
						$sMetaValue = rtrim(trim($sMetaValue), ',');
						break;
					case 'description':
						if (isset($this->_meta['description'])) {
							$sMetaValue = $this->_meta['description'];
							continue;
						}

						$bHasNoDescription = true;
						$sMetaValue = $oPhpfoxParseOutput->shorten($oPhpfoxParseOutput->clean($sMetaValue), Phpfox::getParam('core.meta_description_limit'));
						break;
					case 'robots':
						$bIsCustomMeta = false;
						break;
					default:
						$bIsCustomMeta = true;
						break;
				}
				$sMetaValue = str_replace('"', '\"', $sMetaValue);
				$sMetaValue = Phpfox_Locale::instance()->convert($sMetaValue);
				$sMetaValue = html_entity_decode($sMetaValue, null, 'UTF-8');
				$sMetaValue = str_replace(array('<', '>'), '', $sMetaValue);

				if ($bIsCustomMeta)
				{
					if ($sMeta == 'og:description')
					{
						$sMetaValue = $oPhpfoxParseOutput->shorten($oPhpfoxParseOutput->clean($sMetaValue), Phpfox::getParam('core.meta_description_limit'));
					}

					switch ($sMeta)
					{
						case 'canonical':
							$sCanonical = $sMetaValue;
							$sCanonical = preg_replace('/\/when\_([a-zA-Z0-9\-]+)\//i', '/', $sCanonical);
							$sCanonical = preg_replace('/\/show\_([a-zA-Z0-9\-]+)\//i', '/', $sCanonical);
							$sCanonical = preg_replace('/\/view\_\//i', '/', $sCanonical);

							$sData .= "\t\t<link rel=\"canonical\" href=\"{$sCanonical}\" />\n";

							break;
						default:
							$sData .= "\t\t<meta property=\"{$sMeta}\" content=\"{$sMetaValue}\" />\n";
							break;
					}

				}
				else
				{
					if (strpos($sData, 'meta name="'. $sMeta . '"') !== false)
					{
						$sData = preg_replace("/<meta name=\"{$sMeta}\" content=\"(.*?)\" \/>\n\t/i", "<meta" . ($sMeta == 'description' ? ' property="og:description" ' : '') . " name=\"{$sMeta}\" content=\"" . $sMetaValue . "\" />\n\t", $sData);

					}
					else
					{
						$sData = preg_replace('/<meta/', '<meta name="'. $sMeta . '" content="' . $sMetaValue . '" />' . "\n\t\t" . '<meta', $sData, 1);
					}
				}

			}

			if (!$bHasNoDescription)
			{
				$sData .= "\t\t" . '<meta name="description" content="' . Phpfox::getLib('parse.output')->clean(Phpfox_Locale::instance()->convert(Phpfox::getParam('core.description'))) . '" />' . "\n";
			}
		}

		if ($bundle) {
			$the_name = 'autoload-' . Phpfox::getFullVersion() . '.css';
			$path = PHPFOX_DIR_FILE . 'static' . PHPFOX_DS . $the_name;

			$callback = function ($file = null) {
				static $files = [], $flavor = '';

				if ($file == 'flavor') {
					return $flavor;
				}

				if ($file === null) {
					return $files;
				}

				$file = $file[1];

                $the_file = explode('?', (strpos($file, Phpfox::getLib('cdn')->getUrl(home())) === false ? $file : (PHPFOX_PARENT_DIR . str_replace(Phpfox::getLib('cdn')->getUrl(home()), '', $file))))[0];

                $files[] = $the_file;
			};
			$sData = preg_replace_callback('/<link href="(.*?)"(.*?)>/is', $callback, $sData);
			if (!file_exists($path)) {
			    $files = $callback();
                \Core\Registry::put(self::BUNDLE_CSS_FILES,$files);
				Phpfox_File_Minimize::instance()->css($path, $files);
			}

			$sData .= '<link href="' . Phpfox::getLib('cdn')->getUrl(home()) . 'PF.Base/file/static/' . $the_name . '?v=' . $this->getStaticVersion() . '" rel="stylesheet" type="text/css">';
		}

        if (!defined('PHPFOX_INSTALLER') && Phpfox::getParam('core.google_plus_page_url'))
        {
            $sData .= '<link href="' . Phpfox::getParam('core.google_plus_page_url') . '" rel="publisher"/>';
        }

		// Clear from memory
		$this->_aHeaders = array();
		$this->_aMeta = array();

		return $sData;
	}

	public $footer = '';

	private function _bundle() {
		if (defined('PHPFOX_INSTALLER')) {
			return false;
		}

		if (!setting('pf_core_bundle_js_css', false)) {
			return false;
		}

		$bundle = false;
		if (!PHPFOX_IS_AJAX
			&& !PHPFOX_IS_AJAX_PAGE
			&& !Phpfox::isAdminPanel()
		) {
			$bundle = true;
		}

		return $bundle;
	}

	public function getFooter() {
		$bundle = $this->_bundle();

		$this->_sFooter .= '<div id="show-side-panel"><span></span></div>';

		Core\Event::trigger('lib_phpfox_template_getfooter', $this);

		$this->_sFooter .= $this->footer;

		if (!defined('PHPFOX_INSTALLER')) {
			foreach ((new Core\App())->all() as $App) {
				if ($App->footer && is_array($App->footer)) {
					foreach ($App->footer as $footer) {
						$this->_sFooter .= $footer;
					}
				}

				if ($App->js && is_array($App->js)) {
					foreach ($App->js as $js) {
						$this->_sFooter .= '<script src="' . $js . '"></script>';
					}
				}

				if ($App->js && is_object($App->js)) {
					foreach ($App->js as $js => $setting) {
						if (setting($setting)) {
							$this->_sFooter .= '<script src="' . $js . '"></script>';
						}
					}
				}
			}
		}

		$this->_sFooter .=  '<script src="' . Phpfox::getLib('cdn')->getUrl(Phpfox::getParam('core.url_static_script')) . 'bootstrap.js"></script>';

		(($sPlugin = Phpfox_Plugin::get('template_getfooter_end')) ? eval($sPlugin) : false);

		if ($bundle) {
            $the_controller = str_replace('/', '__', Phpfox_Module::instance()->getFullControllerName()) . '-' . Phpfox::getFullVersion() . '.js';
            $module_path = PHPFOX_DIR_FILE . 'static' . PHPFOX_DS . $the_controller;

            $the_name = 'autoload-' . Phpfox::getFullVersion() . '.js';
            $path = PHPFOX_DIR_FILE . 'static' . PHPFOX_DS . $the_name;

            $callback = function ($file = null) use ($the_controller) {
                static $files = [], $modules = [];

                if ($file === null) {
                    return [
                        'files'   => $files,
                        'modules' => $modules
                    ];
                }

                $script = $file[0];
                $file = $file[1];

                if (substr($file, -11) == '/user/boot/') {
                    return $script;
                }

                $the_file = explode('?', str_replace(Phpfox::getLib('cdn')->getUrl(home()), '', $file))[0];

                if (substr($the_file, 0, 4) == 'http'
                ) {
                    return $script;
                }

                if (substr($the_file, 0, 15) == 'PF.Base/module/') {
                    $modules[] = PHPFOX_PARENT_DIR . $the_file;

                    return '';
                }

                $files[] = PHPFOX_PARENT_DIR . $the_file;

                return '';
            };

            $this->_sFooter = preg_replace_callback('/<script src="(.*?)"><\/script>/is', $callback, $this->_sFooter);
            $module_files = $callback()['modules'];

            $aBundleFiles = \Core\Registry::get(self::BUNDLE_CSS_FILES,[]);
            $aBundleFiles = array_merge($aBundleFiles, $callback()['files'], $module_files);

            $aBundleFiles = array_map(function($file) {
                $file = str_replace(PHPFOX_PARENT_DIR, Phpfox::getLib('cdn')->getUrl(Phpfox::getBaseUrl()), $file);
                $file = str_replace(PHPFOX_DS, '/', $file);
                return $file;
            }, $aBundleFiles);

            $extra = ';$Core.core_bundle_files = ' . json_encode($aBundleFiles) . ';';

            if (!file_exists($path)) {
                Phpfox_File_Minimize::instance()->js($path, $callback()['files'], $extra);
            }


            if ($module_files && !file_exists($module_path)) {
                Phpfox_File_Minimize::instance()->js($module_path, $callback()['modules']);
            }

            $this->_sFooter = str_replace('<!-- autoload -->', '<script src="' . Phpfox::getLib('cdn')->getUrl(home()) . 'PF.Base/file/static/' . $the_name . '?v=' . $this->getStaticVersion() . '"></script>', $this->_sFooter);
            if ($module_files) {
                $this->_sFooter .= '<script src="' . Phpfox::getLib('cdn')->getUrl(home()) . 'PF.Base/file/static/' . $the_controller . '?v=' . $this->getStaticVersion() . '"></script>';
            }

            $this->_sFooter .= '<script>$Core.init();</script>';

        }

		// echo htmlspecialchars($this->_sFooter); exit;

		return $this->_sFooter;
	}

	/**
	 * Get the template header file if it exists.
	 *
	 * @return mixed File path if it exists, otherwise FALSE.
	 */
	public function getHeaderFile()
	{
		$sFile = $this->getStyle('php', 'header.php');
		$sFile = str_replace(Phpfox::getParam('core.path_file'), PHPFOX_DIR, $sFile);

		if (file_exists($sFile))
		{
			return $sFile;
		}

		return false;
	}

    /**
     * Gets the full path of a file based on the current style being used.
     *
     * @param string $sType Type of file we are working with.
     * @param string $sValue File name.
     * @param string $sModule Module name. Only if its part of a module.
     * @param bool $isApp
     * @return string Returns the full path to the item.
     */
	public function getStyle($sType = 'css', $sValue = null, $sModule = null, $isApp = false)
	{
	    if($isApp){
            $sUrl = dirname(Phpfox::getParam('core.path_file')) . PHPFOX_DS. 'PF.Site'.PHPFOX_DS.'Apps' . PHPFOX_DS . $sModule . PHPFOX_DS . 'assets'  . PHPFOX_DS;
            return  $sUrl . $sValue;
        }
		if ($sModule !== null)
		{
			if ($sType == 'static_script')
			{
				$sType = 'jscript';
			}

			$sUrl = Phpfox::getParam('core.path_file') . 'module' . PHPFOX_DS . $sModule . PHPFOX_DS . 'static' . PHPFOX_DS . $sType . PHPFOX_DS;
			$sDir = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DS . 'static' . PHPFOX_DS . $sType . PHPFOX_DS;

			if ($sType == 'jscript')
			{
				$sPath = $sUrl . $sValue;
			}
			else
			{
				$bPass = false;
				if (file_exists($sDir . $this->_sThemeFolder . PHPFOX_DS . $this->_sStyleFolder . PHPFOX_DS . $sValue))
				{
					$bPass = true;
					$sPath = $sUrl . $this->_sThemeFolder . '/' . $this->_sStyleFolder . '/' . $sValue;
				}

				if ($bPass === false)
				{
					$sPath = $sUrl . 'default/default/' . $sValue;
				}
			}

			return isset($sPath) ? $sPath : '';
		}

		if ($sType == 'static_script')
		{
			$sPath = Phpfox::getParam('core.url_static_script') . $sValue;
		}
		else
		{
			$sPath = (defined('PHPFOX_INSTALLER') ? Phpfox_Installer::getHostPath() : Phpfox::getParam('core.path_file')) . 'theme/' . $this->_sThemeLayout . '/' . $this->_sThemeFolder . '/style/' . $this->_sStyleFolder . '/' . $sType . '/';
			if ($sPlugin = Phpfox_Plugin::get('library_template_getstyle_1')){eval($sPlugin);if (isset($bReturnFromPlugin)) return $bReturnFromPlugin;}
			if ($sValue !== null)
			{
				$bPass = false;

				if (isset($this->_aTheme['style_parent_id']) && $this->_aTheme['style_parent_id'] > 0)
				{
					$bPass = false;
					if (file_exists(PHPFOX_DIR . 'theme' . PHPFOX_DS . $this->_sThemeLayout . PHPFOX_DS . $this->_sThemeFolder . PHPFOX_DS . 'style' . PHPFOX_DS . $this->_sStyleFolder . PHPFOX_DS . $sType . PHPFOX_DS . $sValue))
					{
						$bPass = true;
						$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/' . $this->_sThemeLayout . '/' . $this->_sThemeFolder . '/' . 'style' . '/' . $this->_sStyleFolder . '/' . $sType . '/' . $sValue;
					}

					if (isset($this->_aTheme['parent_theme_folder']))
					{
						if ($bPass === false && file_exists(PHPFOX_DIR . 'theme' . PHPFOX_DS . $this->_sThemeLayout . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . 'style' . PHPFOX_DS . $this->_aTheme['parent_style_folder'] . PHPFOX_DS . $sType . PHPFOX_DS . $sValue))
						{
							$bPass = true;
							$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/' . $this->_sThemeLayout . '/' . $this->_aTheme['parent_theme_folder'] . '/' . 'style' . '/' . $this->_aTheme['parent_style_folder'] . '/' . $sType . '/' . $sValue;
						}
					}
					else
					{
						if ($bPass === false && file_exists(PHPFOX_DIR . 'theme' . PHPFOX_DS . $this->_sThemeLayout . PHPFOX_DS . $this->_sThemeFolder . PHPFOX_DS . 'style' . PHPFOX_DS . $this->_aTheme['parent_style_folder'] . PHPFOX_DS . $sType . PHPFOX_DS . $sValue))
						{
							$bPass = true;
							$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/' . $this->_sThemeLayout . '/' . $this->_sThemeFolder . '/' . 'style' . '/' . $this->_aTheme['parent_style_folder'] . '/' . $sType . '/' . $sValue;
						}
					}
				}
				else
				{
					if (!defined('PHPFOX_INSTALLER'))
					{
						if (file_exists(PHPFOX_DIR . 'theme' . PHPFOX_DS . $this->_sThemeLayout . PHPFOX_DS . $this->_sThemeFolder . PHPFOX_DS . 'style' . PHPFOX_DS . $this->_sStyleFolder . PHPFOX_DS . $sType . PHPFOX_DS . $sValue))
						{
							$bPass = true;
							$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/' . $this->_sThemeLayout . '/' . $this->_sThemeFolder . '/' . 'style' . '/' . $this->_sStyleFolder . '/' . $sType . '/' . $sValue;
						}
					}
				}

				if ($bPass === false)
				{
					if (defined('PHPFOX_INSTALLER'))
					{
						$sPath = (defined('PHPFOX_INSTALLER') ? Phpfox_Installer::getHostPath() : Phpfox::getParam('core.path_file')) . 'theme' . '/' . $this->_sThemeLayout . '/' . 'default' . '/' . 'style' . '/' . 'default' . '/' . $sType . '/' . $sValue;
					}
					else
					{
						if (file_exists(PHPFOX_DIR . 'theme' . '/' . $this->_sThemeLayout . '/' . 'default' . '/' . 'style' . '/' . 'default' . '/' . $sType . '/' . $sValue))
						{
							$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/' . $this->_sThemeLayout . '/' . 'default' . '/' . 'style' . '/' . 'default' . '/' . $sType . '/' . $sValue;
						}
						else
						{
							if (file_exists(PHPFOX_DIR . 'theme' . '/frontend/' . $this->_sThemeFolder . '/' . 'style' . '/' . $this->_sStyleFolder . '/' . $sType . '/' . $sValue))
							{
								$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/frontend/' . $this->_sThemeFolder . '/' . 'style' . '/' . $this->_sStyleFolder . '/' . $sType . '/' . $sValue;
							}
							else
							{
								$sPath = Phpfox::getParam('core.path_file') . 'theme' . '/frontend/' . 'default' . '/' . 'style' . '/' . 'default' . '/' . $sType . '/' . $sValue;
							}
						}
					}
				}
			}
		}

		return $sPath;
	}

	/**
	 * Assign a variable so we can use it within an HTML template.
	 *
	 * PHP assign:
	 * <code>
	 * Phpfox_Template::instance()->assign('foo', 'bar');
	 * </code>
	 *
	 * HTML usage:
	 * <code>
	 * {$foo}
	 * // Above will output: bar
	 * </code>
	 *
	 * @param mixed $mVars STRING variable name or ARRAY of variables to assign with both keys and values.
	 * @param string $sValue Variable value, only if the 1st argument is a STRING.
	 * @return $this
	 */
	public function assign($mVars, $sValue = '')
	{
		if (!is_array($mVars))
		{
			$mVars = array($mVars => $sValue);
		}

		foreach ($mVars as $sVar => $sValue)
		{
			if (is_array($sValue) && count($sValue)) {
				if (isset($sValue[0])) {
					$first = $sValue[0];
					if (is_object($first) && method_exists($first, '__toArray')) {
						$sValue = array_map(function($val) {

							return (array) $val;
						}, $sValue);
					}
				}
			}
			if (is_object($sValue) && method_exists($sValue, '__toArray')) {
				$sValue = (array) $sValue;
			}

			$this->_aVars[$sVar] = $sValue;
		}

		return $this;
	}

	/**
	 * Get a variable we assigned with the method assign().
	 *
	 * @see self::assign()
	 * @param string $sName Variable name.
	 * @return string Variable value.
	 */
	public function getVar($sName = null)
	{
		if ($sName === null) {
			return $this->_aVars;
		}

		if (isset($this->_aVars[$sName]) && is_object($this->_aVars[$sName])) {
			$this->_aVars[$sName] = (array) $this->_aVars[$sName];
		}

		return (isset($this->_aVars[$sName]) ? $this->_aVars[$sName] : '');
	}

	/**
	 * Clean all or a specific variable from memory.
	 *
	 * @param mixed $mName Variable name to destroy, or leave blank to destory all variables or pass an ARRAY of variables to destroy.
	 */
	public function clean($mName = '')
	{
		if ($mName)
		{
			if (!is_array($mName))
			{
				$mName = array($mName);
			}

			foreach ($mName as $sName)
			{
				unset($this->_aVars[$sName]);
			}

			return;
		}

		unset($this->_aVars);
	}

	/**
	 * Loads the current template.
	 *
	 * @param string $sName Layout name.
	 * @param bool $bReturn TRUE to return the template code, FALSE will echo it.
	 * @return mixed STRING if 2nd argument is TRUE. Otherwise NULL.
	 */
	public function getLayout($sName, $bReturn = false)
	{
		$this->_getFromCache($this->getLayoutFile($sName));

		if ($bReturn)
		{
			return $this->_returnLayout();
		}
	}

	/**
	 * Get the full path of the current layout file.
	 *
	 * @param string $sName Name of the layout file.
	 * @return string Full path to the layout file.
	 */
	public function getLayoutFile($sName)
	{
		$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . $this->_sThemeFolder . PHPFOX_DS . 'template' . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
		if ($sPlugin = Phpfox_Plugin::get('library_template_getlayoutfile_1')){eval($sPlugin);if (isset($bReturnFromPlugin)) return $bReturnFromPlugin;}
		if (!file_exists($sFile))
		{
			if ($this->_aTheme['theme_parent_id'] > 0 && !empty($this->_aTheme['parent_theme_folder']))
			{
				$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . 'template' . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
			}
		}

		if (!file_exists($sFile))
		{
			$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . 'default' . PHPFOX_DS . 'template' . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
		}

		if ($this->_sThemeLayout == 'mobile' && !file_exists($sFile))
		{
			$sFile = PHPFOX_DIR_THEME . 'frontend' . PHPFOX_DS . 'default' . PHPFOX_DS . 'template' . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
		}

		return $sFile;
	}

	public function getBuiltFile($sTemplate, $feed_id = null)
	{
		$sCacheName = 'block_' . $sTemplate;

		(($sPlugin = Phpfox_Plugin::get('template_gettemplate_pass')) ? eval($sPlugin) : false);

		if (isset($skip_layout)) {

		} else {
			if (!$this->_isCached($sCacheName))
			{
				$mContent = Phpfox_Template::instance()->getTemplateFile($sTemplate, true);
				if (is_array($mContent))
				{
					$mContent = $mContent[0];
				}
				else
				{
					$mContent = file_get_contents($mContent);
				}

				$oTplCache = Phpfox::getLib('template.cache');
				$oTplCache->compile($this->_getCachedName($sCacheName), $mContent, true);
			}

			$path = $this->_getCachedName($sCacheName);
		}

		$do_cache = false;

		if (redis()->enabled()
			&& $feed_id
			&& !defined('PHPFOX_IS_PAGES_VIEW')
			&& strtolower($_SERVER['REQUEST_METHOD'] == 'GET')
		) {
			$do_cache = true;
			$o = ob_get_contents();
			ob_clean();
		}
		if (!isset($skip_layout) && isset($path))
			require($path);

		if ($do_cache) {
			$d = ob_get_contents();

			redis()->set('feed_focus_' . $feed_id, [
				'html' => $d
			]);

			ob_clean();
			echo $o . $d;
		}
	}

	/**
	 * Get the full path to the modular template file we are loading.
	 *
	 * @param string $sTemplate Name of the file.
	 * @param bool $bCheckDb TRUE to check the database if the file exists there.
	 * @return string Full path to the file we are loading.
	 */
	public function getTemplateFile($sTemplate, $bCheckDb = false)
	{
		(($sPlugin = Phpfox_Plugin::get('template_gettemplatefile')) ? eval($sPlugin) : false);

		if (!isset($sFile)) {
			$aParts = explode('.', $sTemplate);
			$sModule = $aParts[0];
			$sFolder = ($this->_sThemeFolder == 'bootstrap') ? 'default' : $this->_sThemeFolder;

			if (defined('PHPFOX_INSTALLER')) {
				return ['', ''];
			}
			unset($aParts[0]);
			$sName = implode('.', $aParts);
			$sName = str_replace('.', PHPFOX_DS, $sName);
			$bPass = false;

			if(!empty($this->_aDirectoryNames[$sModule])){
				$filename = $this->_aDirectoryNames[$sModule] .PHPFOX_DS. implode(PHPFOX_DS, $aParts). '.html.php';
				if(file_exists($filename)){
					$sFile = $filename;
					$bPass =  true;
				}
			}

			if (!$bPass && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX)) {
				$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
				$bPass = true;
			}

			if ($bPass === false && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX)) {
				$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
				$bPass = true;
			}

			if ($bPass === false && isset($aParts[2]) && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_DS . $aParts[2] . PHPFOX_TPL_SUFFIX)) {
				$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $sFolder . PHPFOX_DS . $sName . PHPFOX_DS . $aParts[2] . PHPFOX_TPL_SUFFIX;
				$bPass = true;
			}

			if (isset($this->_aTheme['theme_parent_id']) && $this->_aTheme['theme_parent_id'] > 0 && !empty($this->_aTheme['parent_theme_folder'])) {
				if ($bPass === false && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX)) {
					$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
					$bPass = true;
				}

				if ($bPass === false && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX)) {
					$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
					$bPass = true;
				}

				if ($bPass === false && isset($aParts[2]) && file_exists(PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_DS . $aParts[2] . PHPFOX_TPL_SUFFIX)) {
					$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . $sName . PHPFOX_DS . $aParts[2] . PHPFOX_TPL_SUFFIX;
					$bPass = false;
				}
			}

			if (!isset($sFile)) {
				$sFile = PHPFOX_DIR_MODULE . $sModule . PHPFOX_DIR_MODULE_TPL . PHPFOX_DS . 'default' . PHPFOX_DS . $sName . PHPFOX_TPL_SUFFIX;
			}
		}
		
		if (!file_exists($sFile))
		{
			Phpfox_Error::trigger('Unable to load module template: ' . $sModule .'->' . $sName, E_USER_ERROR);
		}

		return $sFile;
	}

	/**
	 * Get a template that has already been built.
	 *
	 * @param string $sLayout Template name.
	 * @param string $sCacheName Cache name of the file.
	 * @return string HTML layout of the file.
	 */
	public function getBuiltTemplate($sLayout, $sCacheName)
	{
		if (!$this->_isCached($sCacheName))
		{
			if (!defined('PHPFOX_INSTALLER') && !defined('PHPFOX_LIVE_TEMPLATES') && $this->_bIsAdminCp === false)
			{
				$oDb = Phpfox_Database::instance();
				$aTemplate = $oDb->select('html_data')
						->from(Phpfox::getT('theme_template'))
						->where("folder = '" . $this->_sThemeFolder . "' AND type_id = 'layout' AND name = '" . $oDb->escape($sLayout) . ".html.php'")
						->execute('getSlaveRow');
			}

			$sLayoutContent = (isset($aTemplate['html_data']) ? $aTemplate['html_data'] : file_get_contents($this->getLayoutFile($sLayout)));

			$sLayoutContent = str_replace('{layout_content}', '<?php echo $this->_aVars[\'sContent\']; ?>', $sLayoutContent);

			$oTplCache = Phpfox::getLib('template.cache');
			$oTplCache->compile($this->_getCachedName($sCacheName), $sLayoutContent, true);
		}

		$sOriginalContent = ob_get_contents();

		ob_clean();

		require($this->_getCachedName($sCacheName));

		$sCurrentContent = $this->_returnLayout();

		echo $sOriginalContent;

		return $sCurrentContent;
	}

	/**
	 * Get the current template data.
	 *
	 * @param string $sTemplate Template name.
	 * @param bool $bReturn TRUE to return its content or FALSE to just echo it.
	 * @return mixed STRING content only if the 2nd argument is TRUE.
	 */
	public function getTemplate($sTemplate, $bReturn = false)
	{
		(($sPlugin = Phpfox_Plugin::get('template_gettemplate')) ? eval($sPlugin) : false);

		$sFile = $this->getTemplateFile($sTemplate);

		if ($bReturn)
		{
			$sOriginalContent = ob_get_contents();

			ob_clean();
		}

		(($sPlugin = Phpfox_Plugin::get('template_gettemplate_pass')) ? eval($sPlugin) : false);

		if (isset($skip_layout)) {

		} else {
			if ($this->_sSetLayout) {
				if (!$this->_isCached($sFile)) {
					if (!defined('PHPFOX_INSTALLER') && !defined('PHPFOX_LIVE_TEMPLATES') && $this->_bIsAdminCp === false) {
						$oDb = Phpfox_Database::instance();

						$aTemplate = $oDb->select('html_data')
							->from(Phpfox::getT('theme_template'))
							->where("folder = '" . $this->_sThemeFolder . "' AND type_id = 'layout' AND name = '" . $oDb->escape($this->_sSetLayout) . ".html.php'")
							->execute('getSlaveRow');
					}

					$sLayoutContent = (isset($aTemplate['html_data']) ? $aTemplate['html_data'] : file_get_contents($this->getLayoutFile($this->_sSetLayout)));
					$sName = $this->_getCachedName($sFile);

					if (!defined('PHPFOX_INSTALLER') && !defined('PHPFOX_LIVE_TEMPLATES')) {
						if (preg_match("/(.*?)template_(.*?)_template_(.*?)_(.*?)_(.*?)\.php$/i", $this->_getCachedName($sFile), $aMatches) && isset($aMatches[5]) && !preg_match("/admincp_(.*?)/", $aMatches[4])) {
							$oDb = Phpfox_Database::instance();
							$aSubTemplate = $oDb->select('html_data')
								->from(Phpfox::getT('theme_template'))
								->where("folder = '" . $this->_sThemeFolder . "' AND type_id = '" . $oDb->escape($aMatches[4]) . "' AND module_id = '" . $oDb->escape($aMatches[2]) . "' AND name = '" . $oDb->escape(str_replace('_', '/', $aMatches[5])) . "'")
								->execute('getSlaveRow');
						} elseif (preg_match("/(.*?)template_(.*?)_(.*?)_template_(.*?)\.php$/i", str_replace('template/', 'template_', $sName), $aMatches) && isset($aMatches) && $aMatches[2] == 'frontend') {
							$oDb = Phpfox_Database::instance();
							$aSubTemplate = $oDb->select('html_data')
								->from(Phpfox::getT('theme_template'))
								->where("folder = '" . $this->_sThemeFolder . "' AND type_id = 'layout' AND name = '" . $oDb->escape($aMatches[4]) . "'")
								->execute('getSlaveRow');
						} elseif (preg_match("/(.*?)template_(.*?)_template_(.*?)_(.*?)_(.*?)\.php$/i", str_replace('template/', 'template_', $sName), $aMatches) && isset($aMatches[5]) && !preg_match("/admincp_(.*?)/", $aMatches[4])) {
							$oDb = Phpfox_Database::instance();

							$aSubTemplate = $oDb->select('html_data')
								->from(Phpfox::getT('theme_template'))
								->where("folder = '" . $this->_sThemeFolder . "' AND type_id = '" . $oDb->escape($aMatches[4]) . "' AND module_id = '" . $oDb->escape($aMatches[2]) . "' AND name = '" . $oDb->escape(str_replace('_', '/', $aMatches[5])) . "'")
								->execute('getSlaveRow');
						}
					}

					$sLayoutContent = str_replace('{layout_content}', (isset($aSubTemplate['html_data']) ? $aSubTemplate['html_data'] : file_get_contents($sFile)), $sLayoutContent);

					$oTplCache = Phpfox::getLib('template.cache');

					$oTplCache->compile($this->_getCachedName($sFile), $sLayoutContent, true, (isset($aSubTemplate['html_data']) ? true : false));
				}

				$this->_sSetLayout = '';
				$this->_requireFile($sFile);
				$this->_sSetLayout = '';
			} else {
				$this->_getFromCache($sFile, $sTemplate);
			}
		}

		if ($bReturn)
		{
			$sReturn = $this->_returnLayout();
			echo isset($sOriginalContent) ? $sOriginalContent : '';
			return $sReturn;
		}
        return null;
	}

	/**
	 * Rebuild a cached menu.
	 *
	 * @param string $sConnection Menu connection.
	 * @param array $aNewUrl ARRAY of the new values.
	 * @return object Return self.
	 */
	public function rebuildMenu($sConnection, $aNewUrl)
	{
		$this->_aNewUrl[$sConnection] = $aNewUrl;

		return $this;
	}

	/**
	 * Remove a URL from a built cached menu.
	 *
	 * @param string $sConnection Menu connection.
	 * @param string $sUrl URL value to identify what menu to remove.
	 * @return object Return self.
	 */
	public function removeUrl($sConnection, $sUrl)
	{
		$this->_aRemoveUrl[$sConnection][$sUrl] = true;

		return $this;
	}

	private $_aMenus = [];
	public function setMenu($menus) {
		foreach ($menus as $connection => $menu) {
			$this->_aMenus[$connection] = $menu;
		}
	}

	/**
	 * Gets all the sites custom menus, such as the Main, Header, Footer and Sub menus.
	 * Since information is stored in the database we cache the information so we only run
	 * the query once.
	 *
	 * @param string $sConnection Current page we are viewing (Example: account/login)
	 * @return array $aMenus Is an array of the menus data
	 */
	public function getMenu($sConnection = null)
	{
		$oCache = Phpfox::getLib('cache');
		$oReq = Phpfox_Request::instance();

		(($sPlugin = Phpfox_Plugin::get('template_template_getmenu_1')) ? eval($sPlugin) : false);
		if ($sConnection === null)
		{
			$sConnection = Phpfox_Module::instance()->getFullControllerName();
			$bIsModulePage = true;

			$sConnection = preg_replace('/(.*)\.profile/i', '\\1.index', $sConnection);
			
			if (($sConnection == 'user.photo' && $oReq->get('req3') == 'register') || ($sConnection == 'invite.index' && $oReq->get('req2') == 'register'))
			{
				return array();
			}
		}
		$sConnection = strtolower(str_replace('/','.',$sConnection));
		if ($sConnection == 'profile.private' || $sConnection == 'profile.points' || $sConnection == 'profile.app')
		{
			return array();
		}

		$sCachedId = $oCache->set(array('theme', 'menu_' . str_replace(array('/', '\\'), '_', $sConnection) . (Phpfox::isUser() ? Phpfox::getUserBy('user_group_id') : 0)));

		if ((!($aMenus = $oCache->get($sCachedId))) && is_bool($aMenus) && !$aMenus)
		{
			$aParts = explode('.', $sConnection);
			$aMenus1 = $this->
            _getMenu($sConnection);
			$aCached = array();
			foreach ($aMenus1 as $aMenu1)
			{
				$aCached[] = $aMenu1['menu_id'];
			}

			$aMenus2 = $this->_getMenu($aParts[0]);
			foreach ($aMenus2 as $iKey => $aMenu2)
			{
				if (in_array($aMenu2['menu_id'], $aCached))
				{
					unset($aMenus2[$iKey]);
				}
			}

			$aFinal = array_merge($aMenus1, $aMenus2);

			$aMenus = array();
			foreach ($aFinal as $aMenu)
			{
				// test if this menu points to a real location
				if (isset($aMenu['url']) && !empty($aMenu['url']) && strpos($aMenu['url'], 'http') !== false)
				{
					$aMenu['external'] = true;

				}
				else if (isset($aMenu['url']) && $aMenu['url'] == '#')
				{
					$aMenu['no_link'] = true;
				}
				if ($aMenu['parent_id'] > 0)
				{
					continue;
				}

				$aMenus[$aMenu['menu_id']] = $aMenu;
			}
			$aParents = Phpfox_Database::instance()->select('m.menu_id, m.parent_id, m.m_connection, m.var_name, m.disallow_access, mo.module_id AS module, m.url_value AS url, mo.is_active AS module_is_active')
					->from(Phpfox::getT('menu'), 'm')
					->join(Phpfox::getT('module'), 'mo', 'mo.module_id = m.module_id AND mo.is_active = 1')
					->join(Phpfox::getT('product'), 'p', 'm.product_id = p.product_id AND p.is_active = 1')
					->where("m.parent_id > 0 AND m.is_active = 1")
					->order('m.ordering ASC')
					->execute('getRows');

			if (count($aParents))
			{
				foreach ($aParents as $aParent)
				{
					if (!isset($aMenus[$aParent['parent_id']]))
					{
						continue;
					}
					$aMenus[$aParent['parent_id']]['children'][] = $aParent;
				}
			}

			if ($sPlugin = Phpfox_Plugin::get('template_template_getmenu_2')){eval($sPlugin);}
			$oCache->save($sCachedId, $aMenus);
		}

		if (isset($this->_aMenus[$sConnection])) {
			$aMenus = [$this->_aMenus[$sConnection]];
		}

		if ($sPlugin = Phpfox_Plugin::get('template_template_getmenu_3')){eval($sPlugin);}

            if (!is_array($aMenus))
		{
            return array();
		}

		if ($sConnection == 'main' && Phpfox::isUser())
		{
			$sUserMenuCache = Phpfox::getLib('cache')->set(array('user', 'nbselectname_' . Phpfox::getUserId()));
			if (($aUserMenusCache = Phpfox::getLib('cache')->get($sUserMenuCache)) === false)
			{
				$aUserMenus = Phpfox_Database::instance()->select('*')->from(Phpfox::getT('theme_umenu'))->where('user_id = ' . (int) Phpfox::getUserId())->execute('getSlaveRows');
				foreach ((array) $aUserMenus as $aUserMenu)
				{
					$aUserMenusCache[$aUserMenu['menu_id']] = true;
				}
				if ($aUserMenusCache === false) {
                    $aUserMenusCache = [];
                }
				Phpfox::getLib('cache')->save($sUserMenuCache, $aUserMenusCache);
			}
		}

		foreach ($aMenus as $iKey => $aMenu)
		{
			if (substr($aMenu['url'], 0, 1) == '#')
			{
				$aMenus[$iKey]['css_name'] = 'js_core_menu_' . str_replace('#', '', str_replace('-', '_', $aMenu['url']));
			}

			if (Phpfox::isModule('ad') && ($aMenu['url'] == 'ad' || $aMenu['url'] == 'ad.index') && !Phpfox::getUserParam('ad.can_create_ad_campaigns'))
			{
				unset($aMenus[$iKey]);

				continue;
			}

			if ($aMenu['url'] == 'mail.compose' && Phpfox::getUserParam('mail.restrict_message_to_friends') && !Phpfox::isModule('friend'))
			{
				unset($aMenus[$iKey]);

				continue;
			}

			if (isset($aUserMenusCache[$aMenu['menu_id']]))
			{
				$aMenus[$iKey]['is_force_hidden'] = true;
			}
			if(defined('PHPFOX_IS_PAGES_VIEW'))
			{
				if (Phpfox::isModule('pages') && $aMenu['url'] == 'blog.add')
				{
					$iPage = $this->_aVars['aPage']['page_id'];

					$aMenus[$iKey]['url'] = 'blog.add.module_pages.item_' . $iPage;
				}
				if (Phpfox::isModule('pages') && $aMenu['url'] == 'event.add')
				{
					$iPage = $this->_aVars['aPage']['page_id'];

					$aMenus[$iKey]['url'] = 'event.add.module_pages.item_' . $iPage;
				}
				if (Phpfox::isModule('pages') && $aMenu['url'] == 'music.upload')
				{
					$iPage = $this->_aVars['aPage']['page_id'];

					$aMenus[$iKey]['url'] = 'music.upload.module_pages.item_' . $iPage;
				}
				if (Phpfox::isModule('pages') && $aMenu['url'] == 'photo.add')
				{
					$iPage = $this->_aVars['aPage']['page_id'];

					$aMenus[$iKey]['url'] = 'photo.add.module_pages.item_' . $iPage;
				}
				(($sPlugin = Phpfox_Plugin::get('template_template_getmenu_in_pages')) ? eval($sPlugin) : false);
			}

			if ((trim($aMenu['url'], '/') == $oReq->get('req1')) ||
					(empty($aMenu['url']) && $oReq->get('req1') == PHPFOX_MODULE_CORE) ||
					($this->_sUrl !== null && $this->_sUrl == $aMenu['url']) ||
					(str_replace('/','.',$oReq->get('req1').$oReq->get('req2')) == str_replace('.','',$aMenu['url'])))
			{
				$aMenus[$iKey]['is_selected'] = true;
			}

			if ($aMenu['url'] == 'admincp')
			{
				if (!Phpfox::isAdmin())
				{
					unset($aMenus[$iKey]);

					continue;
				}
			}
			else
			{
				if (!empty($aMenu['disallow_access']))
				{
					$aUserGroups = unserialize($aMenu['disallow_access']);
					if (in_array(Phpfox::getUserBy('user_group_id'), $aUserGroups))
					{
						unset($aMenus[$iKey]);

						continue;
					}
				}

				if (isset($aMenu['children']) && is_array($aMenu['children']))
				{
					foreach ($aMenu['children'] as $iChildMenuMain => $aChildMenuMain)
					{
						if (!empty($aChildMenuMain['disallow_access']))
						{
							$aUserGroups = unserialize($aChildMenuMain['disallow_access']);
							if (in_array(Phpfox::getUserBy('user_group_id'), $aUserGroups))
							{
								unset($aMenus[$iKey]['children'][$iChildMenuMain]);
							}
						}
					}
				}
			}

			if (isset($this->_aNewUrl[$sConnection]))
			{
				$aMenus[$iKey]['url'] = $this->_aNewUrl[$sConnection][0] . '.' . implode('.', $this->_aNewUrl[$sConnection][1]) . '.' . $aMenu['url'];
			}

			if (isset($this->_aRemoveUrl[$sConnection][$aMenu['url']]))
			{
				unset($aMenus[$iKey]);

				continue;
			}
		}
		return $aMenus;
	}

	public function menu($title, $url, $extra = '') {
		if (substr($url, 0, 5) != 'http:' && $url != '#') {
			$url = \Phpfox_Url::instance()->makeUrl($url);
		}
		$css_class = '';
		if (is_array($extra)) {
			$css_class = $extra['css_class'];
			$extra = '';
		}
		$this->assign('customMenu', [
				'title' => $title,
				'url' => $url,
				'extra' => $extra,
				'css_class' => $css_class
		]);

		return $this;
	}

	/**
	 * Set the current URL for the site.
	 *
	 * @param string $sUrl URL value.
	 * @return object Return self.
	 */
	public function setUrl($sUrl)
	{
		$this->_sUrl = $sUrl;

		return $this;
	}

	/**
	 * Load and get the XML information about the theme used when custom designing a profile.
	 *
	 * @param string $sXml XML id.
	 * @return string ARRAY of XML data.
	 */
	public function getXml($sXml)
	{
		static $aXml = array();

		if (isset($aXml[$sXml]))
		{
			return $aXml[$sXml];
		}

		$oCache = Phpfox::getLib('cache');
		$sCacheId = $oCache->set(array('theme', 'theme_xml_' . $this->_sThemeLayout));

		if (!($aXml = $oCache->get($sCacheId)))
		{
			$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . $this->_sThemeFolder . PHPFOX_DS . 'xml' . PHPFOX_DS . 'phpfox.xml.php';
			if (!empty($this->_aTheme['parent_theme_folder']) && !file_exists($sFile))
			{
				$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . $this->_aTheme['parent_theme_folder'] . PHPFOX_DS . 'xml' . PHPFOX_DS . 'phpfox.xml.php';
			}

			if (!file_exists($sFile))
			{
				$sFile = PHPFOX_DIR_THEME . $this->_sThemeLayout . PHPFOX_DS . 'default' . PHPFOX_DS . 'xml' . PHPFOX_DS . 'phpfox.xml.php';
			}

			$aXml = Phpfox::getLib('xml.parser')->parse(file_get_contents($sFile));

			$oCache->save($sCacheId, $aXml);
		}

		return $aXml[$sXml];
	}

	/**
	 * Build subsection menu. Also assigns all variables to the template for us.
	 *
	 * @param string $sSection Internal section URL string.
	 * @param array $aFilterMenu Array of menu.
	 */
	public function buildSectionMenu($sSection, $aFilterMenu, $is_app = false)
	{
		// Add a hook with return here
		$sView = Phpfox_Request::instance()->get('view');
		$aFilterMenuCache = array();
		$iFilterCount = 0;
		$bHasMenu = false;

		foreach ($aFilterMenu as $sMenuName => $sMenuLink)
		{
			if (is_numeric($sMenuName))
			{
				$aFilterMenuCache[] = array();

				continue;
			}
			$sMenuName = str_replace('phpfox_numeric_friend_list_', '', $sMenuName);
			$bForceActive = false;
			$bIsView = true;
			if (strpos($sMenuLink, '.'))
			{
				$bIsView = false;
			}

			$iFilterCount++;

			if ($is_app) {
				$u = parse_url($sMenuLink);
				if (isset($u['query'])) {
					parse_str($u['query'], $p);
					if (isset($p['view']) && $p['view'] == $sView) {
						$bForceActive = true;
						$bHasMenu = true;
					}
				}
				if ($sMenuLink == Phpfox_Url::instance()->current()) {
					$bForceActive = true;
					$bHasMenu = true;
				}
			}
			elseif ($bIsView)
			{
				if ($sView == $sMenuLink && $bHasMenu === false)
				{
					$bHasMenu = true;
				}

				if (!empty($sView) && $sView == $sMenuLink)
				{
					$this->setTitle(preg_replace('/<span(.*)>(.*)<\/span>/i', '', $sMenuName));
					$this->setBreadCrumb(preg_replace('/<span(.*)>(.*)<\/span>/i', '', $sMenuName), Phpfox_Url::instance()->makeUrl($sSection, (empty($sMenuLink) ? array() : array('view' => $sMenuLink))), true);
				}
			}
			else
			{
				$sFullUrl = Phpfox_Url::instance()->getFullUrl();
				$sTimeRequest = Phpfox_Request::instance()->get('_');
				if ($sTimeRequest){
					$sFullUrl = str_replace('&_=' . $sTimeRequest, '', $sFullUrl);
				}
				if ((empty($sView) && str_replace('/', '.', Phpfox_Url::instance()->getUrl()) == $sMenuLink)
						|| (!empty($sView) && str_replace('/', '.', Phpfox_Url::instance()->getUrl()) . '.view_' . $sView == $sMenuLink)
						|| (!empty($sView) && Phpfox_Url::instance()->getUrl() . '.view_' . $sView . '.id_' . Phpfox_Request::instance()->getInt('id') == $sMenuLink)
						|| (!empty($sView) && preg_match('/\/view_' . $sView . '\//i', $sMenuLink))
						|| ($sFullUrl == $sMenuLink)
				)
				{
					$bHasMenu = true;
					$bForceActive = true;

//					$this->setTitle(preg_replace('/<span(.*)>(.*)<\/span>/i', '', $sMenuName));
//					$this->setBreadCrumb(preg_replace('/<span(.*)>(.*)<\/span>/i', '', $sMenuName), Phpfox_Url::instance()->makeUrl($sMenuLink), true);

					foreach ($aFilterMenuCache as $iSubKey => $aFilterMenuCacheRow)
					{
						if (isset($aFilterMenuCache[$iSubKey]['active']) && $aFilterMenuCache[$iSubKey]['active'] === true)
						{
							$aFilterMenuCache[$iSubKey]['active'] = false;
							break;
						}
					}
				}
			}

			$aFilterMenuCache[] = array(
					'name' => $sMenuName,
					'link' => (!$bIsView ? Phpfox_Url::instance()->makeUrl($sMenuLink) : Phpfox_Url::instance()->makeUrl($sSection, (empty($sMenuLink) ? array() : array('view' => $sMenuLink)))),
					'active' => ($bForceActive ? true : ($sView == $sMenuLink ? true : false)),
					'last' => (count($aFilterMenu) === $iFilterCount ? true : false)
			);
		}

		if (!$bHasMenu && isset($aFilterMenuCache[0]))
		{
			$aFilterMenuCache[0]['active'] = true;
		}

		$this->assign(array(
						'aFilterMenus' => $aFilterMenuCache,
				)
		);
	}

	public function setSubMenu($menu) {
		$this->_subMenu = $menu;
	}

	public function getSubMenu() {
		if (!$this->_subMenu) {
			return '';
		}

		$current = trim(Phpfox_Request::instance()->uri(), '/');
		if (is_string($this->_subMenu)) {
			$current = Phpfox_Url::instance()->makeUrl($current);
			$this->_subMenu = preg_replace('/href\=\"' . preg_quote($current, '/') . '\"/i', 'href="' . $current . '" class="active"', $this->_subMenu);

			return $this->_subMenu;
		}

		$html = '<div class="section_menu"><ul>';
		foreach ($this->_subMenu as $name => $url) {
			$active = '';
			$check = trim($url, '/');
			if ($check == $current) {
				$active = ' class="active"';
			}

			$html .= '<li><a href="' . Phpfox_Url::instance()->makeUrl($url) . '"' . $active . '>' . $name . '</a></li>';
		}
		$html .= '</ul></div>';

		return $html;
	}

	/**
	 * Gets the JavaScript code needed for section menus built with self::buildPageMenu()
	 *
	 * @see self::buildPageMenu()
	 * @return string Returns JS code
	 */
	public function getSectionMenuJavaScript()
	{
		if (!isset($this->_aSectionMenu['name']))
		{
			return '<script type="text/javascript">$Behavior.pageSectionMenuRequest = function() { }</script>';
		}

		if ($this->_aSectionMenu['bIsFullLink'])
		{
			return '<script type="text/javascript">$Behavior.pageSectionMenuRequest = function() { }</script>';
		}

		$sName = $this->_aSectionMenu['name'];
		$aMenu = $this->_aSectionMenu['menu'];
		$aLink = $this->_aSectionMenu['link'];

		$sReq = Phpfox_Request::instance()->get('req3');
		if (empty($sReq))
		{
			foreach ($aMenu as $sKey => $sValue)
			{
				$sReq = $sKey;
				break;
			}
		}

		$sNewReq = preg_replace('/[^a-z\s]/i', '', $sReq);
		if (!empty($sReq))
		{
			return '<script type="text/javascript">var bIsFirstRun = false; $Behavior.pageSectionMenuRequest = function() { if (!bIsFirstRun) { $Core.pageSectionMenuShow(\'#' . $sName . '_' . $sNewReq . '\'); if ($(\'#page_section_menu_form\').length > 0) { $(\'#page_section_menu_form\').val(\'' . $sName . '_' . $sNewReq . '\'); } bIsFirstRun = true; } }</script>';
		}
	}

	/**
	 * Builds a section menu for adding/editing items
	 *
	 * @param string $sName Name of the menu
	 * @param array $aMenu ARRAY of menus
	 * @param mixed $aLink ARRAY for custom view link, NULL to do nothing
	 */
	public function buildPageMenu($sName, $aMenu, $aLink = null, $bIsFullLink = false)
	{
		$this->_aSectionMenu = array(
				'name' => $sName,
				'menu' => $aMenu,
				'link' => $aLink,
				'bIsFullLink' => $bIsFullLink
		);

		$sPageCurrentUrl = Phpfox_Url::instance()->makeUrl('current');

		$this->assign(array(
						'sPageSectionMenuName' => $sName,
						'aPageSectionMenu' => $aMenu,
						'aPageExtraLink' => $aLink,
						'bPageIsFullLink' => $bIsFullLink,
						'sPageCurrentUrl' => $sPageCurrentUrl
				)
		);
	}

	/**
	 * This function controls if we should load `content` delayed. It is called from the template.cache library
	 */
	public function shouldLoadDelayed($sController)
	{
		$sController = Phpfox_Module::instance()->getFullControllerName();

		$bDelayed = false;

		// Special case
		if ($sController == 'feed.comment')
		{
			return true;
		}

		if (Phpfox::isAdminPanel())
		{
			return false;
		}

		$aControllers = Phpfox::getParam('core.controllers_to_load_delayed');

		if ( (is_array($aControllers) && !empty($aControllers)) &&
				(in_array($sController, $aControllers)) &&
				( (defined('PHPFOX_IS_PAGES_VIEW') && $sController == 'pages.view') || (!defined('PHPFOX_IS_PAGES_VIEW')))
		)
		{
			$bDelayed = true;
		}
		$aSearch = Phpfox_Request::instance()->get('search');
		if (!empty($aSearch))
		{
			$bDelayed = false;
		}

		return $bDelayed;
	}

	public function getCacheName($sName) {
		return $this->_getCachedName($this->getTemplateFile($sName));
	}

	/**
	 * Get a menu.
	 *
	 * @param string $sConnection Connection for the menu.
	 * @param int $iParent Parent ID# number for the menu.
	 * @return array ARRAY of menus.
	 */
	private function _getMenu($sConnection = null, $iParent = 0)
	{
		return Phpfox_Database::instance()->select('m.menu_id, m.parent_id, m.m_connection, m.var_name, m.disallow_access, mo.module_id AS module, m.url_value AS url, mo.is_active AS module_is_active, m.mobile_icon')
				->from(Phpfox::getT('menu'), 'm')
				->join(Phpfox::getT('module'), 'mo', 'mo.module_id = m.module_id AND mo.is_active = 1')
				->join(Phpfox::getT('product'), 'p', 'm.product_id = p.product_id AND p.is_active = 1')
				->where("m.parent_id = " . (int) $iParent . " AND m.m_connection = '" . Phpfox_Database::instance()->escape($sConnection) . "' AND m.is_active = 1")
				->group('m.menu_id', true)
				->order('m.ordering ASC')
				->execute('getRows');
	}

	/**
	 * Returns the content of a template that has already been echoed.
	 *
	 * @return string
	 */
	private function _returnLayout()
	{
		$sContent = ob_get_contents();

		ob_clean();

		return $sContent;
	}

	/**
	 * Gets a template file from cache. If the file does not exist we re-cache the template.
	 *
	 * @param string $sFile Full path of the template we are loading.
	 */
	private function _getFromCache($sFile, $sTemplate = null)
	{
		if (is_array($sFile)) {
			$sContent = $sFile[0];
			$sFile = $sTemplate;
		}

		if (!$this->_isCached($sFile))
		{
			$oTplCache = Phpfox::getLib('template.cache');
			if (!isset($sContent)) {
				$sContent = (file_exists($sFile) ? file_get_contents($sFile) : null);
			}
			$mData = $oTplCache->compile($this->_getCachedName($sFile), $sContent);
		}

		// No cache directory so we must
		if (defined('PHPFOX_INSTALLER_NO_TMP'))
		{
			eval(' ?>' . $mData . '<?php ');
			return;
		}

		(PHPFOX_DEBUG ? Phpfox_Debug::start('template') : false);
		$this->_requireFile($sFile);
		(PHPFOX_DEBUG ? Phpfox_Debug::end('template', array('name' => $sFile)) : false);
	}

	private function _requireFile($sFile)
	{
		if (defined('PHPFOX_IS_HOSTED_SCRIPT'))
		{
			$oCache = Phpfox::getLib('cache');
			$sId = $oCache->set(md5($this->_getCachedName($sFile)));
			eval('?>' . $oCache->get($sId) . '<?php ');
		}
		else
		{
			require($this->_getCachedName($sFile));
		}
	}

	/**
	 * Checks to see if a template has already been cached or not.
	 *
	 * @param string $sName Full path to the template file.
	 * @return bool TRUE is cached, FALSE is not cached.
	 */
	private function _isCached($sName)
	{
		if (defined('PHPFOX_NO_TEMPLATE_CACHE'))
		{
			return false;
		}

		if (!file_exists($this->_getCachedName($sName)))
		{
			return false;
		}

		if (file_exists($sName))
		{
			$iTime = filemtime($sName);

			// Check if a file has been updated recently, if it has make sure we return false to we recache it.
			if (($iTime + $this->_iCacheTime) > PHPFOX_TIME)
			{
				return false;
			}
		}

		return true;
	}

	/**
	 * Gets the full path of the cached template file
	 *
	 * @param string $sName Name of the template
	 * @return string Full path to cached template
	 */
	private function _getCachedName($sName)
	{
		if (!defined('PHPFOX_INSTALLER'))
		{
			if (!is_dir(PHPFOX_DIR_CACHE . 'template' . PHPFOX_DS))
			{
				mkdir(PHPFOX_DIR_CACHE . 'template' . PHPFOX_DS);
				chmod(PHPFOX_DIR_CACHE . 'template' . PHPFOX_DS, 0777);
			}
		}

		return (defined('PHPFOX_IS_HOSTED_SCRIPT') ? PHPFOX_IS_HOSTED_SCRIPT . Phpfox::getCleanVersion() . '' : '') . (defined('PHPFOX_TMP_DIR') ? PHPFOX_TMP_DIR : PHPFOX_DIR_CACHE) . ((defined('PHPFOX_TMP_DIR') || PHPFOX_SAFE_MODE) ? 'template_' : 'template' . PHPFOX_DS) . str_replace(array(PHPFOX_DIR_SITE_APPS, PHPFOX_DIR_THEME, PHPFOX_DIR_MODULE, PHPFOX_DS), array('', '', '', '_'), $sName) . (Phpfox::isAdminPanel() ? '_admincp' : '') . (PHPFOX_IS_AJAX ? '_ajax' : '') . '.php';
	}

    public function getDirectoryName($name)
    {
        return isset($this->_aDirectoryNames[$name])? $this->_aDirectoryNames[$name]: null;
    }

    private function _register($sType, $sFunction, $sImplementation)
	{
		$this->_aPlugins[$sType][$sFunction] = $sImplementation;
	}
}