Merge branch '5.3' into 5.4
authorTim Düsterhus <duesterhus@woltlab.com>
Tue, 3 Aug 2021 10:39:34 +0000 (12:39 +0200)
committerTim Düsterhus <duesterhus@woltlab.com>
Tue, 3 Aug 2021 10:39:34 +0000 (12:39 +0200)
1  2 
wcfsetup/install/files/lib/system/template/TemplateScriptingCompiler.class.php

index 73bd27a98220548965a1f71d77664a8f1c6cefda,3459f00eeacc22b166ad3595607c538d8caa5bab..07708d2fb8a62c3295456bfb58debb0c9dad7b5b
@@@ -10,2195 -8,1814 +10,2195 @@@ use wcf\util\StringUtil
  
  /**
   * Compiles template sources into valid PHP code.
 - * 
 - * @author    Marcel Werk
 - * @copyright 2001-2019 WoltLab GmbH
 - * @license   GNU Lesser General Public License <http://opensource.org/licenses/lgpl-license.php>
 - * @package   WoltLabSuite\Core\System\Template
 + *
 + * @author  Marcel Werk
 + * @copyright   2001-2019 WoltLab GmbH
 + * @license GNU Lesser General Public License <http://opensource.org/licenses/lgpl-license.php>
 + * @package WoltLabSuite\Core\System\Template
   */
 -class TemplateScriptingCompiler {
 -      /**
 -       * template engine object
 -       * @var TemplateEngine
 -       */
 -      protected $template;
 -      
 -      /**
 -       * PHP functions that can be used in the modifier syntax and are unknown
 -       * to PHP's function_exists function
 -       * @var string[]
 -       */
 -      protected $unknownPHPFunctions = ['isset', 'unset', 'empty'];
 -      
 -      /**
 -       * PHP functions that can not be used in the modifier syntax
 -       * @var string[]
 -       */
 -      protected $disabledPHPFunctions = [
 -              'system', 'exec', 'passthru', 'shell_exec', // command line execution
 -              'include', 'require', 'include_once', 'require_once', // includes
 -              'eval', 'virtual', 'call_user_func_array', 'call_user_func', 'assert' // code execution
 -      ];
 -      
 -      /**
 -       * PHP functions and modifiers that can be used in enterprise mode
 -       * @var string[]
 -       */
 -      protected $enterpriseFunctions = [
 -              'addslashes',
 -              'array_diff',
 -              'array_fill',
 -              'array_keys',
 -              'array_pop',
 -              'array_slice',
 -              'array_values',
 -              'base64_decode',
 -              'base64_encode',
 -              'ceil',
 -              'concat',
 -              'constant',
 -              'count',
 -              'currency',
 -              'current',
 -              'date',
 -              'defined',
 -              'doubleval',
 -              'empty',
 -              'end',
 -              'explode',
 -              'file_exists',
 -              'filesize',
 -              'floatval',
 -              'floor',
 -              'function_exists',
 -              'get_class',
 -              'gmdate',
 -              'hash',
 -              'htmlspecialchars',
 -              'html_entity_decode',
 -              'http_build_query',
 -              'implode',
 -              'in_array',
 -              'is_array',
 -              'is_null',
 -              'is_numeric',
 -              'is_object',
 -              'iterator_count',
 -              'intval',
 -              'is_subclass_of',
 -              'isset',
 -              'json_encode',
 -              'key',
 -              'lcfirst',
 -              'ltrim',
 -              'max',
 -              'mb_strpos',
 -              'mb_strlen',
 -              'mb_strpos',
 -              'mb_strtolower',
 -              'mb_strtoupper',
 -              'mb_substr',
 -              'md5',
 -              'method_exists',
 -              'microtime',
 -              'min',
 -              'nl2br',
 -              'number_format',
 -              'parse_url',
 -              'preg_match',
 -              'preg_replace',
 -              'print_r',
 -              'rawurlencode',
 -              'reset',
 -              'round',
 -              'sha1',
 -              'spl_object_hash',
 -              'sprintf',
 -              'strip_tags',
 -              'strlen',
 -              'strpos',
 -              'strtolower',
 -              'strtotime',
 -              'strtoupper',
 -              'str_pad',
 -              'str_repeat',
 -              'str_replace',
 -              'str_ireplace',
 -              'substr',
 -              'trim',
 -              'ucfirst',
 -              'uniqid',
 -              'urlencode',
 -              'version_compare',
 -              'wcfDebug',
 -              'wordwrap'
 -      ];
 -      
 -      /**
 -       * pattern to match variable operators like -> or .
 -       * @var string
 -       */
 -      protected $variableOperatorPattern;
 -      
 -      /**
 -       * pattern to match condition operators like == or <
 -       * @var string
 -       */
 -      protected $conditionOperatorPattern;
 -      
 -      /**
 -       * negative lookbehind for a backslash
 -       * @var string
 -       */
 -      protected $escapedPattern;
 -      
 -      /**
 -       * pattern to match valid variable names
 -       * @var string
 -       */
 -      protected $validVarnamePattern;
 -      
 -      /**
 -       * pattern to match constants like CONSTANT or __CONSTANT
 -       * @var string
 -       */
 -      protected $constantPattern;
 -      
 -      /**
 -       * pattern to match double quoted strings like "blah" or "quote: \"blah\""
 -       * @var string
 -       */
 -      protected $doubleQuotePattern;
 -      
 -      /**
 -       * pattern to match single quoted strings like 'blah' or 'don\'t'
 -       * @var string
 -       */
 -      protected $singleQuotePattern;
 -      
 -      /**
 -       * pattern to match single or double quoted strings
 -       * @var string
 -       */
 -      protected $quotePattern;
 -      
 -      /**
 -       * pattern to match numbers, true, false and null
 -       * @var string
 -       */
 -      protected $numericPattern;
 -      
 -      /**
 -       * pattern to match simple variables like $foo
 -       * @var string
 -       */
 -      protected $simpleVarPattern;
 -      
 -      /**
 -       * pattern to match outputs like @$foo or #CONST
 -       * @var string
 -       */
 -      protected $outputPattern;
 -      
 -      /**
 -       * identifier of currently compiled template
 -       * @var string
 -       */
 -      protected $currentIdentifier;
 -      
 -      /**
 -       * current line number during template compilation
 -       * @var string
 -       */
 -      protected $currentLineNo;
 -      
 -      /**
 -       * list of automatically loaded tenplate plugins
 -       * @var string[]
 -       */
 -      protected $autoloadPlugins = [];
 -      
 -      /**
 -       * stack with template tags data
 -       * @var array
 -       */
 -      protected $tagStack = [];
 -      
 -      /**
 -       * list of loaded compiler plugin objects
 -       * @var ICompilerTemplatePlugin[]
 -       */
 -      protected $compilerPlugins = [];
 -      
 -      /**
 -       * stack used to compile the capture tag
 -       * @var array
 -       */
 -      protected $captureStack = [];
 -      
 -      /**
 -       * left delimiter of template syntax
 -       * @var string
 -       */
 -      protected $leftDelimiter = '{';
 -      
 -      /**
 -       * right delimiter of template syntax
 -       * @var string
 -       */
 -      protected $rightDelimiter = '}';
 -      
 -      /**
 -       * left delimiter of template syntax used in regular expressions
 -       * @var string
 -       */
 -      protected $ldq;
 -      
 -      /**
 -       * right delimiter of template syntax used in regular expressions
 -       * @var string
 -       */
 -      protected $rdq;
 -      
 -      /**
 -       * list of static includes per template
 -       * @var string[][]
 -       */
 -      protected $staticIncludes = [];
 -      
 -      /**
 -       * data of all currently active `foreach` loops; entry data:
 -       * 
 -       *      hash => unique foreach loop hash
 -       *      itemVar => template code for the `item` variable
 -       *      keyVar => (optional) template code for the `key` variable
 -       * 
 -       * @var string[][]
 -       */
 -      protected $foreachLoops = [];
 -      
 -      /**
 -       * Creates a new TemplateScriptingCompiler object.
 -       * 
 -       * @param       TemplateEngine          $template
 -       */
 -      public function __construct(TemplateEngine $template) {
 -              $this->template = $template;
 -              
 -              // quote left and right delimiter for use in regular expressions
 -              $this->ldq = preg_quote($this->leftDelimiter, '~').'(?=\S)';
 -              $this->rdq = '(?<=\S)'.preg_quote($this->rightDelimiter, '~');
 -              
 -              // build regular expressions
 -              $this->buildPattern();
 -      }
 -      
 -      /**
 -       * Compiles the source of a template.
 -       * 
 -       * @param       string          $identifier
 -       * @param       string          $sourceContent
 -       * @param       array           $metaData
 -       * @param       boolean         $isolated
 -       * @return      array|boolean
 -       * @throws      SystemException
 -       */
 -      public function compileString($identifier, $sourceContent, array $metaData = [], $isolated = false) {
 -              $previousData = [];
 -              if ($isolated) {
 -                      $previousData = [
 -                              'autoloadPlugins' => $this->autoloadPlugins,
 -                              'currentIdentifier' => $this->currentIdentifier,
 -                              'currentLineNo' => $this->currentLineNo,
 -                              'tagStack' => $this->tagStack
 -                      ];
 -              }
 -              else {
 -                      $this->staticIncludes = [];
 -              }
 -              
 -              // reset vars
 -              $this->autoloadPlugins = $this->tagStack = [];
 -              $this->currentIdentifier = $identifier;
 -              $this->currentLineNo = 1;
 -              
 -              // apply prefilters
 -              $sourceContent = $this->applyPrefilters($identifier, $sourceContent);
 -              
 -              // replace all {literal} Tags with unique hash values
 -              $sourceContent = $this->replaceLiterals($sourceContent);
 -              
 -              // handle <?php tags
 -              $sourceContent = $this->replacePHPTags($sourceContent);
 -              
 -              // remove comments
 -              $sourceContent = $this->removeComments($sourceContent);
 -              
 -              // match all template tags
 -              $matches = [];
 -              preg_match_all("~".$this->ldq."(.*?)".$this->rdq."~s", $sourceContent, $matches);
 -              $templateTags = $matches[1];
 -              
 -              // Split content by template tags to obtain non-template content
 -              $textBlocks = preg_split("~".$this->ldq.".*?".$this->rdq."~s", $sourceContent);
 -              
 -              // compile the template tags into php-code
 -              $compiledTags = [];
 -              for ($i = 0, $j = count($templateTags); $i < $j; $i++) {
 -                      $this->currentLineNo += mb_substr_count($textBlocks[$i], "\n");
 -                      
 -                      if ($templateTags[$i] === '') {
 -                              // avoid empty JavaScript object literals being recognized
 -                              // as template scripting tags
 -                              $compiledTags[] = '{}';
 -                      }
 -                      else {
 -                              $compiledTags[] = $this->compileTag($templateTags[$i], $identifier, $metaData);
 -                      }
 -                      
 -                      $this->currentLineNo += mb_substr_count($templateTags[$i], "\n");
 -              }
 -              
 -              // throw error messages for unclosed tags
 -              if (count($this->tagStack) > 0) {
 -                      foreach ($this->tagStack as $tagStack) {
 -                              throw new SystemException(static::formatSyntaxError('unclosed tag {'.$tagStack[0].'}', $this->currentIdentifier, $tagStack[1]));
 -                      }
 -                      return false;
 -              }
 -              
 -              $compiledContent = '';
 -              // Interleave the compiled contents and text blocks to get the final result.
 -              for ($i = 0, $j = count($compiledTags); $i < $j; $i++) {
 -                      if ($compiledTags[$i] == '') {
 -                              // tag result empty, remove first newline from following text block
 -                              $textBlocks[$i + 1] = preg_replace('%^(\r\n|\r|\n)%', '', $textBlocks[$i + 1]);
 -                      }
 -                      $compiledContent .= $textBlocks[$i].$compiledTags[$i];
 -              }
 -              $compiledContent .= $textBlocks[$i];
 -              $compiledContent = rtrim($compiledContent);
 -              
 -              // reinsert {literal} Tags
 -              $compiledContent = $this->reinsertLiterals($compiledContent);
 -              
 -              // include Plugins
 -              $compiledAutoloadPlugins = '';
 -              if (count($this->autoloadPlugins) > 0) {
 -                      $compiledAutoloadPlugins = "<?php\n";
 -                      foreach ($this->autoloadPlugins as $className) {
 -                              $compiledAutoloadPlugins .= "if (!isset(\$this->pluginObjects['$className'])) {\n";
 -                              $compiledAutoloadPlugins .= "\$this->pluginObjects['$className'] = new $className;\n";
 -                              $compiledAutoloadPlugins .= "}\n";
 -                      }
 -                      $compiledAutoloadPlugins .= "?>";
 -              }
 -              
 -              // restore data
 -              if ($isolated) {
 -                      $this->autoloadPlugins = $previousData['autoloadPlugins'];
 -                      $this->currentIdentifier = $previousData['currentIdentifier'];
 -                      $this->currentLineNo = $previousData['currentLineNo'];
 -                      $this->tagStack = $previousData['tagStack'];
 -              }
 -              
 -              return [
 -                      'meta' => [
 -                              'include' => $this->staticIncludes
 -                      ],
 -                      'template' => $compiledAutoloadPlugins.$compiledContent
 -              ];
 -      }
 -      
 -      /**
 -       * Compiles a template tag.
 -       * 
 -       * @param       string          $tag
 -       * @param       string          $identifier
 -       * @param       array           $metaData
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileTag($tag, $identifier, array &$metaData) {
 -              if (preg_match('~^'.$this->outputPattern.'~s', $tag)) {
 -                      // variable output
 -                      return $this->compileOutputTag($tag);
 -              }
 -              
 -              $match = [];
 -              // replace 'else if' with 'elseif'
 -              $tag = preg_replace('~^else\s+if(?=\s)~i', 'elseif', $tag);
 -              
 -              if (preg_match('~^(/?\w+)~', $tag, $match)) {
 -                      // build in function or plugin
 -                      $tagCommand = $match[1];
 -                      $tagArgs = mb_substr($tag, mb_strlen($tagCommand));
 -                      
 -                      switch ($tagCommand) {
 -                              case 'if':
 -                                      $this->pushTag('if');
 -                                      return $this->compileIfTag($tagArgs);
 -                                      
 -                              case 'elseif':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'if' && $openTag != 'elseif') {
 -                                              throw new SystemException(static::formatSyntaxError('unxepected {elseif}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      else if ($openTag == 'if') {
 -                                              $this->pushTag('elseif');
 -                                      }
 -                                      return $this->compileIfTag($tagArgs, true);
 -                                      
 -                              case 'else':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'if' && $openTag != 'elseif') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {else}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->pushTag('else');
 -                                      return '<?php } else { ?>';
 -                                      
 -                              case '/if':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'if' && $openTag != 'elseif' && $openTag != 'else') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {/if}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->popTag('if');
 -                                      return '<?php } ?>';
 -                                      
 -                              case 'include':
 -                                      return $this->compileIncludeTag($tagArgs, $identifier, $metaData);
 -                                      
 -                              case 'foreach':
 -                                      $this->pushTag('foreach');
 -                                      return $this->compileForeachTag($tagArgs);
 -                                      
 -                              case 'foreachelse':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'foreach') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {foreachelse}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->pushTag('foreachelse');
 -                                      return '<?php } } else { { ?>';
 -                                      
 -                              case '/foreach':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'foreach' && $openTag != 'foreachelse') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {/foreach}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->popTag('foreach');
 -                                      return $this->compileForeachEndTag();
 -                                      
 -                              case 'section':
 -                                      $this->pushTag('section');
 -                                      return $this->compileSectionTag($tagArgs);
 -                                      
 -                              case 'sectionelse':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'section') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {sectionelse}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->pushTag('sectionelse');
 -                                      return '<?php } } else { { ?>';
 -                                      
 -                              case '/section':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'section' && $openTag != 'sectionelse') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {/section}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->popTag('section');
 -                                      return "<?php } } ?>";
 -                                      
 -                              case 'capture':
 -                                      $this->pushTag('capture');
 -                                      return $this->compileCaptureTag(true, $tagArgs);
 -                                      
 -                              case '/capture':
 -                                      list($openTag) = end($this->tagStack);
 -                                      if ($openTag != 'capture') {
 -                                              throw new SystemException(static::formatSyntaxError('unexpected {/capture}', $this->currentIdentifier, $this->currentLineNo));
 -                                      }
 -                                      $this->popTag('capture');
 -                                      return $this->compileCaptureTag(false);
 -                                      
 -                              case 'ldelim':
 -                                      return $this->leftDelimiter;
 -                                      
 -                              case 'rdelim':
 -                                      return $this->rightDelimiter;
 -                                      
 -                              default:
 -                                      // 1) compiler functions first
 -                                      if ($phpCode = $this->compileCompilerPlugin($tagCommand, $tagArgs)) {
 -                                              return $phpCode;
 -                                      }
 -                                      // 2) block functions
 -                                      if ($phpCode = $this->compileBlockPlugin($tagCommand, $tagArgs)) {
 -                                              return $phpCode;
 -                                      }
 -                                      // 3) functions
 -                                      if ($phpCode = $this->compileFunctionPlugin($tagCommand, $tagArgs)) {
 -                                              return $phpCode;
 -                                      }
 -                      }
 -              }
 -              
 -              throw new SystemException(static::formatSyntaxError('unknown tag {'.$tag.'}', $this->currentIdentifier, $this->currentLineNo));
 -      }
 -      
 -      /**
 -       * Compiles a function plugin and returns the output of the plugin or false
 -       * if the plugin doesn't exist.
 -       * 
 -       * @param       string          $tagCommand
 -       * @param       string          $tagArgs
 -       * @return      mixed
 -       */
 -      protected function compileFunctionPlugin($tagCommand, $tagArgs) {
 -              $className = $this->template->getPluginClassName('function', $tagCommand);
 -              if (!class_exists($className)) {
 -                      return false;
 -              }
 -              $this->autoloadPlugins[$className] = $className;
 -              
 -              $tagArgs = static::makeArgString($this->parseTagArgs($tagArgs, $tagCommand));
 -              
 -              return "<?=\$this->pluginObjects['".$className."']->execute([".$tagArgs."], \$this);?>";
 -      }
 -      
 -      /**
 -       * Compiles a block plugin and returns the output of the plugin or false
 -       * if the plugin doesn't exist.
 -       * 
 -       * @param       string          $tagCommand
 -       * @param       string          $tagArgs
 -       * @return      mixed
 -       * @throws      SystemException
 -       */
 -      protected function compileBlockPlugin($tagCommand, $tagArgs) {
 -              // check whether this is the start ({block}) or the
 -              // end tag ({/block})
 -              if (substr($tagCommand, 0, 1) == '/') {
 -                      $tagCommand = substr($tagCommand, 1);
 -                      $startTag = false;
 -              }
 -              else {
 -                      $startTag = true;
 -              }
 -              
 -              $className = $this->template->getPluginClassName('block', $tagCommand);
 -              if (!class_exists($className)) {
 -                      return false;
 -              }
 -              $this->autoloadPlugins[$className] = $className;
 -              
 -              if ($startTag) {
 -                      $this->pushTag($tagCommand);
 -                      
 -                      $tagArgs = static::makeArgString($this->parseTagArgs($tagArgs, $tagCommand));
 -                      
 -                      $phpCode = "<?php \$this->tagStack[] = ['".$tagCommand."', [".$tagArgs."]];\n";
 -                      $phpCode .= "\$this->pluginObjects['".$className."']->init(\$this->tagStack[count(\$this->tagStack) - 1][1], \$this);\n";
 -                      $phpCode .= "while (\$this->pluginObjects['".$className."']->next(\$this)) { ob_start(); ?>";
 -              }
 -              else {
 -                      list($openTag) = end($this->tagStack);
 -                      if ($openTag != $tagCommand) {
 -                              throw new SystemException(static::formatSyntaxError('unexpected {/'.$tagCommand.'}', $this->currentIdentifier, $this->currentLineNo));
 -                      }
 -                      $this->popTag($tagCommand);
 -                      $phpCode = "<?php echo \$this->pluginObjects['".$className."']->execute(\$this->tagStack[count(\$this->tagStack) - 1][1], ob_get_clean(), \$this); }\n";
 -                      $phpCode .= "array_pop(\$this->tagStack);?>";
 -              }
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Compiles a compiler function/block and returns the output of the plugin
 -       * or false if the plugin doesn't exist.
 -       * 
 -       * @param       string          $tagCommand
 -       * @param       string          $tagArgs
 -       * @return      mixed
 -       * @throws      SystemException
 -       */
 -      protected function compileCompilerPlugin($tagCommand, $tagArgs) {
 -              // check whether this is the start ({block}) or the
 -              // end tag ({/block})
 -              if (substr($tagCommand, 0, 1) == '/') {
 -                      $tagCommand = substr($tagCommand, 1);
 -                      $startTag = false;
 -              }
 -              else {
 -                      $startTag = true;
 -              }
 -              
 -              $className = $this->template->getPluginClassName('compiler', $tagCommand);
 -              // if necessary load plugin from plugin-dir
 -              if (!isset($this->compilerPlugins[$className])) {
 -                      if (!class_exists($className)) {
 -                              return false;
 -                      }
 -                      
 -                      $this->compilerPlugins[$className] = new $className();
 -                      
 -                      if (!($this->compilerPlugins[$className] instanceof ICompilerTemplatePlugin)) {
 -                              throw new SystemException(static::formatSyntaxError("Compiler plugin '".$tagCommand."' does not implement the interface 'ICompilerTemplatePlugin'", $this->currentIdentifier));
 -                      }
 -              }
 -              
 -              // execute plugin
 -              if ($startTag) {
 -                      $tagArgs = $this->parseTagArgs($tagArgs, $tagCommand);
 -                      $phpCode = $this->compilerPlugins[$className]->executeStart($tagArgs, $this);
 -              }
 -              else {
 -                      $phpCode = $this->compilerPlugins[$className]->executeEnd($this);
 -              }
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Compiles a capture tag and returns the compiled PHP code.
 -       * 
 -       * @param       boolean         $startTag
 -       * @param       string          $captureTag
 -       * @return      string
 -       */
 -      protected function compileCaptureTag($startTag, $captureTag = null) {
 -              if ($startTag) {
 -                      $append = false;
 -                      $args = $this->parseTagArgs($captureTag, 'capture');
 -                      
 -                      if (!isset($args['name'])) {
 -                              $args['name'] = "'default'";
 -                      }
 -                      
 -                      if (!isset($args['assign'])) {
 -                              if (isset($args['append'])) {
 -                                      $args['assign'] = $args['append'];
 -                                      $append = true;
 -                              }
 -                              else {
 -                                      $args['assign'] = '';
 -                              }
 -                      }
 -                      
 -                      $this->captureStack[] = ['name' => $args['name'], 'variable' => $args['assign'], 'append' => $append];
 -                      return '<?php ob_start(); ?>';
 -              }
 -              else {
 -                      $capture = array_pop($this->captureStack);
 -                      $phpCode = "<?php\n";
 -                      $phpCode .= "\$this->v['tpl']['capture'][".$capture['name']."] = ob_get_clean();\n";
 -                      if (!empty($capture['variable'])) $phpCode .= "\$this->".($capture['append'] ? 'append' : 'assign')."(".$capture['variable'].", \$this->v['tpl']['capture'][".$capture['name']."]);\n";
 -                      $phpCode .= "?>";
 -                      return $phpCode;
 -              }
 -      }
 -      
 -      /**
 -       * Compiles a section tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $sectionTag
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileSectionTag($sectionTag) {
 -              $args = $this->parseTagArgs($sectionTag, 'section');
 -              
 -              // check arguments
 -              if (!isset($args['loop'])) {
 -                      throw new SystemException(static::formatSyntaxError("missing 'loop' attribute in section tag", $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              if (!isset($args['name'])) {
 -                      throw new SystemException(static::formatSyntaxError("missing 'name' attribute in section tag", $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              if (!isset($args['show'])) {
 -                      $args['show'] = true;
 -              }
 -              
 -              $sectionProp = "\$this->v['tpl']['section'][".$args['name']."]";
 -              
 -              $phpCode = "<?php\n";
 -              $phpCode .= "if (".$args['loop'].") {\n";
 -              $phpCode .= $sectionProp." = [];\n";
 -              $phpCode .= $sectionProp."['loop'] = (is_array(".$args['loop'].") ? count(".$args['loop'].") : max(0, (int)".$args['loop']."));\n";
 -              $phpCode .= $sectionProp."['show'] = ".$args['show'].";\n";
 -              if (!isset($args['step'])) {
 -                      $phpCode .= $sectionProp."['step'] = 1;\n";
 -              }
 -              else {
 -                      $phpCode .= $sectionProp."['step'] = ".$args['step'].";\n";
 -              }
 -              if (!isset($args['max'])) {
 -                      $phpCode .= $sectionProp."['max'] = ".$sectionProp."['loop'];\n";
 -              }
 -              else {
 -                      $phpCode .= $sectionProp."['max'] = (".$args['max']." < 0 ? ".$sectionProp."['loop'] : ".$args['max'].");\n";
 -              }
 -              if (!isset($args['start'])) {
 -                      $phpCode .= $sectionProp."['start'] = (".$sectionProp."['step'] > 0 ? 0 : ".$sectionProp."['loop'] - 1);\n";
 -              }
 -              else {
 -                      $phpCode .= $sectionProp."['start'] = ".$args['start'].";\n";
 -                      $phpCode .= "if (".$sectionProp."['start'] < 0) {\n";
 -                      $phpCode .= $sectionProp."['start'] = max(".$sectionProp."['step'] > 0 ? 0 : -1, ".$sectionProp."['loop'] + ".$sectionProp."['start']);\n}\n";
 -                      $phpCode .= "else {\n";
 -                      $phpCode .= $sectionProp."['start'] = min(".$sectionProp."['start'], ".$sectionProp."['step'] > 0 ? ".$sectionProp."['loop'] : ".$sectionProp."['loop'] - 1);\n}\n";
 -              }
 -              
 -              if (!isset($args['start']) && !isset($args['step']) && !isset($args['max'])) {
 -                      $phpCode .= $sectionProp."['total'] = ".$sectionProp."['loop'];\n";
 -              } else {
 -                      $phpCode .= $sectionProp."['total'] = min(ceil((".$sectionProp."['step'] > 0 ? ".$sectionProp."['loop'] - ".$sectionProp."['start'] : ".$sectionProp."['start'] + 1) / abs(".$sectionProp."['step'])), ".$sectionProp."['max']);\n";
 -              }
 -              $phpCode .= "if (".$sectionProp."['total'] == 0) ".$sectionProp."['show'] = false;\n";
 -              $phpCode .= "} else {\n";
 -              $phpCode .= "".$sectionProp."['total'] = 0;\n";
 -              $phpCode .= "".$sectionProp."['show'] = false;}\n";
 -              
 -              $phpCode .= "if (".$sectionProp."['show']) {\n";
 -              $phpCode .= "for (".$sectionProp."['index'] = ".$sectionProp."['start'], ".$sectionProp."['rowNumber'] = 1;\n";
 -              $phpCode .= $sectionProp."['rowNumber'] <= ".$sectionProp."['total'];\n";
 -              $phpCode .= $sectionProp."['index'] += ".$sectionProp."['step'], ".$sectionProp."['rowNumber']++) {\n";
 -              $phpCode .= "\$this->v[".$args['name']."] = ".$sectionProp."['index'];\n";
 -              $phpCode .= $sectionProp."['previousIndex'] = ".$sectionProp."['index'] - ".$sectionProp."['step'];\n";
 -              $phpCode .= $sectionProp."['nextIndex'] = ".$sectionProp."['index'] + ".$sectionProp."['step'];\n";
 -              $phpCode .= $sectionProp."['first'] = (".$sectionProp."['rowNumber'] == 1);\n";
 -              $phpCode .= $sectionProp."['last'] = (".$sectionProp."['rowNumber'] == ".$sectionProp."['total']);\n";
 -              $phpCode .= "?>";
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Compiles a foreach tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $foreachTag
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileForeachTag($foreachTag) {
 -              $args = $this->parseTagArgs($foreachTag, 'foreach');
 -              
 -              // check arguments
 -              if (!isset($args['from'])) {
 -                      throw new SystemException(static::formatSyntaxError("missing 'from' attribute in foreach tag", $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              if (!isset($args['item'])) {
 -                      throw new SystemException(static::formatSyntaxError("missing 'item' attribute in foreach tag", $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              
 -              $foreachProp = '';
 -              if (isset($args['name'])) {
 -                      $foreachProp = "\$this->v['tpl']['foreach'][".$args['name']."]";
 -              }
 -              
 -              $hash = StringUtil::getRandomID();
 -              $foreachHash = "\$_foreach_" . $hash;
 -              $foreachData = ['hash' => $hash];
 -              
 -              $phpCode = "<?php\n";
 -              $phpCode .= $foreachHash." = ".$args['from'].";\n";
 -              
 -              $itemVar = mb_substr($args['item'], 0, 1) != '$' ? "\$this->v[".$args['item']."]" : $args['item'];
 -              $foreachData['itemVar'] = $itemVar;
 -              
 -              $phpCode .= "\$this->foreachVars['{$hash}'] = [];\n";
 -              $phpCode .= "\$this->foreachVars['{$hash}']['item'] = {$itemVar} ?? null;\n";
 -
 -              if (empty($foreachProp)) {
 -                      $phpCode .= "if ((is_countable(".$foreachHash.") && count(".$foreachHash.") > 0) || (!is_countable(".$foreachHash.") && ".$foreachHash.")) {\n";
 -              }
 -              else {
 -                      $phpCode .= $foreachHash."_cnt = (".$foreachHash." !== null ? 1 : 0);\n";
 -                      $phpCode .= "if (is_countable(".$foreachHash.")) {\n";
 -                      $phpCode .= $foreachHash."_cnt = count(".$foreachHash.");\n";
 -                      $phpCode .= "}\n";
 -                      $phpCode .= $foreachProp."['total'] = ".$foreachHash."_cnt;\n";
 -                      $phpCode .= $foreachProp."['show'] = (".$foreachProp."['total'] > 0 ? true : false);\n";
 -                      $phpCode .= $foreachProp."['iteration'] = 0;\n";
 -                      $phpCode .= "if (".$foreachHash."_cnt > 0) {\n";
 -              }
 -              
 -              if (isset($args['key'])) {
 -                      $keyVar = mb_substr($args['key'], 0, 1) != '$' ? "\$this->v[".$args['key']."]" : $args['key'];
 -                      $foreachData['keyVar'] = $keyVar;
 -                      
 -                      $phpCode .= "\$this->foreachVars['{$hash}']['key'] = {$keyVar} ?? null;\n";
 -                      
 -                      $phpCode .= "foreach (".$foreachHash." as {$keyVar} => {$itemVar}) {\n";
 -              }
 -              else {
 -                      $phpCode .= "foreach (".$foreachHash." as {$itemVar}) {\n";
 -              }
 -              
 -              if (!empty($foreachProp)) {
 -                      $phpCode .= $foreachProp."['first'] = (".$foreachProp."['iteration'] == 0 ? true : false);\n";
 -                      $phpCode .= $foreachProp."['last'] = ((".$foreachProp."['iteration'] == ".$foreachProp."['total'] - 1) ? true : false);\n";
 -                      $phpCode .= $foreachProp."['iteration']++;\n";
 -              }
 -              
 -              $phpCode .= "?>";
 -              
 -              $this->foreachLoops[] = $foreachData;
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Compiles a `/foreach` tag and returns the compiled PHP code.
 -       * 
 -       * @return      string
 -       * @since       5.2
 -       */
 -      protected function compileForeachEndTag() {
 -              $foreachData = array_pop($this->foreachLoops);
 -              
 -              // unset `item` and `key` variables and restore their sandboxed values
 -              $phpCode = "<?php }\n";
 -              $phpCode .= "unset({$foreachData['itemVar']});";
 -              $phpCode .= "{$foreachData['itemVar']} = \$this->foreachVars['{$foreachData['hash']}']['item'];\n";
 -              
 -              if (isset($foreachData['keyVar'])) {
 -                      $phpCode .= "unset({$foreachData['keyVar']});";
 -                      $phpCode .= "if (array_key_exists('key', \$this->foreachVars['{$foreachData['hash']}'])) {\n";
 -                      $phpCode .= "{$foreachData['keyVar']} = \$this->foreachVars['{$foreachData['hash']}']['key'];\n";
 -                      $phpCode .= "}\n";
 -              }
 -              
 -              $phpCode .= "unset(\$this->foreachVars['{$foreachData['hash']}']);\n";
 -              $phpCode .= " } ?>";
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Compiles an include tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $includeTag
 -       * @param       string          $identifier
 -       * @param       array           $metaData
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileIncludeTag($includeTag, $identifier, array $metaData) {
 -              $args = $this->parseTagArgs($includeTag, 'include');
 -              $append = false;
 -              
 -              // check arguments
 -              if (!isset($args['file'])) {
 -                      throw new SystemException(static::formatSyntaxError("missing 'file' attribute in include tag", $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              
 -              // get filename
 -              $file = $args['file'];
 -              unset($args['file']);
 -              
 -              // special parameters
 -              $assignVar = false;
 -              if (isset($args['assign'])) {
 -                      $assignVar = $args['assign'];
 -                      unset($args['assign']);
 -              }
 -              
 -              if (isset($args['append'])) {
 -                      $assignVar = $args['append'];
 -                      $append = true;
 -                      unset($args['append']);
 -              }
 -              
 -              $once = false;
 -              if (isset($args['once'])) {
 -                      $once = $args['once'];
 -                      unset($args['once']);
 -              }
 -              
 -              $application = "'wcf'";
 -              if (isset($args['application'])) {
 -                      $application = $args['application'];
 -                      unset($args['application']);
 -              }
 -              
 -              if (preg_match('~^(\'|\")(.*)\1$~', $application, $matches)) {
 -                      $application = $matches[2];
 -              }
 -              
 -              $sandbox = false;
 -              if (isset($args['sandbox'])) {
 -                      $sandbox = $args['sandbox'];
 -                      unset($args['sandbox']);
 -              }
 -              
 -              $sandbox = ($sandbox === 'true' || $sandbox === true || $sandbox == 1);
 -              
 -              $staticInclude = true;
 -              if ($sandbox || $assignVar !== false || $once !== false || strpos($application, '$') !== false || strpos($file, '$') !== false) {
 -                      $staticInclude = false;
 -              }
 -              
 -              $templateName = substr($file, 1, -1);
 -              
 -              // check for static includes
 -              if ($staticInclude) {
 -                      $phpCode = '';
 -                      if (!isset($this->staticIncludes[$application])) {
 -                              $this->staticIncludes[$application] = [];
 -                      }
 -                      
 -                      if (!in_array($templateName, $this->staticIncludes[$application])) {
 -                              $this->staticIncludes[$application][] = $templateName;
 -                      }
 -                      
 -                      // pass remaining tag args as variables
 -                      if (!empty($args)) {
 -                              foreach ($args as $variable => $value) {
 -                                      if (substr($value, 0, 1) == "'") {
 -                                              // string values
 -                                              $phpCode .= "\$this->v['".$variable."'] = ".$value.";\n";
 -                                      }
 -                                      else {
 -                                              if (preg_match('~^\$this->v\[\'(.*)\'\]$~U', $value, $matches)) {
 -                                                      // value is a variable itself
 -                                                      $phpCode .= "\$this->v['".$variable."'] = ".$value.";\n";
 -                                              }
 -                                              else {
 -                                                      // value is boolean, an integer or anything else
 -                                                      $phpCode .= "\$this->v['".$variable."'] = ".$value.";\n";
 -                                              }
 -                                      }
 -                              }
 -                      }
 -                      if (!empty($phpCode)) $phpCode = "<?php\n".$phpCode."\n?>";
 -                      
 -                      $sourceFilename = $this->template->getSourceFilename($templateName, $application);
 -                      
 -                      $data = $this->compileString($templateName, file_get_contents($sourceFilename), [
 -                              'application' => $application,
 -                              'data' => null,
 -                              'filename' => ''
 -                      ], true);
 -                      
 -                      return $phpCode . $data['template'];
 -              }
 -              
 -              // make argument string
 -              $argString = static::makeArgString($args);
 -              
 -              // build phpCode
 -              $phpCode = "<?php\n";
 -              if ($once) $phpCode .= "if (!isset(\$this->v['tpl']['includedTemplates'][".$file."])) {\n";
 -              $hash = StringUtil::getRandomID();
 -              $phpCode .= "\$outerTemplateName".$hash." = \$this->v['tpl']['template'];\n";
 -              
 -              if ($assignVar !== false) {
 -                      $phpCode .= "ob_start();\n";
 -              }
 -              
 -              if (strpos($application, '$') === false) {
 -                      $application = "'" . $application . "'";
 -              }
 -              $phpCode .= '$this->includeTemplate('.$file.', '.$application.', ['.$argString.'], '.($sandbox ? 1 : 0).');'."\n";
 -              
 -              if ($assignVar !== false) {
 -                      $phpCode .= '$this->'.($append ? 'append' : 'assign').'('.$assignVar.', ob_get_clean());'."\n";
 -              }
 -              
 -              $phpCode .= "\$this->v['tpl']['template'] = \$outerTemplateName".$hash.";\n";
 -              $phpCode .= "\$this->v['tpl']['includedTemplates'][".$file."] = 1;\n";
 -              if ($once) $phpCode .= "}\n";
 -              $phpCode .= '?>';
 -              
 -              return $phpCode;
 -      }
 -      
 -      /**
 -       * Parses an argument list and returns the keys and values in an associative
 -       * array.
 -       * 
 -       * @param       string          $tagArgs
 -       * @param       string          $tag
 -       * @return      array
 -       * @throws      SystemException
 -       */
 -      public function parseTagArgs($tagArgs, $tag) {
 -              // replace strings
 -              $tagArgs = $this->replaceQuotes($tagArgs);
 -              
 -              // validate tag arguments
 -              if (!preg_match('~^(?:\s+\w+\s*=\s*[^=]*(?=\s|$))*$~s', $tagArgs)) {
 -                      throw new SystemException(static::formatSyntaxError('syntax error in tag {'.$tag.'}', $this->currentIdentifier, $this->currentLineNo));
 -              }
 -              
 -              // parse tag arguments
 -              $matches = [];
 -              // find all variables
 -              preg_match_all('~\s+(\w+)\s*=\s*([^=]*)(?=\s|$)~s', $tagArgs, $matches);
 -              $args = [];
 -              for ($i = 0, $j = count($matches[1]); $i < $j; $i++) {
 -                      $name = $matches[1][$i];
 -                      $string = $this->compileVariableTag($matches[2][$i], false);
 -                      
 -                      // reinserts strings
 -                      foreach (StringStack::getStack('singleQuote') as $hash => $value) {
 -                              if (mb_strpos($string, $hash) !== false) {
 -                                      $string = str_replace($hash, $value, $string);
 -                              }
 -                      }
 -                      foreach (StringStack::getStack('doubleQuote') as $hash => $value) {
 -                              if (mb_strpos($string, $hash) !== false) {
 -                                      $string = str_replace($hash, $value, $string);
 -                              }
 -                      }
 -                      
 -                      $args[$name] = $string;
 -              }
 -              
 -              // clear stack
 -              $this->reinsertQuotes('');
 -              
 -              return $args;
 -      }
 -      
 -      /**
 -       * Takes an array created by TemplateCompiler::parseTagArgs() and creates
 -       * a string.
 -       * 
 -       * @param       array           $args
 -       * @return      string          $args
 -       */
 -      public static function makeArgString($args) {
 -              $argString = '';
 -              foreach ($args as $key => $val) {
 -                      if ($argString != '') {
 -                              $argString .= ', ';
 -                      }
 -                      $argString .= "'$key' => $val";
 -              }
 -              return $argString;
 -      }
 -      
 -      /**
 -       * Returns a formatted syntax error message.
 -       * 
 -       * @param       string          $errorMsg
 -       * @param       string          $file
 -       * @param       integer         $line
 -       * @return      string
 -       */
 -      public static function formatSyntaxError($errorMsg, $file = null, $line = null) {
 -              $errorMsg = 'Template compilation failed: '.$errorMsg;
 -              if ($file && $line) {
 -                      $errorMsg .= " in template '$file' on line $line";
 -              }
 -              else if ($file && !$line) {
 -                      $errorMsg .= " in template '$file'";
 -              }
 -              return $errorMsg;
 -      }
 -      
 -      /**
 -       * Compiles an {if} tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $tagArgs
 -       * @param       boolean         $elseif         true, if this tag is an else tag
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileIfTag($tagArgs, $elseif = false) {
 -              $tagArgs = $this->replaceQuotes($tagArgs);
 -              $tagArgs = str_replace([' ', "\n"], '', $tagArgs);
 -              
 -              // split tags
 -              preg_match_all('~('.$this->conditionOperatorPattern.')~', $tagArgs, $matches);
 -              $operators = $matches[1];
 -              $values = preg_split('~(?:'.$this->conditionOperatorPattern.')~', $tagArgs);
 -              $leftParentheses = 0;
 -              $result = '';
 -              
 -              for ($i = 0, $j = count($values); $i < $j; $i++) {
 -                      $operator = (isset($operators[$i]) ? $operators[$i] : null);
 -                      
 -                      if ($operator !== '!' && $values[$i] == '') {
 -                              throw new SystemException(static::formatSyntaxError('syntax error in tag {'.($elseif ? 'elseif' : 'if').'}', $this->currentIdentifier, $this->currentLineNo));
 -                      }
 -                      
 -                      $leftParenthesis = mb_substr_count($values[$i], '(');
 -                      $rightParenthesis = mb_substr_count($values[$i], ')');
 -                      if ($leftParenthesis > $rightParenthesis) {
 -                              $leftParentheses += $leftParenthesis - $rightParenthesis;
 -                              $value = mb_substr($values[$i], $leftParenthesis - $rightParenthesis);
 -                              $result .= str_repeat('(', $leftParenthesis - $rightParenthesis);
 -                              
 -                              if (str_replace('(', '', mb_substr($values[$i], 0, $leftParenthesis - $rightParenthesis)) != '') {
 -                                      throw new SystemException(static::formatSyntaxError('syntax error in tag {'.($elseif ? 'elseif' : 'if').'}', $this->currentIdentifier, $this->currentLineNo));
 -                              }
 -                      }
 -                      else if ($leftParenthesis < $rightParenthesis) {
 -                              $leftParentheses += $leftParenthesis - $rightParenthesis;
 -                              $value = mb_substr($values[$i], 0, $leftParenthesis - $rightParenthesis);
 -                              
 -                              if ($leftParentheses < 0 || str_replace(')', '', mb_substr($values[$i], $leftParenthesis - $rightParenthesis)) != '') {
 -                                      throw new SystemException(static::formatSyntaxError('syntax error in tag {'.($elseif ? 'elseif' : 'if').'}', $this->currentIdentifier, $this->currentLineNo));
 -                              }
 -                      }
 -                      else $value = $values[$i];
 -                      
 -                      try {
 -                              $result .= $this->compileVariableTag($value, false);
 -                      }
 -                      catch (SystemException $e) {
 -                              throw new SystemException(static::formatSyntaxError('syntax error in tag {'.($elseif ? 'elseif' : 'if').'}', $this->currentIdentifier, $this->currentLineNo), 0, nl2br($e, false));
 -                      }
 -                      
 -                      if ($leftParenthesis < $rightParenthesis) {
 -                              $result .= str_repeat(')', $rightParenthesis - $leftParenthesis);
 -                      }
 -                      
 -                      if ($operator) $result .= ' '.$operator.' ';
 -              }
 -              
 -              return '<?php '.($elseif ? '} elseif' : 'if').' ('.$result.') { ?>';
 -      }
 -      
 -      /**
 -       * Adds a tag to the tag stack.
 -       * 
 -       * @param       string          $tag
 -       */
 -      public function pushTag($tag) {
 -              $this->tagStack[] = [$tag, $this->currentLineNo];
 -      }
 -      
 -      /**
 -       * Deletes a tag from the tag stack.
 -       * 
 -       * @param       string          $tag
 -       * @return      string          $tag
 -       */
 -      public function popTag($tag) {
 -              list($openTag, ) = array_pop($this->tagStack);
 -              if ($tag == $openTag) {
 -                      return $openTag;
 -              }
 -              if ($tag == 'if' && ($openTag == 'else' || $openTag == 'elseif')) {
 -                      return $this->popTag($tag);
 -              }
 -              if ($tag == 'foreach' && $openTag == 'foreachelse') {
 -                      return $this->popTag($tag);
 -              }
 -              if ($tag == 'section' && $openTag == 'sectionelse') {
 -                      return $this->popTag($tag);
 -              }
 -      }
 -      
 -      /**
 -       * Compiles an output tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $tag
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      protected function compileOutputTag($tag) {
 -              $encodeHTML = false;
 -              $formatNumeric = false;
 -              if ($tag[0] == '@') {
 -                      $tag = mb_substr($tag, 1);
 -              }
 -              else if ($tag[0] == '#') {
 -                      $tag = mb_substr($tag, 1);
 -                      $formatNumeric = true;
 -              }
 -              else {
 -                      $encodeHTML = true;
 -              }
 -              
 -              // check for forbidden constants
 -              if (preg_match('~^(RELATIVE_)?([A-Z]+)_DIR$~', $tag, $matches)) {
 -                      $application = mb_strtolower($matches[2]);
 -                      if ($application == 'wcf') {
 -                              $application = '';
 -                      }
 -                      else {
 -                              $application = "'{$application}'";
 -                      }
 -                      
 -                      throw new SystemException("Accessing internal constant '".$tag."' is disallowed, please use '\$__wcf->getPath(".$application.")' instead");
 -              }
 -              
 -              $parsedTag = $this->compileVariableTag($tag);
 -              
 -              // the @ operator at the beginning of an output avoids
 -              // the default call of StringUtil::encodeHTML()
 -              if ($encodeHTML) {
 -                      $parsedTag = 'wcf\util\StringUtil::encodeHTML('.$parsedTag.')';
 -              }
 -              // the # operator at the beginning of an output instructs
 -              // the complier to call the StringUtil::formatNumeric() method
 -              else if ($formatNumeric) {
 -                      $parsedTag = 'wcf\util\StringUtil::formatNumeric('.$parsedTag.')';
 -              }
 -              
 -              return '<?='.$parsedTag.';?>';
 -      }
 -      
 -      /**
 -       * Compiles a variable tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $variable
 -       * @param       string          $type
 -       * @param       boolean         $allowConstants
 -       * @return      string
 -       */
 -      protected function compileSimpleVariable($variable, $type = '', $allowConstants = true) {
 -              if ($type == '') $type = $this->getVariableType($variable);
 -              
 -              if ($type == 'variable') return '$this->v[\''.substr($variable, 1).'\']';
 -              else if ($type == 'string') return $variable;
 -              else if ($allowConstants && ($variable == 'true' || $variable == 'false' || $variable == 'null' || preg_match('/^[A-Z0-9_]*$/', $variable))) return $variable;
 -              else return "'".$variable."'";
 -      }
 -      
 -      /**
 -       * Compiles a modifier tag and returns the compiled PHP code.
 -       * 
 -       * @param       array           $data
 -       * @return      string
 -       */
 -      protected function compileModifier($data) {
 -              if (isset($data['className'])) {
 -                      return "\$this->pluginObjects['".$data['className']."']->execute([".implode(',', $data['parameter'])."], \$this)";
 -              }
 -              else {
 -                      return $data['name'].'('.implode(',', $data['parameter']).')';
 -              }
 -      }
 -      
 -      /**
 -       * Returns type of the given variable.
 -       * 
 -       * @param       string          $variable
 -       * @return      string
 -       */
 -      protected function getVariableType($variable) {
 -              if (substr($variable, 0, 1) == '$') return 'variable';
 -              else if (substr($variable, 0, 2) == '@@') return 'string';
 -              else return 'constant';
 -      }
 -      
 -      /**
 -       * Compiles a variable tag and returns the compiled PHP code.
 -       * 
 -       * @param       string          $tag
 -       * @param       boolean         $replaceQuotes
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      public function compileVariableTag($tag, $replaceQuotes = true) {
 -              // replace all quotes with unique hash values
 -              $compiledTag = $tag;
 -              if ($replaceQuotes) $compiledTag = $this->replaceQuotes($compiledTag);
 -              // replace numbers and special constants
 -              $compiledTag = $this->replaceConstants($compiledTag);
 -              
 -              // split tags
 -              preg_match_all('~('.$this->variableOperatorPattern.')~', $compiledTag, $matches);
 -              $operators = $matches[1];
 -              $values = preg_split('~(?:'.$this->variableOperatorPattern.')~', $compiledTag);
 -              
 -              // parse tags
 -              $statusStack = [0 => 'start'];
 -              $result = '';
 -              $modifierData = null;
 -              for ($i = 0, $j = count($values); $i < $j; $i++) {
 -                      // check value
 -                      $status = end($statusStack);
 -                      $operator = (isset($operators[$i]) ? $operators[$i] : null);
 -                      $values[$i] = trim($values[$i]);
 -                      
 -                      if ($values[$i] !== '') {
 -                              $variableType = $this->getVariableType($values[$i]);
 -                              
 -                              switch ($status) {
 -                                      case 'start': 
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[0] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'object access':
 -                                              if (/*strpos($values[$i], '$') !== false || */strpos($values[$i], '@@') !== false) {
 -                                                      throw new SystemException(static::formatSyntaxError("unexpected '->".$values[$i]."' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                              }
 -                                              if (strpos($values[$i], '$') !== false) {
 -                                                      $result .= '{'.$this->compileSimpleVariable($values[$i], $variableType).'}';
 -                                              }
 -                                              else {
 -                                                      $result .= $values[$i];
 -                                              }
 -                                              
 -                                              $statusStack[count($statusStack) - 1] = $status = 'object';
 -                                      break;
 -                                      
 -                                      case 'object method start':
 -                                              $statusStack[count($statusStack) - 1] = 'object method';
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'object method parameter separator':
 -                                              array_pop($statusStack);
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'dot access':
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType, false);
 -                                              $result .= ']';
 -                                              $statusStack[count($statusStack) - 1] = $status = 'variable';
 -                                      break;
 -                                      
 -                                      case 'object method':
 -                                      case 'left parenthesis':
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'bracket open':
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType, false);
 -                                              $statusStack[] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'math':
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[count($statusStack) - 1] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'modifier end':
 -                                              $result .= $this->compileSimpleVariable($values[$i], $variableType);
 -                                              $statusStack[] = $status = $variableType;
 -                                      break;
 -                                      
 -                                      case 'modifier':
 -                                              if (strpos($values[$i], '$') !== false || strpos($values[$i], '@@') !== false) {
 -                                                      throw new SystemException(static::formatSyntaxError("unknown modifier '".$values[$i]."'", $this->currentIdentifier, $this->currentLineNo));
 -                                              }
 -                                              
 -                                              // handle modifier name
 -                                              $modifierData['name'] = $values[$i];
 -                                              $className = $this->template->getPluginClassName('modifier', $modifierData['name']);
 -                                              if (class_exists($className)) {
 -                                                      $modifierData['className'] = $className;
 -                                                      $this->autoloadPlugins[$modifierData['className']] = $modifierData['className'];
 -                                              }
 -                                              else if (!function_exists($modifierData['name']) && !in_array($modifierData['name'], $this->unknownPHPFunctions)) {
 -                                                      throw new SystemException(static::formatSyntaxError(
 -                                                              "unknown modifier '".$values[$i]."'",
 -                                                              $this->currentIdentifier,
 -                                                              $this->currentLineNo
 -                                                      ));
 -                                              }
 -                                              else if (
 -                                                      in_array($modifierData['name'], $this->disabledPHPFunctions)
 -                                                      || (ENABLE_ENTERPRISE_MODE && !in_array($modifierData['name'], $this->enterpriseFunctions))
 -                                              ) {
 -                                                      throw new SystemException(static::formatSyntaxError(
 -                                                              "disabled function '".$values[$i]."'",
 -                                                              $this->currentIdentifier,
 -                                                              $this->currentLineNo
 -                                                      ));
 -                                              }
 -                                              
 -                                              $statusStack[count($statusStack) - 1] = $status = 'modifier end';
 -                                      break;
 -                                      
 -                                      case 'object':
 -                                      case 'constant': 
 -                                      case 'variable':
 -                                      case 'string': 
 -                                              throw new SystemException(static::formatSyntaxError('unknown tag {'.$tag.'}', $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                              }
 -                      }
 -                      
 -                      // check operator
 -                      if ($operator !== null) {
 -                              switch ($operator) {
 -                                      case '.': 
 -                                              if ($status == 'variable' || $status == 'object') {
 -                                                      if ($status == 'object') $statusStack[count($statusStack) - 1] = 'variable';
 -                                                      $result .= '[';
 -                                                      $statusStack[] = 'dot access';
 -                                                      break;
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected '.' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // object access
 -                                      case '->':
 -                                              if ($status == 'variable' || $status == 'object') {
 -                                                      $result .= $operator;
 -                                                      $statusStack[count($statusStack) - 1] = 'object access';
 -                                                      break;
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected '->' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // left parenthesis
 -                                      case '(':
 -                                              if ($status == 'object') {
 -                                                      $statusStack[count($statusStack) - 1] = 'variable';
 -                                                      $statusStack[] = 'object method start';
 -                                                      $result .= $operator;
 -                                                      break;
 -                                              }
 -                                              else if ($status == 'math' || $status == 'start' || $status == 'left parenthesis' || $status == 'bracket open' || $status == 'modifier end') {
 -                                                      if ($status == 'start') $statusStack[count($statusStack) - 1] = 'constant';
 -                                                      $statusStack[] = 'left parenthesis';
 -                                                      $result .= $operator;
 -                                                      break;
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected '(' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // right parenthesis
 -                                      case ')': 
 -                                              while ($oldStatus = array_pop($statusStack)) {
 -                                                      if ($oldStatus != 'variable' && $oldStatus != 'object' && $oldStatus != 'constant' && $oldStatus != 'string') {
 -                                                              if ($oldStatus == 'object method start' || $oldStatus == 'object method' || $oldStatus == 'left parenthesis') {
 -                                                                      $result .= $operator;
 -                                                                      break 2;
 -                                                              }
 -                                                              else break;
 -                                                      }
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected ')' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // bracket open
 -                                      case '[': 
 -                                              if ($status == 'variable' || $status == 'object') {
 -                                                      if ($status == 'object') $statusStack[count($statusStack) - 1] = 'variable';
 -                                                      $statusStack[] = 'bracket open';
 -                                                      $result .= $operator;
 -                                                      break;
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected '[' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // bracket close
 -                                      case ']':
 -                                              while ($oldStatus = array_pop($statusStack)) {
 -                                                      if ($oldStatus != 'variable' && $oldStatus != 'object' && $oldStatus != 'constant' && $oldStatus != 'string') {
 -                                                              if ($oldStatus == 'bracket open') {
 -                                                                      $result .= $operator;
 -                                                                      break 2;
 -                                                              }
 -                                                              else break;
 -                                                      }
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected ']' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      // modifier
 -                                      case '|': 
 -                                              // handle previous modifier
 -                                              if ($modifierData !== null) {
 -                                                      if ($result !== '') $modifierData['parameter'][] = $result;
 -                                                      $result = $this->compileModifier($modifierData);
 -                                              }
 -                                              
 -                                              // clear status stack
 -                                              while ($oldStatus = array_pop($statusStack)) {
 -                                                      if ($oldStatus != 'variable' && $oldStatus != 'object' && $oldStatus != 'constant' && $oldStatus != 'string' && $oldStatus != 'modifier end') {
 -                                                              throw new SystemException(static::formatSyntaxError("unexpected '|' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                                      }
 -                                              }
 -                                              
 -                                              $statusStack = [0 => 'modifier'];
 -                                              $modifierData = ['name' => '', 'parameter' => [0 => $result]];
 -                                              $result = '';
 -                                      break;
 -                                      
 -                                      // modifier parameter
 -                                      case ':': 
 -                                              while ($oldStatus = array_pop($statusStack)) {
 -                                                      if ($oldStatus != 'variable' && $oldStatus != 'object' && $oldStatus != 'constant' && $oldStatus != 'string') {
 -                                                              if ($oldStatus == 'modifier end') {
 -                                                                      $statusStack[] = 'modifier end';
 -                                                                      if ($result !== '') $modifierData['parameter'][] = $result;
 -                                                                      $result = '';
 -                                                                      break 2;
 -                                                              }
 -                                                              else break;
 -                                                      }
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected ':' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                                      
 -                                      case ',':
 -                                              while ($oldStatus = array_pop($statusStack)) {
 -                                                      if ($oldStatus != 'variable' && $oldStatus != 'object' && $oldStatus != 'constant' && $oldStatus != 'string') {
 -                                                              if ($oldStatus == 'object method') {
 -                                                                      $result .= $operator;
 -                                                                      $statusStack[] = 'object method';
 -                                                                      $statusStack[] = 'object method parameter separator';
 -                                                                      break 2;
 -                                                              }
 -                                                              else break;
 -                                                      }
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected ',' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                              break;
 -                                      
 -                                      // math operators
 -                                      case '+':
 -                                      case '-':
 -                                      case '*':
 -                                      case '/':
 -                                      case '%':
 -                                      case '^':
 -                                              if ($status == 'variable' || $status == 'object' || $status == 'constant' || $status == 'string' || $status == 'modifier end') {
 -                                                      $result .= $operator;
 -                                                      $statusStack[count($statusStack) - 1] = 'math';
 -                                                      break;
 -                                              }
 -                                              
 -                                              throw new SystemException(static::formatSyntaxError("unexpected '".$operator."' in tag '".$tag."'", $this->currentIdentifier, $this->currentLineNo));
 -                                      break;
 -                              }
 -                      }
 -              }
 -              
 -              // handle open modifier
 -              if ($modifierData !== null) {
 -                      if ($result !== '') $modifierData['parameter'][] = $result;
 -                      $result = $this->compileModifier($modifierData);
 -              }
 -              
 -              // reinserts strings
 -              $result = $this->reinsertQuotes($result);
 -              $result = $this->reinsertConstants($result);
 -              
 -              return $result;
 -      }
 -      
 -      /**
 -       * Generates the regexp pattern.
 -       */
 -      protected function buildPattern() {
 -              $this->variableOperatorPattern = '\-\>|\.|\(|\)|\[|\]|\||\:|\+|\-|\*|\/|\%|\^|\,';
 -              $this->conditionOperatorPattern = '===|!==|==|!=|<=|<|>=|(?<!-)>|\|\||&&|!|=';
 -              $this->escapedPattern = '(?<!\\\\)';
 -              $this->validVarnamePattern = '(?:[a-zA-Z_][a-zA-Z_0-9]*)';
 -              $this->constantPattern = '(?:[A-Z_][A-Z_0-9]*)';
 -              $this->doubleQuotePattern = '"(?:[^"\\\\]+|\\\\.)*"';
 -              $this->singleQuotePattern = '\'(?:[^\'\\\\]+|\\\\.)*\'';
 -              $this->quotePattern = '(?:' . $this->doubleQuotePattern . '|' . $this->singleQuotePattern . ')';
 -              $this->numericPattern = '(?i)(?:(?:\-?\d+(?:\.\d+)?)|true|false|null)';
 -              $this->simpleVarPattern = '(?:\$('.$this->validVarnamePattern.'))';
 -              $this->outputPattern = '(?:(?:@|#)?(?:'.$this->constantPattern.'|'.$this->quotePattern.'|'.$this->numericPattern.'|'.$this->simpleVarPattern.'|\())';
 -      }
 -      
 -      /**
 -       * Returns the instance of the template engine class.
 -       * 
 -       * @return      TemplateEngine
 -       */
 -      public function getTemplate() {
 -              return $this->template;
 -      }
 -      
 -      /**
 -       * Returns the left delimiter for template tags.
 -       * 
 -       * @return      string
 -       */
 -      public function getLeftDelimiter() {
 -              return $this->leftDelimiter;
 -      }
 -      
 -      /**
 -       * Returns the right delimiter for template tags.
 -       * 
 -       * @return      string
 -       */
 -      public function getRightDelimiter() {
 -              return $this->rightDelimiter;
 -      }
 -      
 -      /**
 -       * Returns the name of the current template.
 -       * 
 -       * @return      string
 -       */
 -      public function getCurrentIdentifier() {
 -              return $this->currentIdentifier;
 -      }
 -      
 -      /**
 -       * Returns the current line number.
 -       * 
 -       * @return      integer
 -       */
 -      public function getCurrentLineNo() {
 -              return $this->currentLineNo;
 -      }
 -      
 -      /**
 -       * Applies the prefilters to the given string.
 -       * 
 -       * @param       string          $templateName
 -       * @param       string          $string
 -       * @return      string
 -       * @throws      SystemException
 -       */
 -      public function applyPrefilters($templateName, $string) {
 -              foreach ($this->template->getPrefilters() as $prefilter) {
 -                      if (!is_object($prefilter)) {
 -                              $className = $this->template->getPluginClassName('prefilter', $prefilter);
 -                              if (!class_exists($className)) {
 -                                      throw new SystemException(static::formatSyntaxError('unable to find prefilter class '.$className, $this->currentIdentifier));
 -                              }
 -                              $prefilter = new $className();
 -                      }
 -                      
 -                      if ($prefilter instanceof IPrefilterTemplatePlugin) {
 -                              $string = $prefilter->execute($templateName, $string, $this);
 -                      }
 -                      else {
 -                              throw new SystemException(static::formatSyntaxError("Prefilter '".(is_object($prefilter) ? get_class($prefilter) : $prefilter)."' does not implement the interface 'IPrefilterTemplatePlugin'", $this->currentIdentifier));
 -                      }
 -              }
 -              
 -              return $string;
 -      }
 -      
 -      /**
 -       * Replaces all {literal} Tags with unique hash values.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function replaceLiterals($string) {
 -              return preg_replace_callback("~".$this->ldq."literal".$this->rdq."(.*?)".$this->ldq."/literal".$this->rdq."~s", [$this, 'replaceLiteralsCallback'], $string);
 -      }
 -      
 -      /**
 -       * Reinserts the literal tags.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function reinsertLiterals($string) {
 -              return StringStack::reinsertStrings($string, 'literal');
 -      }
 -      
 -      /**
 -       * Callback function used in replaceLiterals()
 -       * 
 -       * @param       string[]        $matches
 -       * @return      string
 -       */
 -      private function replaceLiteralsCallback($matches) {
 -              return StringStack::pushToStringStack($matches[1], 'literal');
 -      }
 -      
 -      /**
 -       * Removes template comments
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function removeComments($string) {
 -              return preg_replace("~".$this->ldq."\*.*?\*".$this->rdq."~s", '', $string);
 -      }
 -      
 -      /**
 -       * Replaces all quotes with unique hash values.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function replaceQuotes($string) {
 -              $string = preg_replace_callback('~\'([^\'\\\\]+|\\\\.)*\'~', [$this, 'replaceSingleQuotesCallback'], $string);
 -              $string = preg_replace_callback('~"([^"\\\\]+|\\\\.)*"~', [$this, 'replaceDoubleQuotesCallback'], $string);
 -              
 -              return $string;
 -      }
 -      
 -      /**
 -       * Callback function used in replaceQuotes()
 -       *
 -       * @param       string[]        $matches
 -       * @return      string
 -       */
 -      private function replaceSingleQuotesCallback($matches) {
 -              return StringStack::pushToStringStack($matches[0], 'singleQuote');
 -      }
 -      
 -      /**
 -       * Callback function used in replaceQuotes()
 -       *
 -       * @param       string[]        $matches
 -       * @return      string
 -       */
 -      private function replaceDoubleQuotesCallback($matches) {
 -              // parse unescaped simple vars in double quotes
 -              // replace $foo with {$this->v['foo']}
 -              $matches[0] = preg_replace('~'.$this->escapedPattern.$this->simpleVarPattern.'~', '{$this->v[\'\\1\']}', $matches[0]);
 -              return StringStack::pushToStringStack($matches[0], 'doubleQuote');
 -      }
 -      
 -      /**
 -       * Reinserts the quotes.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function reinsertQuotes($string) {
 -              $string = StringStack::reinsertStrings($string, 'singleQuote');
 -              $string = StringStack::reinsertStrings($string, 'doubleQuote');
 -              
 -              return $string;
 -      }
 -      
 -      /**
 -       * Replaces all constants with unique hash values.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function replaceConstants($string) {
 -              return preg_replace_callback('~(?<=^|'.$this->variableOperatorPattern.')(?i)((?:\-?\d+(?:\.\d+)?)|true|false|null)(?=$|'.$this->variableOperatorPattern.')~', [$this, 'replaceConstantsCallback'], $string);
 -      }
 -      
 -      /**
 -       * Callback function used in replaceConstants()
 -       *
 -       * @param       string[]        $matches
 -       * @return      string
 -       */
 -      private function replaceConstantsCallback($matches) {
 -              return StringStack::pushToStringStack($matches[1], 'constants');
 -      }
 -      
 -      /**
 -       * Reinserts the constants.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function reinsertConstants($string) {
 -              return StringStack::reinsertStrings($string, 'constants');
 -      }
 -      
 -      /**
 -       * Replaces all php tags.
 -       * 
 -       * @param       string          $string
 -       * @return      string
 -       */
 -      public function replacePHPTags($string) {
 -              if (mb_strpos($string, '<?') !== false) {
 -                      $string = str_replace('<?php', '@@PHP_START_TAG@@', $string);
 -                      $string = str_replace('<?', '@@PHP_SHORT_START_TAG@@', $string);
 -                      $string = str_replace('?>', '@@PHP_END_TAG@@', $string);
 -                      $string = str_replace('@@PHP_END_TAG@@', "<?='?>';?>\n", $string);
 -                      $string = str_replace('@@PHP_SHORT_START_TAG@@', "<?='<?';?>\n", $string);
 -                      $string = str_replace('@@PHP_START_TAG@@', "<?='<?php';?>\n", $string);
 -              }
 -              
 -              return $string;
 -      }
 +class TemplateScriptingCompiler
 +{
 +    /**
 +     * template engine object
 +     * @var TemplateEngine
 +     */
 +    protected $template;
 +
 +    /**
 +     * PHP functions that can be used in the modifier syntax and are unknown
 +     * to PHP's function_exists function
 +     * @var string[]
 +     */
 +    protected $unknownPHPFunctions = ['isset', 'unset', 'empty'];
 +
 +    /**
 +     * PHP functions that can not be used in the modifier syntax
 +     * @var string[]
 +     */
 +    protected $disabledPHPFunctions = [
 +        'system',
 +        'exec',
 +        'passthru',
 +        'shell_exec', // command line execution
 +        'include',
 +        'require',
 +        'include_once',
 +        'require_once', // includes
 +        'eval',
 +        'virtual',
 +        'call_user_func_array',
 +        'call_user_func',
 +        'assert', // code execution
 +    ];
 +
 +    /**
 +     * PHP functions and modifiers that can be used in enterprise mode
 +     * @var string[]
 +     */
 +    protected $enterpriseFunctions = [
 +        'addslashes',
 +        'array_diff',
 +        'array_fill',
 +        'array_keys',
 +        'array_pop',
 +        'array_slice',
 +        'array_values',
 +        'base64_decode',
 +        'base64_encode',
 +        'ceil',
 +        'concat',
 +        'constant',
 +        'count',
 +        'currency',
 +        'current',
 +        'date',
 +        'defined',
 +        'doubleval',
 +        'empty',
 +        'end',
 +        'explode',
 +        'file_exists',
 +        'filesize',
 +        'floatval',
 +        'floor',
 +        'function_exists',
 +        'get_class',
 +        'gmdate',
 +        'hash',
 +        'htmlspecialchars',
 +        'html_entity_decode',
 +        'http_build_query',
 +        'implode',
 +        'in_array',
 +        'is_array',
 +        'is_null',
 +        'is_numeric',
 +        'is_object',
 +        'iterator_count',
 +        'intval',
 +        'is_subclass_of',
 +        'isset',
 +        'json_encode',
 +        'key',
 +        'lcfirst',
 +        'ltrim',
 +        'max',
 +        'mb_strpos',
 +        'mb_strlen',
 +        'mb_strpos',
 +        'mb_strtolower',
 +        'mb_strtoupper',
 +        'mb_substr',
 +        'md5',
 +        'method_exists',
 +        'microtime',
 +        'min',
 +        'nl2br',
 +        'number_format',
 +        'parse_url',
 +        'preg_match',
 +        'preg_replace',
 +        'print_r',
 +        'rawurlencode',
 +        'reset',
 +        'round',
 +        'sha1',
 +        'spl_object_hash',
 +        'sprintf',
 +        'strip_tags',
 +        'strlen',
 +        'strpos',
 +        'strtolower',
 +        'strtotime',
 +        'strtoupper',
 +        'str_pad',
 +        'str_repeat',
 +        'str_replace',
 +        'str_ireplace',
 +        'substr',
 +        'trim',
 +        'ucfirst',
 +        'uniqid',
 +        'urlencode',
 +        'version_compare',
 +        'wcfDebug',
 +        'wordwrap',
 +    ];
 +
 +    /**
 +     * pattern to match variable operators like -> or .
 +     * @var string
 +     */
 +    protected $variableOperatorPattern;
 +
 +    /**
 +     * pattern to match condition operators like == or <
 +     * @var string
 +     */
 +    protected $conditionOperatorPattern;
 +
 +    /**
 +     * negative lookbehind for a backslash
 +     * @var string
 +     */
 +    protected $escapedPattern;
 +
 +    /**
 +     * pattern to match valid variable names
 +     * @var string
 +     */
 +    protected $validVarnamePattern;
 +
 +    /**
 +     * pattern to match constants like CONSTANT or __CONSTANT
 +     * @var string
 +     */
 +    protected $constantPattern;
 +
 +    /**
 +     * pattern to match double quoted strings like "blah" or "quote: \"blah\""
 +     * @var string
 +     */
 +    protected $doubleQuotePattern;
 +
 +    /**
 +     * pattern to match single quoted strings like 'blah' or 'don\'t'
 +     * @var string
 +     */
 +    protected $singleQuotePattern;
 +
 +    /**
 +     * pattern to match single or double quoted strings
 +     * @var string
 +     */
 +    protected $quotePattern;
 +
 +    /**
 +     * pattern to match numbers, true, false and null
 +     * @var string
 +     */
 +    protected $numericPattern;
 +
 +    /**
 +     * pattern to match simple variables like $foo
 +     * @var string
 +     */
 +    protected $simpleVarPattern;
 +
 +    /**
 +     * pattern to match outputs like @$foo or #CONST
 +     * @var string
 +     */
 +    protected $outputPattern;
 +
 +    /**
 +     * identifier of currently compiled template
 +     * @var string
 +     */
 +    protected $currentIdentifier;
 +
 +    /**
 +     * current line number during template compilation
 +     * @var string
 +     */
 +    protected $currentLineNo;
 +
 +    /**
 +     * list of automatically loaded tenplate plugins
 +     * @var string[]
 +     */
 +    protected $autoloadPlugins = [];
 +
 +    /**
 +     * stack with template tags data
 +     * @var array
 +     */
 +    protected $tagStack = [];
 +
 +    /**
 +     * list of loaded compiler plugin objects
 +     * @var ICompilerTemplatePlugin[]
 +     */
 +    protected $compilerPlugins = [];
 +
 +    /**
 +     * stack used to compile the capture tag
 +     * @var array
 +     */
 +    protected $captureStack = [];
 +
 +    /**
 +     * left delimiter of template syntax
 +     * @var string
 +     */
 +    protected $leftDelimiter = '{';
 +
 +    /**
 +     * right delimiter of template syntax
 +     * @var string
 +     */
 +    protected $rightDelimiter = '}';
 +
 +    /**
 +     * left delimiter of template syntax used in regular expressions
 +     * @var string
 +     */
 +    protected $ldq;
 +
 +    /**
 +     * right delimiter of template syntax used in regular expressions
 +     * @var string
 +     */
 +    protected $rdq;
 +
 +    /**
 +     * list of static includes per template
 +     * @var string[][]
 +     */
 +    protected $staticIncludes = [];
 +
 +    /**
 +     * data of all currently active `foreach` loops; entry data:
 +     *
 +     *  hash => unique foreach loop hash
 +     *  itemVar => template code for the `item` variable
 +     *  keyVar => (optional) template code for the `key` variable
 +     *
 +     * @var string[][]
 +     */
 +    protected $foreachLoops = [];
 +
 +    /**
 +     * Creates a new TemplateScriptingCompiler object.
 +     *
 +     * @param TemplateEngine $template
 +     */
 +    public function __construct(TemplateEngine $template)
 +    {
 +        $this->template = $template;
 +
 +        // quote left and right delimiter for use in regular expressions
 +        $this->ldq = \preg_quote($this->leftDelimiter, '~') . '(?=\S)';
 +        $this->rdq = '(?<=\S)' . \preg_quote($this->rightDelimiter, '~');
 +
 +        // build regular expressions
 +        $this->buildPattern();
 +    }
 +
 +    /**
 +     * Compiles the source of a template.
 +     *
 +     * @param string $identifier
 +     * @param string $sourceContent
 +     * @param array $metaData
 +     * @param bool $isolated
 +     * @return  array|bool
 +     * @throws  SystemException
 +     */
 +    public function compileString($identifier, $sourceContent, array $metaData = [], $isolated = false)
 +    {
 +        $previousData = [];
 +        if ($isolated) {
 +            $previousData = [
 +                'autoloadPlugins' => $this->autoloadPlugins,
 +                'currentIdentifier' => $this->currentIdentifier,
 +                'currentLineNo' => $this->currentLineNo,
 +                'tagStack' => $this->tagStack,
 +            ];
 +        } else {
 +            $this->staticIncludes = [];
 +        }
 +
 +        // reset vars
 +        $this->autoloadPlugins = $this->tagStack = [];
 +        $this->currentIdentifier = $identifier;
 +        $this->currentLineNo = 1;
 +
 +        // apply prefilters
 +        $sourceContent = $this->applyPrefilters($identifier, $sourceContent);
 +
 +        // replace all {literal} Tags with unique hash values
 +        $sourceContent = $this->replaceLiterals($sourceContent);
 +
 +        // handle <?php tags
 +        $sourceContent = $this->replacePHPTags($sourceContent);
 +
 +        // remove comments
 +        $sourceContent = $this->removeComments($sourceContent);
 +
 +        // match all template tags
 +        $matches = [];
 +        \preg_match_all("~" . $this->ldq . "(.*?)" . $this->rdq . "~s", $sourceContent, $matches);
 +        $templateTags = $matches[1];
 +
 +        // Split content by template tags to obtain non-template content
 +        $textBlocks = \preg_split("~" . $this->ldq . ".*?" . $this->rdq . "~s", $sourceContent);
 +
 +        // compile the template tags into php-code
 +        $compiledTags = [];
 +        for ($i = 0, $j = \count($templateTags); $i < $j; $i++) {
 +            $this->currentLineNo += \mb_substr_count($textBlocks[$i], "\n");
 +
 +            if ($templateTags[$i] === '') {
 +                // avoid empty JavaScript object literals being recognized
 +                // as template scripting tags
 +                $compiledTags[] = '{}';
 +            } else {
 +                $compiledTags[] = $this->compileTag($templateTags[$i], $identifier, $metaData);
 +            }
 +
 +            $this->currentLineNo += \mb_substr_count($templateTags[$i], "\n");
 +        }
 +
 +        // throw error messages for unclosed tags
 +        if (\count($this->tagStack) > 0) {
 +            foreach ($this->tagStack as $tagStack) {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        'unclosed tag {' . $tagStack[0] . '}',
 +                        $this->currentIdentifier,
 +                        $tagStack[1]
 +                    )
 +                );
 +            }
 +
 +            return false;
 +        }
 +
 +        $compiledContent = '';
 +        // Interleave the compiled contents and text blocks to get the final result.
 +        for ($i = 0, $j = \count($compiledTags); $i < $j; $i++) {
 +            if ($compiledTags[$i] == '') {
 +                // tag result empty, remove first newline from following text block
 +                $textBlocks[$i + 1] = \preg_replace('%^(\r\n|\r|\n)%', '', $textBlocks[$i + 1]);
 +            }
 +            $compiledContent .= $textBlocks[$i] . $compiledTags[$i];
 +        }
 +        $compiledContent .= $textBlocks[$i];
 +        $compiledContent = \rtrim($compiledContent);
 +
 +        // reinsert {literal} Tags
 +        $compiledContent = $this->reinsertLiterals($compiledContent);
 +
 +        // include Plugins
 +        $compiledAutoloadPlugins = '';
 +        if (\count($this->autoloadPlugins) > 0) {
 +            $compiledAutoloadPlugins = "<?php\n";
 +            foreach ($this->autoloadPlugins as $className) {
 +                $compiledAutoloadPlugins .= "if (!isset(\$this->pluginObjects['{$className}'])) {\n";
 +                $compiledAutoloadPlugins .= "\$this->pluginObjects['{$className}'] = new {$className};\n";
 +                $compiledAutoloadPlugins .= "}\n";
 +            }
 +            $compiledAutoloadPlugins .= "?>";
 +        }
 +
 +        // restore data
 +        if ($isolated) {
 +            $this->autoloadPlugins = $previousData['autoloadPlugins'];
 +            $this->currentIdentifier = $previousData['currentIdentifier'];
 +            $this->currentLineNo = $previousData['currentLineNo'];
 +            $this->tagStack = $previousData['tagStack'];
 +        }
 +
 +        return [
 +            'meta' => [
 +                'include' => $this->staticIncludes,
 +            ],
 +            'template' => $compiledAutoloadPlugins . $compiledContent,
 +        ];
 +    }
 +
 +    /**
 +     * Compiles a template tag.
 +     *
 +     * @param string $tag
 +     * @param string $identifier
 +     * @param array $metaData
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileTag($tag, $identifier, array &$metaData)
 +    {
 +        if (\preg_match('~^' . $this->outputPattern . '~s', $tag)) {
 +            // variable output
 +            return $this->compileOutputTag($tag);
 +        }
 +
 +        $match = [];
 +        // replace 'else if' with 'elseif'
 +        $tag = \preg_replace('~^else\s+if(?=\s)~i', 'elseif', $tag);
 +
 +        if (\preg_match('~^(/?\w+)~', $tag, $match)) {
 +            // build in function or plugin
 +            $tagCommand = $match[1];
 +            $tagArgs = \mb_substr($tag, \mb_strlen($tagCommand));
 +
 +            switch ($tagCommand) {
 +                case 'if':
 +                    $this->pushTag('if');
 +
 +                    return $this->compileIfTag($tagArgs);
 +
 +                case 'elseif':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'if' && $openTag != 'elseif') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unxepected {elseif}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    } elseif ($openTag == 'if') {
 +                        $this->pushTag('elseif');
 +                    }
 +
 +                    return $this->compileIfTag($tagArgs, true);
 +
 +                case 'else':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'if' && $openTag != 'elseif') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {else}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->pushTag('else');
 +
 +                    return '<?php } else { ?>';
 +
 +                case '/if':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'if' && $openTag != 'elseif' && $openTag != 'else') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {/if}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->popTag('if');
 +
 +                    return '<?php } ?>';
 +
 +                case 'include':
 +                    return $this->compileIncludeTag($tagArgs, $identifier, $metaData);
 +
 +                case 'foreach':
 +                    $this->pushTag('foreach');
 +
 +                    return $this->compileForeachTag($tagArgs);
 +
 +                case 'foreachelse':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'foreach') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {foreachelse}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->pushTag('foreachelse');
 +
 +                    return '<?php } } else { { ?>';
 +
 +                case '/foreach':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'foreach' && $openTag != 'foreachelse') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {/foreach}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->popTag('foreach');
 +
 +                    return $this->compileForeachEndTag();
 +
 +                case 'section':
 +                    $this->pushTag('section');
 +
 +                    return $this->compileSectionTag($tagArgs);
 +
 +                case 'sectionelse':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'section') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {sectionelse}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->pushTag('sectionelse');
 +
 +                    return '<?php } } else { { ?>';
 +
 +                case '/section':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'section' && $openTag != 'sectionelse') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {/section}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->popTag('section');
 +
 +                    return "<?php } } ?>";
 +
 +                case 'capture':
 +                    $this->pushTag('capture');
 +
 +                    return $this->compileCaptureTag(true, $tagArgs);
 +
 +                case '/capture':
 +                    [$openTag] = \end($this->tagStack);
 +                    if ($openTag != 'capture') {
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unexpected {/capture}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                    }
 +                    $this->popTag('capture');
 +
 +                    return $this->compileCaptureTag(false);
 +
 +                case 'ldelim':
 +                    return $this->leftDelimiter;
 +
 +                case 'rdelim':
 +                    return $this->rightDelimiter;
 +
 +                default:
 +                    // 1) compiler functions first
 +                    if ($phpCode = $this->compileCompilerPlugin($tagCommand, $tagArgs)) {
 +                        return $phpCode;
 +                    }
 +                    // 2) block functions
 +                    if ($phpCode = $this->compileBlockPlugin($tagCommand, $tagArgs)) {
 +                        return $phpCode;
 +                    }
 +                    // 3) functions
 +                    if ($phpCode = $this->compileFunctionPlugin($tagCommand, $tagArgs)) {
 +                        return $phpCode;
 +                    }
 +            }
 +        }
 +
 +        throw new SystemException(
 +            static::formatSyntaxError(
 +                'unknown tag {' . $tag . '}',
 +                $this->currentIdentifier,
 +                $this->currentLineNo
 +            )
 +        );
 +    }
 +
 +    /**
 +     * Compiles a function plugin and returns the output of the plugin or false
 +     * if the plugin doesn't exist.
 +     *
 +     * @param string $tagCommand
 +     * @param string $tagArgs
 +     * @return  mixed
 +     */
 +    protected function compileFunctionPlugin($tagCommand, $tagArgs)
 +    {
 +        $className = $this->template->getPluginClassName('function', $tagCommand);
 +        if (!\class_exists($className)) {
 +            return false;
 +        }
 +        $this->autoloadPlugins[$className] = $className;
 +
 +        $tagArgs = static::makeArgString($this->parseTagArgs($tagArgs, $tagCommand));
 +
 +        return "<?=\$this->pluginObjects['" . $className . "']->execute([" . $tagArgs . "], \$this);?>";
 +    }
 +
 +    /**
 +     * Compiles a block plugin and returns the output of the plugin or false
 +     * if the plugin doesn't exist.
 +     *
 +     * @param string $tagCommand
 +     * @param string $tagArgs
 +     * @return  mixed
 +     * @throws  SystemException
 +     */
 +    protected function compileBlockPlugin($tagCommand, $tagArgs)
 +    {
 +        // check whether this is the start ({block}) or the
 +        // end tag ({/block})
 +        if (\substr($tagCommand, 0, 1) == '/') {
 +            $tagCommand = \substr($tagCommand, 1);
 +            $startTag = false;
 +        } else {
 +            $startTag = true;
 +        }
 +
 +        $className = $this->template->getPluginClassName('block', $tagCommand);
 +        if (!\class_exists($className)) {
 +            return false;
 +        }
 +        $this->autoloadPlugins[$className] = $className;
 +
 +        if ($startTag) {
 +            $this->pushTag($tagCommand);
 +
 +            $tagArgs = static::makeArgString($this->parseTagArgs($tagArgs, $tagCommand));
 +
 +            $phpCode = "<?php \$this->tagStack[] = ['" . $tagCommand . "', [" . $tagArgs . "]];\n";
 +            $phpCode .= "\$this->pluginObjects['" . $className . "']->init(\$this->tagStack[count(\$this->tagStack) - 1][1], \$this);\n";
 +            $phpCode .= "while (\$this->pluginObjects['" . $className . "']->next(\$this)) { ob_start(); ?>";
 +        } else {
 +            [$openTag] = \end($this->tagStack);
 +            if ($openTag != $tagCommand) {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        'unexpected {/' . $tagCommand . '}',
 +                        $this->currentIdentifier,
 +                        $this->currentLineNo
 +                    )
 +                );
 +            }
 +            $this->popTag($tagCommand);
 +            $phpCode = "<?php echo \$this->pluginObjects['" . $className . "']->execute(\$this->tagStack[count(\$this->tagStack) - 1][1], ob_get_clean(), \$this); }\n";
 +            $phpCode .= "array_pop(\$this->tagStack);?>";
 +        }
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Compiles a compiler function/block and returns the output of the plugin
 +     * or false if the plugin doesn't exist.
 +     *
 +     * @param string $tagCommand
 +     * @param string $tagArgs
 +     * @return  mixed
 +     * @throws  SystemException
 +     */
 +    protected function compileCompilerPlugin($tagCommand, $tagArgs)
 +    {
 +        // check whether this is the start ({block}) or the
 +        // end tag ({/block})
 +        if (\substr($tagCommand, 0, 1) == '/') {
 +            $tagCommand = \substr($tagCommand, 1);
 +            $startTag = false;
 +        } else {
 +            $startTag = true;
 +        }
 +
 +        $className = $this->template->getPluginClassName('compiler', $tagCommand);
 +        // if necessary load plugin from plugin-dir
 +        if (!isset($this->compilerPlugins[$className])) {
 +            if (!\class_exists($className)) {
 +                return false;
 +            }
 +
 +            $this->compilerPlugins[$className] = new $className();
 +
 +            if (!($this->compilerPlugins[$className] instanceof ICompilerTemplatePlugin)) {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        "Compiler plugin '" . $tagCommand . "' does not implement the interface 'ICompilerTemplatePlugin'",
 +                        $this->currentIdentifier
 +                    )
 +                );
 +            }
 +        }
 +
 +        // execute plugin
 +        if ($startTag) {
 +            $tagArgs = $this->parseTagArgs($tagArgs, $tagCommand);
 +            $phpCode = $this->compilerPlugins[$className]->executeStart($tagArgs, $this);
 +        } else {
 +            $phpCode = $this->compilerPlugins[$className]->executeEnd($this);
 +        }
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Compiles a capture tag and returns the compiled PHP code.
 +     *
 +     * @param bool $startTag
 +     * @param string $captureTag
 +     * @return  string
 +     */
 +    protected function compileCaptureTag($startTag, $captureTag = null)
 +    {
 +        if ($startTag) {
 +            $append = false;
 +            $args = $this->parseTagArgs($captureTag, 'capture');
 +
 +            if (!isset($args['name'])) {
 +                $args['name'] = "'default'";
 +            }
 +
 +            if (!isset($args['assign'])) {
 +                if (isset($args['append'])) {
 +                    $args['assign'] = $args['append'];
 +                    $append = true;
 +                } else {
 +                    $args['assign'] = '';
 +                }
 +            }
 +
 +            $this->captureStack[] = ['name' => $args['name'], 'variable' => $args['assign'], 'append' => $append];
 +
 +            return '<?php ob_start(); ?>';
 +        } else {
 +            $capture = \array_pop($this->captureStack);
 +            $phpCode = "<?php\n";
 +            $phpCode .= "\$this->v['tpl']['capture'][" . $capture['name'] . "] = ob_get_clean();\n";
 +            if (!empty($capture['variable'])) {
 +                $phpCode .= "\$this->" . ($capture['append'] ? 'append' : 'assign') . "(" . $capture['variable'] . ", \$this->v['tpl']['capture'][" . $capture['name'] . "]);\n";
 +            }
 +            $phpCode .= "?>";
 +
 +            return $phpCode;
 +        }
 +    }
 +
 +    /**
 +     * Compiles a section tag and returns the compiled PHP code.
 +     *
 +     * @param string $sectionTag
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileSectionTag($sectionTag)
 +    {
 +        $args = $this->parseTagArgs($sectionTag, 'section');
 +
 +        // check arguments
 +        if (!isset($args['loop'])) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    "missing 'loop' attribute in section tag",
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +        if (!isset($args['name'])) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    "missing 'name' attribute in section tag",
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +        if (!isset($args['show'])) {
 +            $args['show'] = true;
 +        }
 +
 +        $sectionProp = "\$this->v['tpl']['section'][" . $args['name'] . "]";
 +
 +        $phpCode = "<?php\n";
 +        $phpCode .= "if (" . $args['loop'] . ") {\n";
 +        $phpCode .= $sectionProp . " = [];\n";
 +        $phpCode .= $sectionProp . "['loop'] = (is_array(" . $args['loop'] . ") ? count(" . $args['loop'] . ") : max(0, (int)" . $args['loop'] . "));\n";
 +        $phpCode .= $sectionProp . "['show'] = " . $args['show'] . ";\n";
 +        if (!isset($args['step'])) {
 +            $phpCode .= $sectionProp . "['step'] = 1;\n";
 +        } else {
 +            $phpCode .= $sectionProp . "['step'] = " . $args['step'] . ";\n";
 +        }
 +        if (!isset($args['max'])) {
 +            $phpCode .= $sectionProp . "['max'] = " . $sectionProp . "['loop'];\n";
 +        } else {
 +            $phpCode .= $sectionProp . "['max'] = (" . $args['max'] . " < 0 ? " . $sectionProp . "['loop'] : " . $args['max'] . ");\n";
 +        }
 +        if (!isset($args['start'])) {
 +            $phpCode .= $sectionProp . "['start'] = (" . $sectionProp . "['step'] > 0 ? 0 : " . $sectionProp . "['loop'] - 1);\n";
 +        } else {
 +            $phpCode .= $sectionProp . "['start'] = " . $args['start'] . ";\n";
 +            $phpCode .= "if (" . $sectionProp . "['start'] < 0) {\n";
 +            $phpCode .= $sectionProp . "['start'] = max(" . $sectionProp . "['step'] > 0 ? 0 : -1, " . $sectionProp . "['loop'] + " . $sectionProp . "['start']);\n}\n";
 +            $phpCode .= "else {\n";
 +            $phpCode .= $sectionProp . "['start'] = min(" . $sectionProp . "['start'], " . $sectionProp . "['step'] > 0 ? " . $sectionProp . "['loop'] : " . $sectionProp . "['loop'] - 1);\n}\n";
 +        }
 +
 +        if (!isset($args['start']) && !isset($args['step']) && !isset($args['max'])) {
 +            $phpCode .= $sectionProp . "['total'] = " . $sectionProp . "['loop'];\n";
 +        } else {
 +            $phpCode .= $sectionProp . "['total'] = min(ceil((" . $sectionProp . "['step'] > 0 ? " . $sectionProp . "['loop'] - " . $sectionProp . "['start'] : " . $sectionProp . "['start'] + 1) / abs(" . $sectionProp . "['step'])), " . $sectionProp . "['max']);\n";
 +        }
 +        $phpCode .= "if (" . $sectionProp . "['total'] == 0) " . $sectionProp . "['show'] = false;\n";
 +        $phpCode .= "} else {\n";
 +        $phpCode .= "" . $sectionProp . "['total'] = 0;\n";
 +        $phpCode .= "" . $sectionProp . "['show'] = false;}\n";
 +
 +        $phpCode .= "if (" . $sectionProp . "['show']) {\n";
 +        $phpCode .= "for (" . $sectionProp . "['index'] = " . $sectionProp . "['start'], " . $sectionProp . "['rowNumber'] = 1;\n";
 +        $phpCode .= $sectionProp . "['rowNumber'] <= " . $sectionProp . "['total'];\n";
 +        $phpCode .= $sectionProp . "['index'] += " . $sectionProp . "['step'], " . $sectionProp . "['rowNumber']++) {\n";
 +        $phpCode .= "\$this->v[" . $args['name'] . "] = " . $sectionProp . "['index'];\n";
 +        $phpCode .= $sectionProp . "['previousIndex'] = " . $sectionProp . "['index'] - " . $sectionProp . "['step'];\n";
 +        $phpCode .= $sectionProp . "['nextIndex'] = " . $sectionProp . "['index'] + " . $sectionProp . "['step'];\n";
 +        $phpCode .= $sectionProp . "['first'] = (" . $sectionProp . "['rowNumber'] == 1);\n";
 +        $phpCode .= $sectionProp . "['last'] = (" . $sectionProp . "['rowNumber'] == " . $sectionProp . "['total']);\n";
 +        $phpCode .= "?>";
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Compiles a foreach tag and returns the compiled PHP code.
 +     *
 +     * @param string $foreachTag
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileForeachTag($foreachTag)
 +    {
 +        $args = $this->parseTagArgs($foreachTag, 'foreach');
 +
 +        // check arguments
 +        if (!isset($args['from'])) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    "missing 'from' attribute in foreach tag",
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +        if (!isset($args['item'])) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    "missing 'item' attribute in foreach tag",
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +
 +        $foreachProp = '';
 +        if (isset($args['name'])) {
 +            $foreachProp = "\$this->v['tpl']['foreach'][" . $args['name'] . "]";
 +        }
 +
 +        $hash = StringUtil::getRandomID();
 +        $foreachHash = "\$_foreach_" . $hash;
 +        $foreachData = ['hash' => $hash];
 +
 +        $phpCode = "<?php\n";
 +        $phpCode .= $foreachHash . " = " . $args['from'] . ";\n";
 +
++        $itemVar = \mb_substr($args['item'], 0, 1) != '$' ? "\$this->v[" . $args['item'] . "]" : $args['item'];
++        $foreachData['itemVar'] = $itemVar;
++
++        $phpCode .= "\$this->foreachVars['{$hash}'] = [];\n";
++        $phpCode .= "\$this->foreachVars['{$hash}']['item'] = {$itemVar} ?? null;\n";
++
 +        if (empty($foreachProp)) {
 +            $phpCode .= "if ((is_countable(" . $foreachHash . ") && count(" . $foreachHash . ") > 0) || (!is_countable(" . $foreachHash . ") && " . $foreachHash . ")) {\n";
 +        } else {
 +            $phpCode .= $foreachHash . "_cnt = (" . $foreachHash . " !== null ? 1 : 0);\n";
 +            $phpCode .= "if (is_countable(" . $foreachHash . ")) {\n";
 +            $phpCode .= $foreachHash . "_cnt = count(" . $foreachHash . ");\n";
 +            $phpCode .= "}\n";
 +            $phpCode .= $foreachProp . "['total'] = " . $foreachHash . "_cnt;\n";
 +            $phpCode .= $foreachProp . "['show'] = (" . $foreachProp . "['total'] > 0 ? true : false);\n";
 +            $phpCode .= $foreachProp . "['iteration'] = 0;\n";
 +            $phpCode .= "if (" . $foreachHash . "_cnt > 0) {\n";
 +        }
 +
-         $itemVar = \mb_substr($args['item'], 0, 1) != '$' ? "\$this->v[" . $args['item'] . "]" : $args['item'];
-         $foreachData['itemVar'] = $itemVar;
-         $phpCode .= "\$this->foreachVars['{$hash}'] = [];\n";
-         $phpCode .= "\$this->foreachVars['{$hash}']['item'] = {$itemVar} ?? null;\n";
 +        if (isset($args['key'])) {
 +            $keyVar = \mb_substr($args['key'], 0, 1) != '$' ? "\$this->v[" . $args['key'] . "]" : $args['key'];
 +            $foreachData['keyVar'] = $keyVar;
 +
 +            $phpCode .= "\$this->foreachVars['{$hash}']['key'] = {$keyVar} ?? null;\n";
 +
 +            $phpCode .= "foreach (" . $foreachHash . " as {$keyVar} => {$itemVar}) {\n";
 +        } else {
 +            $phpCode .= "foreach (" . $foreachHash . " as {$itemVar}) {\n";
 +        }
 +
 +        if (!empty($foreachProp)) {
 +            $phpCode .= $foreachProp . "['first'] = (" . $foreachProp . "['iteration'] == 0 ? true : false);\n";
 +            $phpCode .= $foreachProp . "['last'] = ((" . $foreachProp . "['iteration'] == " . $foreachProp . "['total'] - 1) ? true : false);\n";
 +            $phpCode .= $foreachProp . "['iteration']++;\n";
 +        }
 +
 +        $phpCode .= "?>";
 +
 +        $this->foreachLoops[] = $foreachData;
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Compiles a `/foreach` tag and returns the compiled PHP code.
 +     *
 +     * @return  string
 +     * @since   5.2
 +     */
 +    protected function compileForeachEndTag()
 +    {
 +        $foreachData = \array_pop($this->foreachLoops);
 +
 +        // unset `item` and `key` variables and restore their sandboxed values
 +        $phpCode = "<?php }\n";
 +        $phpCode .= "unset({$foreachData['itemVar']});";
 +        $phpCode .= "{$foreachData['itemVar']} = \$this->foreachVars['{$foreachData['hash']}']['item'];\n";
 +
 +        if (isset($foreachData['keyVar'])) {
 +            $phpCode .= "unset({$foreachData['keyVar']});";
 +            $phpCode .= "if (array_key_exists('key', \$this->foreachVars['{$foreachData['hash']}'])) {\n";
 +            $phpCode .= "{$foreachData['keyVar']} = \$this->foreachVars['{$foreachData['hash']}']['key'];\n";
 +            $phpCode .= "}\n";
 +        }
 +
 +        $phpCode .= "unset(\$this->foreachVars['{$foreachData['hash']}']);\n";
 +        $phpCode .= " } ?>";
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Compiles an include tag and returns the compiled PHP code.
 +     *
 +     * @param string $includeTag
 +     * @param string $identifier
 +     * @param array $metaData
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileIncludeTag($includeTag, $identifier, array $metaData)
 +    {
 +        $args = $this->parseTagArgs($includeTag, 'include');
 +        $append = false;
 +
 +        // check arguments
 +        if (!isset($args['file'])) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    "missing 'file' attribute in include tag",
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +
 +        // get filename
 +        $file = $args['file'];
 +        unset($args['file']);
 +
 +        // special parameters
 +        $assignVar = false;
 +        if (isset($args['assign'])) {
 +            $assignVar = $args['assign'];
 +            unset($args['assign']);
 +        }
 +
 +        if (isset($args['append'])) {
 +            $assignVar = $args['append'];
 +            $append = true;
 +            unset($args['append']);
 +        }
 +
 +        $once = false;
 +        if (isset($args['once'])) {
 +            $once = $args['once'];
 +            unset($args['once']);
 +        }
 +
 +        $application = "'wcf'";
 +        if (isset($args['application'])) {
 +            $application = $args['application'];
 +            unset($args['application']);
 +        }
 +
 +        if (\preg_match('~^(\'|\")(.*)\1$~', $application, $matches)) {
 +            $application = $matches[2];
 +        }
 +
 +        $sandbox = false;
 +        if (isset($args['sandbox'])) {
 +            $sandbox = $args['sandbox'];
 +            unset($args['sandbox']);
 +        }
 +
 +        $sandbox = ($sandbox === 'true' || $sandbox === true || $sandbox == 1);
 +
 +        $staticInclude = true;
 +        if (
 +            $sandbox
 +            || $assignVar !== false
 +            || $once !== false
 +            || \strpos($application, '$') !== false
 +            || \strpos($file, '$') !== false
 +        ) {
 +            $staticInclude = false;
 +        }
 +
 +        $templateName = \substr($file, 1, -1);
 +
 +        // check for static includes
 +        if ($staticInclude) {
 +            $phpCode = '';
 +            if (!isset($this->staticIncludes[$application])) {
 +                $this->staticIncludes[$application] = [];
 +            }
 +
 +            if (!\in_array($templateName, $this->staticIncludes[$application])) {
 +                $this->staticIncludes[$application][] = $templateName;
 +            }
 +
 +            // pass remaining tag args as variables
 +            if (!empty($args)) {
 +                foreach ($args as $variable => $value) {
 +                    if (\substr($value, 0, 1) == "'") {
 +                        // string values
 +                        $phpCode .= "\$this->v['" . $variable . "'] = " . $value . ";\n";
 +                    } else {
 +                        if (\preg_match('~^\$this->v\[\'(.*)\'\]$~U', $value, $matches)) {
 +                            // value is a variable itself
 +                            $phpCode .= "\$this->v['" . $variable . "'] = " . $value . ";\n";
 +                        } else {
 +                            // value is boolean, an integer or anything else
 +                            $phpCode .= "\$this->v['" . $variable . "'] = " . $value . ";\n";
 +                        }
 +                    }
 +                }
 +            }
 +            if (!empty($phpCode)) {
 +                $phpCode = "<?php\n" . $phpCode . "\n?>";
 +            }
 +
 +            $sourceFilename = $this->template->getSourceFilename($templateName, $application);
 +
 +            $data = $this->compileString($templateName, \file_get_contents($sourceFilename), [
 +                'application' => $application,
 +                'data' => null,
 +                'filename' => '',
 +            ], true);
 +
 +            return $phpCode . $data['template'];
 +        }
 +
 +        // make argument string
 +        $argString = static::makeArgString($args);
 +
 +        // build phpCode
 +        $phpCode = "<?php\n";
 +        if ($once) {
 +            $phpCode .= "if (!isset(\$this->v['tpl']['includedTemplates'][" . $file . "])) {\n";
 +        }
 +        $hash = StringUtil::getRandomID();
 +        $phpCode .= "\$outerTemplateName" . $hash . " = \$this->v['tpl']['template'];\n";
 +
 +        if ($assignVar !== false) {
 +            $phpCode .= "ob_start();\n";
 +        }
 +
 +        if (\strpos($application, '$') === false) {
 +            $application = "'" . $application . "'";
 +        }
 +        $phpCode .= '$this->includeTemplate(' . $file . ', ' . $application . ', [' . $argString . '], ' . ($sandbox ? 1 : 0) . ');' . "\n";
 +
 +        if ($assignVar !== false) {
 +            $phpCode .= '$this->' . ($append ? 'append' : 'assign') . '(' . $assignVar . ', ob_get_clean());' . "\n";
 +        }
 +
 +        $phpCode .= "\$this->v['tpl']['template'] = \$outerTemplateName" . $hash . ";\n";
 +        $phpCode .= "\$this->v['tpl']['includedTemplates'][" . $file . "] = 1;\n";
 +        if ($once) {
 +            $phpCode .= "}\n";
 +        }
 +        $phpCode .= '?>';
 +
 +        return $phpCode;
 +    }
 +
 +    /**
 +     * Parses an argument list and returns the keys and values in an associative
 +     * array.
 +     *
 +     * @param string $tagArgs
 +     * @param string $tag
 +     * @return  array
 +     * @throws  SystemException
 +     */
 +    public function parseTagArgs($tagArgs, $tag)
 +    {
 +        // replace strings
 +        $tagArgs = $this->replaceQuotes($tagArgs);
 +
 +        // validate tag arguments
 +        if (!\preg_match('~^(?:\s+\w+\s*=\s*[^=]*(?=\s|$))*$~s', $tagArgs)) {
 +            throw new SystemException(
 +                static::formatSyntaxError(
 +                    'syntax error in tag {' . $tag . '}',
 +                    $this->currentIdentifier,
 +                    $this->currentLineNo
 +                )
 +            );
 +        }
 +
 +        // parse tag arguments
 +        $matches = [];
 +        // find all variables
 +        \preg_match_all('~\s+(\w+)\s*=\s*([^=]*)(?=\s|$)~s', $tagArgs, $matches);
 +        $args = [];
 +        for ($i = 0, $j = \count($matches[1]); $i < $j; $i++) {
 +            $name = $matches[1][$i];
 +            $string = $this->compileVariableTag($matches[2][$i], false);
 +
 +            // reinserts strings
 +            foreach (StringStack::getStack('singleQuote') as $hash => $value) {
 +                if (\mb_strpos($string, $hash) !== false) {
 +                    $string = \str_replace($hash, $value, $string);
 +                }
 +            }
 +            foreach (StringStack::getStack('doubleQuote') as $hash => $value) {
 +                if (\mb_strpos($string, $hash) !== false) {
 +                    $string = \str_replace($hash, $value, $string);
 +                }
 +            }
 +
 +            $args[$name] = $string;
 +        }
 +
 +        // clear stack
 +        $this->reinsertQuotes('');
 +
 +        return $args;
 +    }
 +
 +    /**
 +     * Takes an array created by TemplateCompiler::parseTagArgs() and creates
 +     * a string.
 +     *
 +     * @param array $args
 +     * @return  string      $args
 +     */
 +    public static function makeArgString($args)
 +    {
 +        $argString = '';
 +        foreach ($args as $key => $val) {
 +            if ($argString != '') {
 +                $argString .= ', ';
 +            }
 +            $argString .= "'{$key}' => {$val}";
 +        }
 +
 +        return $argString;
 +    }
 +
 +    /**
 +     * Returns a formatted syntax error message.
 +     *
 +     * @param string $errorMsg
 +     * @param string $file
 +     * @param int $line
 +     * @return  string
 +     */
 +    public static function formatSyntaxError($errorMsg, $file = null, $line = null)
 +    {
 +        $errorMsg = 'Template compilation failed: ' . $errorMsg;
 +        if ($file && $line) {
 +            $errorMsg .= " in template '{$file}' on line {$line}";
 +        } elseif ($file && !$line) {
 +            $errorMsg .= " in template '{$file}'";
 +        }
 +
 +        return $errorMsg;
 +    }
 +
 +    /**
 +     * Compiles an {if} tag and returns the compiled PHP code.
 +     *
 +     * @param string $tagArgs
 +     * @param bool $elseif true, if this tag is an else tag
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileIfTag($tagArgs, $elseif = false)
 +    {
 +        $tagArgs = $this->replaceQuotes($tagArgs);
 +        $tagArgs = \str_replace([' ', "\n"], '', $tagArgs);
 +
 +        // split tags
 +        \preg_match_all('~(' . $this->conditionOperatorPattern . ')~', $tagArgs, $matches);
 +        $operators = $matches[1];
 +        $values = \preg_split('~(?:' . $this->conditionOperatorPattern . ')~', $tagArgs);
 +        $leftParentheses = 0;
 +        $result = '';
 +
 +        for ($i = 0, $j = \count($values); $i < $j; $i++) {
 +            $operator = ($operators[$i] ?? null);
 +
 +            if ($operator !== '!' && $values[$i] == '') {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        'syntax error in tag {' . ($elseif ? 'elseif' : 'if') . '}',
 +                        $this->currentIdentifier,
 +                        $this->currentLineNo
 +                    )
 +                );
 +            }
 +
 +            $leftParenthesis = \mb_substr_count($values[$i], '(');
 +            $rightParenthesis = \mb_substr_count($values[$i], ')');
 +            if ($leftParenthesis > $rightParenthesis) {
 +                $leftParentheses += $leftParenthesis - $rightParenthesis;
 +                $value = \mb_substr($values[$i], $leftParenthesis - $rightParenthesis);
 +                $result .= \str_repeat('(', $leftParenthesis - $rightParenthesis);
 +
 +                if (\str_replace('(', '', \mb_substr($values[$i], 0, $leftParenthesis - $rightParenthesis)) != '') {
 +                    throw new SystemException(
 +                        static::formatSyntaxError(
 +                            'syntax error in tag {' . ($elseif ? 'elseif' : 'if') . '}',
 +                            $this->currentIdentifier,
 +                            $this->currentLineNo
 +                        )
 +                    );
 +                }
 +            } elseif ($leftParenthesis < $rightParenthesis) {
 +                $leftParentheses += $leftParenthesis - $rightParenthesis;
 +                $value = \mb_substr($values[$i], 0, $leftParenthesis - $rightParenthesis);
 +
 +                if (
 +                    $leftParentheses < 0 || \str_replace(
 +                        ')',
 +                        '',
 +                        \mb_substr($values[$i], $leftParenthesis - $rightParenthesis)
 +                    ) != ''
 +                ) {
 +                    throw new SystemException(
 +                        static::formatSyntaxError(
 +                            'syntax error in tag {' . ($elseif ? 'elseif' : 'if') . '}',
 +                            $this->currentIdentifier,
 +                            $this->currentLineNo
 +                        )
 +                    );
 +                }
 +            } else {
 +                $value = $values[$i];
 +            }
 +
 +            try {
 +                $result .= $this->compileVariableTag($value, false);
 +            } catch (SystemException $e) {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        'syntax error in tag {' . ($elseif ? 'elseif' : 'if') . '}',
 +                        $this->currentIdentifier,
 +                        $this->currentLineNo
 +                    ),
 +                    0,
 +                    \nl2br($e, false)
 +                );
 +            }
 +
 +            if ($leftParenthesis < $rightParenthesis) {
 +                $result .= \str_repeat(')', $rightParenthesis - $leftParenthesis);
 +            }
 +
 +            if ($operator) {
 +                $result .= ' ' . $operator . ' ';
 +            }
 +        }
 +
 +        return '<?php ' . ($elseif ? '} elseif' : 'if') . ' (' . $result . ') { ?>';
 +    }
 +
 +    /**
 +     * Adds a tag to the tag stack.
 +     *
 +     * @param string $tag
 +     */
 +    public function pushTag($tag)
 +    {
 +        $this->tagStack[] = [$tag, $this->currentLineNo];
 +    }
 +
 +    /**
 +     * Deletes a tag from the tag stack.
 +     *
 +     * @param string $tag
 +     * @return  string      $tag
 +     */
 +    public function popTag($tag)
 +    {
 +        [$openTag,] = \array_pop($this->tagStack);
 +        if ($tag == $openTag) {
 +            return $openTag;
 +        }
 +        if ($tag == 'if' && ($openTag == 'else' || $openTag == 'elseif')) {
 +            return $this->popTag($tag);
 +        }
 +        if ($tag == 'foreach' && $openTag == 'foreachelse') {
 +            return $this->popTag($tag);
 +        }
 +        if ($tag == 'section' && $openTag == 'sectionelse') {
 +            return $this->popTag($tag);
 +        }
 +    }
 +
 +    /**
 +     * Compiles an output tag and returns the compiled PHP code.
 +     *
 +     * @param string $tag
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    protected function compileOutputTag($tag)
 +    {
 +        $encodeHTML = false;
 +        $formatNumeric = false;
 +        if ($tag[0] == '@') {
 +            $tag = \mb_substr($tag, 1);
 +        } elseif ($tag[0] == '#') {
 +            $tag = \mb_substr($tag, 1);
 +            $formatNumeric = true;
 +        } else {
 +            $encodeHTML = true;
 +        }
 +
 +        // check for forbidden constants
 +        if (\preg_match('~^(RELATIVE_)?([A-Z]+)_DIR$~', $tag, $matches)) {
 +            $application = \mb_strtolower($matches[2]);
 +            if ($application == 'wcf') {
 +                $application = '';
 +            } else {
 +                $application = "'{$application}'";
 +            }
 +
 +            throw new SystemException("Accessing internal constant '" . $tag . "' is disallowed, please use '\$__wcf->getPath(" . $application . ")' instead");
 +        }
 +
 +        $parsedTag = $this->compileVariableTag($tag);
 +
 +        // the @ operator at the beginning of an output avoids
 +        // the default call of StringUtil::encodeHTML()
 +        if ($encodeHTML) {
 +            $parsedTag = 'wcf\util\StringUtil::encodeHTML(' . $parsedTag . ')';
 +        }
 +        // the # operator at the beginning of an output instructs
 +        // the complier to call the StringUtil::formatNumeric() method
 +        elseif ($formatNumeric) {
 +            $parsedTag = 'wcf\util\StringUtil::formatNumeric(' . $parsedTag . ')';
 +        }
 +
 +        return '<?=' . $parsedTag . ';?>';
 +    }
 +
 +    /**
 +     * Compiles a variable tag and returns the compiled PHP code.
 +     *
 +     * @param string $variable
 +     * @param string $type
 +     * @param bool $allowConstants
 +     * @return  string
 +     */
 +    protected function compileSimpleVariable($variable, $type = '', $allowConstants = true)
 +    {
 +        if ($type == '') {
 +            $type = $this->getVariableType($variable);
 +        }
 +
 +        if ($type == 'variable') {
 +            return '$this->v[\'' . \substr($variable, 1) . '\']';
 +        } elseif ($type == 'string') {
 +            return $variable;
 +        } elseif (
 +            $allowConstants
 +            && (
 +                $variable == 'true'
 +                || $variable == 'false'
 +                || $variable == 'null'
 +                || \preg_match('/^[A-Z0-9_]*$/', $variable)
 +            )
 +        ) {
 +            return $variable;
 +        } else {
 +            return "'" . $variable . "'";
 +        }
 +    }
 +
 +    /**
 +     * Compiles a modifier tag and returns the compiled PHP code.
 +     *
 +     * @param array $data
 +     * @return  string
 +     */
 +    protected function compileModifier($data)
 +    {
 +        if (isset($data['className'])) {
 +            return "\$this->pluginObjects['" . $data['className'] . "']->execute([" . \implode(
 +                ',',
 +                $data['parameter']
 +            ) . "], \$this)";
 +        } else {
 +            return $data['name'] . '(' . \implode(',', $data['parameter']) . ')';
 +        }
 +    }
 +
 +    /**
 +     * Returns type of the given variable.
 +     *
 +     * @param string $variable
 +     * @return  string
 +     */
 +    protected function getVariableType($variable)
 +    {
 +        if (\substr($variable, 0, 1) == '$') {
 +            return 'variable';
 +        } elseif (\substr($variable, 0, 2) == '@@') {
 +            return 'string';
 +        } else {
 +            return 'constant';
 +        }
 +    }
 +
 +    /**
 +     * Compiles a variable tag and returns the compiled PHP code.
 +     *
 +     * @param string $tag
 +     * @param bool $replaceQuotes
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    public function compileVariableTag($tag, $replaceQuotes = true)
 +    {
 +        // replace all quotes with unique hash values
 +        $compiledTag = $tag;
 +        if ($replaceQuotes) {
 +            $compiledTag = $this->replaceQuotes($compiledTag);
 +        }
 +        // replace numbers and special constants
 +        $compiledTag = $this->replaceConstants($compiledTag);
 +
 +        // split tags
 +        \preg_match_all('~(' . $this->variableOperatorPattern . ')~', $compiledTag, $matches);
 +        $operators = $matches[1];
 +        $values = \preg_split('~(?:' . $this->variableOperatorPattern . ')~', $compiledTag);
 +
 +        // parse tags
 +        $statusStack = [0 => 'start'];
 +        $result = '';
 +        $modifierData = null;
 +        for ($i = 0, $j = \count($values); $i < $j; $i++) {
 +            // check value
 +            $status = \end($statusStack);
 +            $operator = ($operators[$i] ?? null);
 +            $values[$i] = \trim($values[$i]);
 +
 +            if ($values[$i] !== '') {
 +                $variableType = $this->getVariableType($values[$i]);
 +
 +                switch ($status) {
 +                    case 'start':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[0] = $status = $variableType;
 +                        break;
 +
 +                    case 'object access':
 +                        if (\strpos($values[$i], '@@') !== false) {
 +                            throw new SystemException(
 +                                static::formatSyntaxError(
 +                                    "unexpected '->" . $values[$i] . "' in tag '" . $tag . "'",
 +                                    $this->currentIdentifier,
 +                                    $this->currentLineNo
 +                                )
 +                            );
 +                        }
 +                        if (\strpos($values[$i], '$') !== false) {
 +                            $result .= '{' . $this->compileSimpleVariable($values[$i], $variableType) . '}';
 +                        } else {
 +                            $result .= $values[$i];
 +                        }
 +
 +                        $statusStack[\count($statusStack) - 1] = $status = 'object';
 +                        break;
 +
 +                    case 'object method start':
 +                        $statusStack[\count($statusStack) - 1] = 'object method';
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[] = $status = $variableType;
 +                        break;
 +
 +                    case 'object method parameter separator':
 +                        \array_pop($statusStack);
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[] = $status = $variableType;
 +                        break;
 +
 +                    case 'dot access':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType, false);
 +                        $result .= ']';
 +                        $statusStack[\count($statusStack) - 1] = $status = 'variable';
 +                        break;
 +
 +                    case 'object method':
 +                    case 'left parenthesis':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[] = $status = $variableType;
 +                        break;
 +
 +                    case 'bracket open':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType, false);
 +                        $statusStack[] = $status = $variableType;
 +                        break;
 +
 +                    case 'math':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[\count($statusStack) - 1] = $status = $variableType;
 +                        break;
 +
 +                    case 'modifier end':
 +                        $result .= $this->compileSimpleVariable($values[$i], $variableType);
 +                        $statusStack[] = $status = $variableType;
 +                        break;
 +
 +                    case 'modifier':
 +                        if (\strpos($values[$i], '$') !== false || \strpos($values[$i], '@@') !== false) {
 +                            throw new SystemException(
 +                                static::formatSyntaxError(
 +                                    "unknown modifier '" . $values[$i] . "'",
 +                                    $this->currentIdentifier,
 +                                    $this->currentLineNo
 +                                )
 +                            );
 +                        }
 +
 +                        // handle modifier name
 +                        $modifierData['name'] = $values[$i];
 +                        $className = $this->template->getPluginClassName('modifier', $modifierData['name']);
 +                        if (\class_exists($className)) {
 +                            $modifierData['className'] = $className;
 +                            $this->autoloadPlugins[$modifierData['className']] = $modifierData['className'];
 +                        } elseif (
 +                            !\function_exists($modifierData['name'])
 +                            && !\in_array($modifierData['name'], $this->unknownPHPFunctions)
 +                        ) {
 +                            throw new SystemException(static::formatSyntaxError(
 +                                "unknown modifier '" . $values[$i] . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            ));
 +                        } elseif (
 +                            \in_array($modifierData['name'], $this->disabledPHPFunctions)
 +                            || (ENABLE_ENTERPRISE_MODE && !\in_array($modifierData['name'], $this->enterpriseFunctions))
 +                        ) {
 +                            throw new SystemException(static::formatSyntaxError(
 +                                "disabled function '" . $values[$i] . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            ));
 +                        }
 +
 +                        $statusStack[\count($statusStack) - 1] = $status = 'modifier end';
 +                        break;
 +
 +                    case 'object':
 +                    case 'constant':
 +                    case 'variable':
 +                    case 'string':
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                'unknown tag {' . $tag . '}',
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +                }
 +            }
 +
 +            // check operator
 +            if ($operator !== null) {
 +                switch ($operator) {
 +                    case '.':
 +                        if ($status == 'variable' || $status == 'object') {
 +                            if ($status == 'object') {
 +                                $statusStack[\count($statusStack) - 1] = 'variable';
 +                            }
 +                            $result .= '[';
 +                            $statusStack[] = 'dot access';
 +                            break;
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected '.' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // object access
 +                    case '->':
 +                        if ($status == 'variable' || $status == 'object') {
 +                            $result .= $operator;
 +                            $statusStack[\count($statusStack) - 1] = 'object access';
 +                            break;
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected '->' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // left parenthesis
 +                    case '(':
 +                        if ($status == 'object') {
 +                            $statusStack[\count($statusStack) - 1] = 'variable';
 +                            $statusStack[] = 'object method start';
 +                            $result .= $operator;
 +                            break;
 +                        } elseif (
 +                            $status == 'math'
 +                            || $status == 'start'
 +                            || $status == 'left parenthesis'
 +                            || $status == 'bracket open'
 +                            || $status == 'modifier end'
 +                        ) {
 +                            if ($status == 'start') {
 +                                $statusStack[\count($statusStack) - 1] = 'constant';
 +                            }
 +                            $statusStack[] = 'left parenthesis';
 +                            $result .= $operator;
 +                            break;
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected '(' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // right parenthesis
 +                    case ')':
 +                        while ($oldStatus = \array_pop($statusStack)) {
 +                            if (
 +                                $oldStatus != 'variable'
 +                                && $oldStatus != 'object'
 +                                && $oldStatus != 'constant'
 +                                && $oldStatus != 'string'
 +                            ) {
 +                                if (
 +                                    $oldStatus == 'object method start'
 +                                    || $oldStatus == 'object method'
 +                                    || $oldStatus == 'left parenthesis'
 +                                ) {
 +                                    $result .= $operator;
 +                                    break 2;
 +                                } else {
 +                                    break;
 +                                }
 +                            }
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected ')' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // bracket open
 +                    case '[':
 +                        if ($status == 'variable' || $status == 'object') {
 +                            if ($status == 'object') {
 +                                $statusStack[\count($statusStack) - 1] = 'variable';
 +                            }
 +                            $statusStack[] = 'bracket open';
 +                            $result .= $operator;
 +                            break;
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected '[' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // bracket close
 +                    case ']':
 +                        while ($oldStatus = \array_pop($statusStack)) {
 +                            if (
 +                                $oldStatus != 'variable'
 +                                && $oldStatus != 'object'
 +                                && $oldStatus != 'constant'
 +                                && $oldStatus != 'string'
 +                            ) {
 +                                if ($oldStatus == 'bracket open') {
 +                                    $result .= $operator;
 +                                    break 2;
 +                                } else {
 +                                    break;
 +                                }
 +                            }
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected ']' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // modifier
 +                    case '|':
 +                        // handle previous modifier
 +                        if ($modifierData !== null) {
 +                            if ($result !== '') {
 +                                $modifierData['parameter'][] = $result;
 +                            }
 +                            $result = $this->compileModifier($modifierData);
 +                        }
 +
 +                        // clear status stack
 +                        while ($oldStatus = \array_pop($statusStack)) {
 +                            if (
 +                                $oldStatus != 'variable'
 +                                && $oldStatus != 'object'
 +                                && $oldStatus != 'constant'
 +                                && $oldStatus != 'string'
 +                                && $oldStatus != 'modifier end'
 +                            ) {
 +                                throw new SystemException(
 +                                    static::formatSyntaxError(
 +                                        "unexpected '|' in tag '" . $tag . "'",
 +                                        $this->currentIdentifier,
 +                                        $this->currentLineNo
 +                                    )
 +                                );
 +                            }
 +                        }
 +
 +                        $statusStack = [0 => 'modifier'];
 +                        $modifierData = ['name' => '', 'parameter' => [0 => $result]];
 +                        $result = '';
 +                        break;
 +
 +                    // modifier parameter
 +                    case ':':
 +                        while ($oldStatus = \array_pop($statusStack)) {
 +                            if (
 +                                $oldStatus != 'variable'
 +                                && $oldStatus != 'object'
 +                                && $oldStatus != 'constant'
 +                                && $oldStatus != 'string'
 +                            ) {
 +                                if ($oldStatus == 'modifier end') {
 +                                    $statusStack[] = 'modifier end';
 +                                    if ($result !== '') {
 +                                        $modifierData['parameter'][] = $result;
 +                                    }
 +                                    $result = '';
 +                                    break 2;
 +                                } else {
 +                                    break;
 +                                }
 +                            }
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected ':' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    case ',':
 +                        while ($oldStatus = \array_pop($statusStack)) {
 +                            if (
 +                                $oldStatus != 'variable'
 +                                && $oldStatus != 'object'
 +                                && $oldStatus != 'constant'
 +                                && $oldStatus != 'string'
 +                            ) {
 +                                if ($oldStatus == 'object method') {
 +                                    $result .= $operator;
 +                                    $statusStack[] = 'object method';
 +                                    $statusStack[] = 'object method parameter separator';
 +                                    break 2;
 +                                } else {
 +                                    break;
 +                                }
 +                            }
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected ',' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +
 +                    // math operators
 +                    case '+':
 +                    case '-':
 +                    case '*':
 +                    case '/':
 +                    case '%':
 +                    case '^':
 +                        if (
 +                            $status == 'variable'
 +                            || $status == 'object'
 +                            || $status == 'constant'
 +                            || $status == 'string'
 +                            || $status == 'modifier end'
 +                        ) {
 +                            $result .= $operator;
 +                            $statusStack[\count($statusStack) - 1] = 'math';
 +                            break;
 +                        }
 +
 +                        throw new SystemException(
 +                            static::formatSyntaxError(
 +                                "unexpected '" . $operator . "' in tag '" . $tag . "'",
 +                                $this->currentIdentifier,
 +                                $this->currentLineNo
 +                            )
 +                        );
 +                        break;
 +                }
 +            }
 +        }
 +
 +        // handle open modifier
 +        if ($modifierData !== null) {
 +            if ($result !== '') {
 +                $modifierData['parameter'][] = $result;
 +            }
 +            $result = $this->compileModifier($modifierData);
 +        }
 +
 +        // reinserts strings
 +        $result = $this->reinsertQuotes($result);
 +        $result = $this->reinsertConstants($result);
 +
 +        return $result;
 +    }
 +
 +    /**
 +     * Generates the regexp pattern.
 +     */
 +    protected function buildPattern()
 +    {
 +        $this->variableOperatorPattern = '\-\>|\.|\(|\)|\[|\]|\||\:|\+|\-|\*|\/|\%|\^|\,';
 +        $this->conditionOperatorPattern = '===|!==|==|!=|<=|<|>=|(?<!-)>|\|\||&&|!|=';
 +        $this->escapedPattern = '(?<!\\\\)';
 +        $this->validVarnamePattern = '(?:[a-zA-Z_][a-zA-Z_0-9]*)';
 +        $this->constantPattern = '(?:[A-Z_][A-Z_0-9]*)';
 +        $this->doubleQuotePattern = '"(?:[^"\\\\]+|\\\\.)*"';
 +        $this->singleQuotePattern = '\'(?:[^\'\\\\]+|\\\\.)*\'';
 +        $this->quotePattern = '(?:' . $this->doubleQuotePattern . '|' . $this->singleQuotePattern . ')';
 +        $this->numericPattern = '(?i)(?:(?:\-?\d+(?:\.\d+)?)|true|false|null)';
 +        $this->simpleVarPattern = '(?:\$(' . $this->validVarnamePattern . '))';
 +        $this->outputPattern = '(?:(?:@|#)?(?:' . $this->constantPattern . '|' . $this->quotePattern . '|' . $this->numericPattern . '|' . $this->simpleVarPattern . '|\())';
 +    }
 +
 +    /**
 +     * Returns the instance of the template engine class.
 +     *
 +     * @return  TemplateEngine
 +     */
 +    public function getTemplate()
 +    {
 +        return $this->template;
 +    }
 +
 +    /**
 +     * Returns the left delimiter for template tags.
 +     *
 +     * @return  string
 +     */
 +    public function getLeftDelimiter()
 +    {
 +        return $this->leftDelimiter;
 +    }
 +
 +    /**
 +     * Returns the right delimiter for template tags.
 +     *
 +     * @return  string
 +     */
 +    public function getRightDelimiter()
 +    {
 +        return $this->rightDelimiter;
 +    }
 +
 +    /**
 +     * Returns the name of the current template.
 +     *
 +     * @return  string
 +     */
 +    public function getCurrentIdentifier()
 +    {
 +        return $this->currentIdentifier;
 +    }
 +
 +    /**
 +     * Returns the current line number.
 +     *
 +     * @return  int
 +     */
 +    public function getCurrentLineNo()
 +    {
 +        return $this->currentLineNo;
 +    }
 +
 +    /**
 +     * Applies the prefilters to the given string.
 +     *
 +     * @param string $templateName
 +     * @param string $string
 +     * @return  string
 +     * @throws  SystemException
 +     */
 +    public function applyPrefilters($templateName, $string)
 +    {
 +        foreach ($this->template->getPrefilters() as $prefilter) {
 +            if (!\is_object($prefilter)) {
 +                $className = $this->template->getPluginClassName('prefilter', $prefilter);
 +                if (!\class_exists($className)) {
 +                    throw new SystemException(
 +                        static::formatSyntaxError(
 +                            'unable to find prefilter class ' . $className,
 +                            $this->currentIdentifier
 +                        )
 +                    );
 +                }
 +                $prefilter = new $className();
 +            }
 +
 +            if ($prefilter instanceof IPrefilterTemplatePlugin) {
 +                $string = $prefilter->execute($templateName, $string, $this);
 +            } else {
 +                throw new SystemException(
 +                    static::formatSyntaxError(
 +                        "Prefilter '" . (\is_object($prefilter) ? \get_class($prefilter) : $prefilter) . "' does not implement the interface 'IPrefilterTemplatePlugin'",
 +                        $this->currentIdentifier
 +                    )
 +                );
 +            }
 +        }
 +
 +        return $string;
 +    }
 +
 +    /**
 +     * Replaces all {literal} Tags with unique hash values.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function replaceLiterals($string)
 +    {
 +        return \preg_replace_callback(
 +            "~" . $this->ldq . "literal" . $this->rdq . "(.*?)" . $this->ldq . "/literal" . $this->rdq . "~s",
 +            [$this, 'replaceLiteralsCallback'],
 +            $string
 +        );
 +    }
 +
 +    /**
 +     * Reinserts the literal tags.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function reinsertLiterals($string)
 +    {
 +        return StringStack::reinsertStrings($string, 'literal');
 +    }
 +
 +    /**
 +     * Callback function used in replaceLiterals()
 +     *
 +     * @param string[] $matches
 +     * @return  string
 +     */
 +    private function replaceLiteralsCallback($matches)
 +    {
 +        return StringStack::pushToStringStack($matches[1], 'literal');
 +    }
 +
 +    /**
 +     * Removes template comments
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function removeComments($string)
 +    {
 +        return \preg_replace("~" . $this->ldq . "\\*.*?\\*" . $this->rdq . "~s", '', $string);
 +    }
 +
 +    /**
 +     * Replaces all quotes with unique hash values.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function replaceQuotes($string)
 +    {
 +        $string = \preg_replace_callback('~\'([^\'\\\\]+|\\\\.)*\'~', [$this, 'replaceSingleQuotesCallback'], $string);
 +
 +        return \preg_replace_callback('~"([^"\\\\]+|\\\\.)*"~', [$this, 'replaceDoubleQuotesCallback'], $string);
 +    }
 +
 +    /**
 +     * Callback function used in replaceQuotes()
 +     *
 +     * @param string[] $matches
 +     * @return  string
 +     */
 +    private function replaceSingleQuotesCallback($matches)
 +    {
 +        return StringStack::pushToStringStack($matches[0], 'singleQuote');
 +    }
 +
 +    /**
 +     * Callback function used in replaceQuotes()
 +     *
 +     * @param string[] $matches
 +     * @return  string
 +     */
 +    private function replaceDoubleQuotesCallback($matches)
 +    {
 +        // parse unescaped simple vars in double quotes
 +        // replace $foo with {$this->v['foo']}
 +        $matches[0] = \preg_replace(
 +            '~' . $this->escapedPattern . $this->simpleVarPattern . '~',
 +            '{$this->v[\'\\1\']}',
 +            $matches[0]
 +        );
 +
 +        return StringStack::pushToStringStack($matches[0], 'doubleQuote');
 +    }
 +
 +    /**
 +     * Reinserts the quotes.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function reinsertQuotes($string)
 +    {
 +        $string = StringStack::reinsertStrings($string, 'singleQuote');
 +
 +        return StringStack::reinsertStrings($string, 'doubleQuote');
 +    }
 +
 +    /**
 +     * Replaces all constants with unique hash values.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function replaceConstants($string)
 +    {
 +        return \preg_replace_callback(
 +            '~(?<=^|' . $this->variableOperatorPattern . ')(?i)((?:\-?\d+(?:\.\d+)?)|true|false|null)(?=$|' . $this->variableOperatorPattern . ')~',
 +            [$this, 'replaceConstantsCallback'],
 +            $string
 +        );
 +    }
 +
 +    /**
 +     * Callback function used in replaceConstants()
 +     *
 +     * @param string[] $matches
 +     * @return  string
 +     */
 +    private function replaceConstantsCallback($matches)
 +    {
 +        return StringStack::pushToStringStack($matches[1], 'constants');
 +    }
 +
 +    /**
 +     * Reinserts the constants.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function reinsertConstants($string)
 +    {
 +        return StringStack::reinsertStrings($string, 'constants');
 +    }
 +
 +    /**
 +     * Replaces all php tags.
 +     *
 +     * @param string $string
 +     * @return  string
 +     */
 +    public function replacePHPTags($string)
 +    {
 +        if (\mb_strpos($string, '<?') !== false) {
 +            $string = \str_replace('<?php', '@@PHP_START_TAG@@', $string);
 +            $string = \str_replace('<?', '@@PHP_SHORT_START_TAG@@', $string);
 +            $string = \str_replace('?>', '@@PHP_END_TAG@@', $string);
 +            $string = \str_replace('@@PHP_END_TAG@@', "<?='?>';?>\n", $string);
 +            $string = \str_replace('@@PHP_SHORT_START_TAG@@', "<?='<?';?>\n", $string);
 +            $string = \str_replace('@@PHP_START_TAG@@', "<?='<?php';?>\n", $string);
 +        }
 +
 +        return $string;
 +    }
  }