* @author Behrooz Shabani * @copyright 2010-2012 (c) Justin Hileman * @copyright 2012 (c) ParsPooyesh Co * @copyright 2013 (c) Behrooz Shabani * @license MIT * @version GIT: $Id$ * @link http://xamin.ir */ namespace Handlebars; /** * Handlebars parser (based on mustache) * * This class is responsible for turning raw template source into a set of * Handlebars tokens. * * @category Xamin * @package Handlebars * @author fzerorubigd * @copyright 2010-2012 (c) Justin Hileman * @copyright 2012 (c) ParsPooyesh Co * @license MIT * @version Release: @package_version@ * @link http://xamin.ir */ class Parser { /** * Process array of tokens and convert them into parse tree * * @param array $tokens Set of * * @return array Token parse tree */ public function parse(array $tokens = array()) { return $this->_buildTree(new \ArrayIterator($tokens)); } /** * Helper method for recursively building a parse tree. * * @param \ArrayIterator $tokens Stream of tokens * * @throws \LogicException when nesting errors or mismatched section tags * are encountered. * @return array Token parse tree * */ private function _buildTree(\ArrayIterator $tokens) { $stack = array(); do { $token = $tokens->current(); $tokens->next(); if ($token !== null) { switch ($token[Tokenizer::TYPE]) { case Tokenizer::T_END_SECTION: $newNodes = array(); do { $result = array_pop($stack); if ($result === null) { throw new \LogicException( 'Unexpected closing tag: /' . $token[Tokenizer::NAME] ); } if (!array_key_exists(Tokenizer::NODES, $result) && isset($result[Tokenizer::NAME]) && $result[Tokenizer::NAME] == $token[Tokenizer::NAME] ) { $result[Tokenizer::NODES] = $newNodes; $result[Tokenizer::END] = $token[Tokenizer::INDEX]; array_push($stack, $result); break; } else { array_unshift($newNodes, $result); } } while (true); break; default: array_push($stack, $token); } } } while ($tokens->valid()); return $stack; } }