mirror of
				https://github.com/Mibew/handlebars.php.git
				synced 2025-11-04 12:05:09 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			254 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
<?php
 | 
						|
/**
 | 
						|
 * This file is part of Handlebars-php
 | 
						|
 * Base on mustache-php https://github.com/bobthecow/mustache.php
 | 
						|
 * 
 | 
						|
 * PHP version 5.3
 | 
						|
 * 
 | 
						|
 * @category  Xamin
 | 
						|
 * @package   Handlebars
 | 
						|
 * @author    fzerorubigd <fzerorubigd@gmail.com>
 | 
						|
 * @author    Behrooz Shabani <everplays@gmail.com>
 | 
						|
 * @copyright 2012 (c) ParsPooyesh Co
 | 
						|
 * @license   MIT <http://opensource.org/licenses/MIT>
 | 
						|
 * @version   GIT: $Id$
 | 
						|
 * @link      http://xamin.ir
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * Handlebars context
 | 
						|
 * Context for a template
 | 
						|
 * 
 | 
						|
 * @category  Xamin
 | 
						|
 * @package   Handlebars
 | 
						|
 * @author    fzerorubigd <fzerorubigd@gmail.com>
 | 
						|
 * @author    Behrooz Shabani <everplays@gmail.com>
 | 
						|
 * @copyright 2012 (c) ParsPooyesh Co
 | 
						|
 * @license   MIT <http://opensource.org/licenses/MIT>
 | 
						|
 * @version   Release: @package_version@
 | 
						|
 * @link      http://xamin.ir
 | 
						|
 */
 | 
						|
namespace Handlebars;
 | 
						|
 | 
						|
class Context
 | 
						|
{
 | 
						|
    /**
 | 
						|
     * @var array stack for context only top stack is available
 | 
						|
     */ 
 | 
						|
    protected $stack = array();
 | 
						|
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var index stack for sections
 | 
						|
     */
 | 
						|
    protected $index = array();
 | 
						|
 | 
						|
    /**
 | 
						|
     * @var key stack for objects
 | 
						|
     */
 | 
						|
    protected $key = array();
 | 
						|
 | 
						|
    /**
 | 
						|
     * Mustache rendering Context constructor.
 | 
						|
     *
 | 
						|
     * @param mixed $context Default rendering context (default: null)
 | 
						|
     */
 | 
						|
    public function __construct($context = null)
 | 
						|
    {
 | 
						|
        if ($context !== null) {
 | 
						|
            $this->stack = array($context);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push a new Context frame onto the stack.
 | 
						|
     *
 | 
						|
     * @param mixed $value Object or array to use for context
 | 
						|
     *
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function push($value)
 | 
						|
    {
 | 
						|
        array_push($this->stack, $value);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push an Index onto the index stack
 | 
						|
     *
 | 
						|
     * @param int $index Index of the current section item.
 | 
						|
     */
 | 
						|
    public function pushIndex($index)
 | 
						|
    {
 | 
						|
        array_push($this->index, $index);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push a Key onto the key stack
 | 
						|
     *
 | 
						|
     * @param string $key Key of the current object property.
 | 
						|
     */
 | 
						|
    public function pushKey($key)
 | 
						|
    {
 | 
						|
        array_push($this->key, $key);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Pop the last Context frame from the stack.
 | 
						|
     *
 | 
						|
     * @return mixed Last Context frame (object or array)
 | 
						|
     */
 | 
						|
    public function pop()
 | 
						|
    {
 | 
						|
        return array_pop($this->stack);
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    /**
 | 
						|
     * Pop the last index from the stack.
 | 
						|
     *
 | 
						|
     * @return int Last index
 | 
						|
     */
 | 
						|
    public function popIndex()
 | 
						|
    {
 | 
						|
        return array_pop($this->index);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Pop the last key from the stack.
 | 
						|
     *
 | 
						|
     * @return string Last key
 | 
						|
     */
 | 
						|
    public function popKey()
 | 
						|
    {
 | 
						|
        return array_pop($this->key);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get the last Context frame.
 | 
						|
     *
 | 
						|
     * @return mixed Last Context frame (object or array)
 | 
						|
     */
 | 
						|
    public function last()
 | 
						|
    {
 | 
						|
        return end($this->stack);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get the index of current section item.
 | 
						|
     *
 | 
						|
     * @return mixed Last index
 | 
						|
     */
 | 
						|
    public function lastIndex()
 | 
						|
    {
 | 
						|
        return end($this->index);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get the key of current object property.
 | 
						|
     *
 | 
						|
     * @return mixed Last key
 | 
						|
     */
 | 
						|
    public function lastKey()
 | 
						|
    {
 | 
						|
        return end($this->key);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Change the current context to one of current context members
 | 
						|
     *
 | 
						|
     * @param string $variableName name of variable or a callable on current context
 | 
						|
     *
 | 
						|
     * @return mixed actual value
 | 
						|
     */
 | 
						|
    public function with($variableName)
 | 
						|
    {
 | 
						|
        $value = $this->get($variableName);
 | 
						|
        $this->push($value);
 | 
						|
        return $value;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Get a avariable from current context
 | 
						|
     * Supported types : 
 | 
						|
     * variable , ../variable , variable.variable , .
 | 
						|
     * 
 | 
						|
     * @param string  $variableName variavle name to get from current context
 | 
						|
     * @param boolean $strict       strict search? if not found then throw exception
 | 
						|
     *
 | 
						|
     * @return mixed
 | 
						|
     * @throw InvalidArgumentException in strict mode and variable not found
 | 
						|
     */
 | 
						|
    public function get($variableName, $strict = false)
 | 
						|
    {
 | 
						|
        //Need to clean up
 | 
						|
        $variableName = trim($variableName);
 | 
						|
        $level = 0;
 | 
						|
        while (substr($variableName, 0, 3) == '../') {
 | 
						|
            $variableName = trim(substr($variableName, 3));
 | 
						|
            $level++;
 | 
						|
        }
 | 
						|
        if (count($this->stack) < $level) {
 | 
						|
            if ($strict) {
 | 
						|
                throw new \InvalidArgumentException('can not find variable in context');
 | 
						|
            }                
 | 
						|
            return '';
 | 
						|
        }
 | 
						|
        end($this->stack);
 | 
						|
        while ($level) {
 | 
						|
            prev($this->stack);
 | 
						|
            $level--;
 | 
						|
        }
 | 
						|
        $current = current($this->stack);
 | 
						|
        if (!$variableName) {
 | 
						|
            if ($strict) {
 | 
						|
                throw new \InvalidArgumentException('can not find variable in context');
 | 
						|
            }                
 | 
						|
            return '';
 | 
						|
        } elseif ($variableName == '.' || $variableName == 'this') {
 | 
						|
            return $current;
 | 
						|
        } else {
 | 
						|
            $chunks = explode('.', $variableName);
 | 
						|
            foreach ($chunks as $chunk) {
 | 
						|
                if (is_string($current) and $current == '') {
 | 
						|
                    return $current;
 | 
						|
                }
 | 
						|
                $current = $this->_findVariableInContext($current, $chunk, $strict);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        return $current;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Check if $variable->$inside is available
 | 
						|
     *
 | 
						|
     * @param mixed   $variable variable to check
 | 
						|
     * @param string  $inside   property/method to check
 | 
						|
     * @param boolean $strict   strict search? if not found then throw exception
 | 
						|
     *
 | 
						|
     * @return boolean true if exist
 | 
						|
     * @throw InvalidArgumentException in strict mode and variable not found
 | 
						|
     */ 
 | 
						|
    private function _findVariableInContext($variable, $inside, $strict = false)
 | 
						|
    {
 | 
						|
        $value = '';
 | 
						|
        if ( empty( $inside ) || ( $inside == 'this' ) ) {
 | 
						|
            return $variable;
 | 
						|
        } elseif (is_array($variable)) {
 | 
						|
            if (isset($variable[$inside])) {
 | 
						|
                $value = $variable[$inside];
 | 
						|
            }
 | 
						|
        } elseif (is_object($variable)) {
 | 
						|
            if (isset($variable->$inside)) {
 | 
						|
                $value = $variable->$inside;
 | 
						|
            } elseif (is_callable(array($variable, $inside))) {
 | 
						|
                $value = call_user_func(array($variable, $inside));
 | 
						|
            }                
 | 
						|
        } elseif ($inside === '.') {
 | 
						|
            $value = $variable;
 | 
						|
        } elseif ($strict) {
 | 
						|
            throw new \InvalidArgumentException('can not find variable in context');
 | 
						|
        }
 | 
						|
        return $value;
 | 
						|
    }
 | 
						|
}    
 |