Вход Регистрация
Файл: libs/fenom/Fenom.php
Строк: 822
<?php
/*
 * This file is part of Fenom.
 *
 * (c) 2013 Ivan Shalganov
 *
 * For the full copyright and license information, please view the license.md
 * file that was distributed with this source code.
 */
use FenomProviderInterface;
use 
FenomTemplate;

/**
 * Fenom Template Engine
 *
 * @author     Ivan Shalganov <a.cobest@gmail.com>
 */
class Fenom
{
    const 
VERSION '2.0';
    
/* Actions */
    
const INLINE_COMPILER 1;
    const 
BLOCK_COMPILER  5;
    const 
INLINE_FUNCTION 2;
    const 
BLOCK_FUNCTION  7;

    
/* Options */
    
const DENY_ACCESSOR     0x8;
    const 
DENY_METHODS      0x10;
    const 
DENY_NATIVE_FUNCS 0x20;
    const 
FORCE_INCLUDE     0x40;
    const 
AUTO_RELOAD       0x80;
    const 
FORCE_COMPILE     0x100;
    const 
AUTO_ESCAPE       0x200;
    const 
DISABLE_CACHE     0x400;
    const 
FORCE_VERIFY      0x800;
    const 
AUTO_TRIM         0x1000// reserved
    
const DENY_STATICS      0x2000;
    const 
AUTO_STRIP        0x4000;

    
/* Default parsers */
    
const DEFAULT_CLOSE_COMPILER 'FenomCompiler::stdClose';
    const 
DEFAULT_FUNC_PARSER    'FenomCompiler::stdFuncParser';
    const 
DEFAULT_FUNC_OPEN      'FenomCompiler::stdFuncOpen';
    const 
DEFAULT_FUNC_CLOSE     'FenomCompiler::stdFuncClose';
    const 
SMART_FUNC_PARSER      'FenomCompiler::smartFuncParser';

    const 
MAX_MACRO_RECURSIVE 32;

    
/**
     * @var int[] of possible options, as associative array
     * @see setOptions
     */
    
private static $_options_list = array(
        
"disable_accessor"     => self::DENY_ACCESSOR,
        
"disable_methods"      => self::DENY_METHODS,
        
"disable_native_funcs" => self::DENY_NATIVE_FUNCS,
        
"disable_cache"        => self::DISABLE_CACHE,
        
"force_compile"        => self::FORCE_COMPILE,
        
"auto_reload"          => self::AUTO_RELOAD,
        
"force_include"        => self::FORCE_INCLUDE,
        
"auto_escape"          => self::AUTO_ESCAPE,
        
"force_verify"         => self::FORCE_VERIFY,
        
"auto_trim"            => self::AUTO_TRIM,
        
"disable_statics"      => self::DENY_STATICS,
        
"strip"                => self::AUTO_STRIP,
    );

    
/**
     * @var callable[]
     */
    
public $pre_filters = array();

    
/**
     * @var callable[]
     */
    
public $filters = array();

    
/**
     * @var callable[]
     */
    
public $tag_filters = array();

    
/**
     * @var callable[]
     */
    
public $post_filters = array();

    
/**
     * @var FenomRender[] Templates storage
     */
    
protected $_storage = array();

    
/**
     * @var string compile directory
     */
    
protected $_compile_dir "/tmp";

    
/**
     * @var string[] compile directory for custom provider
     */
    
protected $_compiles = array();

    
/**
     * @var int masked options
     */
    
protected $_options 0;

    
/**
     * @var ProviderInterface
     */
    
private $_provider;
    
/**
     * @var FenomProviderInterface[]
     */
    
protected $_providers = array();

    
/**
     * @var string[] list of modifiers [modifier_name => callable]
     */
    
protected $_modifiers = array(
        
"upper"       => 'strtoupper',
        
"up"          => 'strtoupper',
        
"lower"       => 'strtolower',
        
"low"         => 'strtolower',
        
"date_format" => 'FenomModifier::dateFormat',
        
"date"        => 'FenomModifier::date',
        
"truncate"    => 'FenomModifier::truncate',
        
"escape"      => 'FenomModifier::escape',
        
"e"           => 'FenomModifier::escape'// alias of escape
        
"unescape"    => 'FenomModifier::unescape',
        
"strip"       => 'FenomModifier::strip',
        
"length"      => 'FenomModifier::length',
        
"iterable"    => 'FenomModifier::isIterable',
        
"replace"     => 'FenomModifier::replace',
        
"ereplace"    => 'FenomModifier::ereplace',
        
"match"       => 'FenomModifier::match',
        
"ematch"      => 'FenomModifier::ematch',
        
"split"       => 'FenomModifier::split',
        
"esplit"      => 'FenomModifier::esplit',
        
"join"        => 'FenomModifier::join',
        
"in"          => 'FenomModifier::in',
    );

    
/**
     * @var array of allowed PHP functions
     */
    
protected $_allowed_funcs = array(
        
"count"       => 1,
        
"is_string"   => 1,
        
"is_array"    => 1,
        
"is_numeric"  => 1,
        
"is_int"      => 1,
        
'constant'    => 1,
        
"is_object"   => 1,
        
"strtotime"   => 1,
        
"gettype"     => 1,
        
"is_double"   => 1,
        
"json_encode" => 1,
        
"json_decode" => 1,
        
"ip2long"     => 1,
        
"long2ip"     => 1,
        
"strip_tags"  => 1,
        
"nl2br"       => 1,
        
"explode"     => 1,
        
"implode"     => 1
    
);

    
/**
     * @var array[] of compilers and functions
     */
    
protected $_actions = array(
        
'foreach'    => array( // {foreach ...} {break} {continue} {foreachelse} {/foreach}
            
'type'       => self::BLOCK_COMPILER,
            
'open'       => 'FenomCompiler::foreachOpen',
            
'close'      => 'FenomCompiler::foreachClose',
            
'tags'       => array(
                
'foreachelse' => 'FenomCompiler::foreachElse',
                
'break'       => 'FenomCompiler::tagBreak',
                
'continue'    => 'FenomCompiler::tagContinue',
            ),
            
'float_tags' => array('break' => 1'continue' => 1)
        ),
        
'if'         => array( // {if ...} {elseif ...} {else} {/if}
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::ifOpen',
            
'close' => 'FenomCompiler::stdClose',
            
'tags'  => array(
                
'elseif' => 'FenomCompiler::tagElseIf',
                
'else'   => 'FenomCompiler::tagElse'
            
)
        ),
        
'switch'     => array( // {switch ...} {case ..., ...}  {default} {/switch}
            
'type'       => self::BLOCK_COMPILER,
            
'open'       => 'FenomCompiler::switchOpen',
            
'close'      => 'FenomCompiler::switchClose',
            
'tags'       => array(
                
'case'    => 'FenomCompiler::tagCase',
                
'default' => 'FenomCompiler::tagDefault'
            
),
            
'float_tags' => array('break' => 1)
        ),
        
'for'        => array( // {for ...} {break} {continue} {/for}
            
'type'       => self::BLOCK_COMPILER,
            
'open'       => 'FenomCompiler::forOpen',
            
'close'      => 'FenomCompiler::forClose',
            
'tags'       => array(
                
'forelse'  => 'FenomCompiler::forElse',
                
'break'    => 'FenomCompiler::tagBreak',
                
'continue' => 'FenomCompiler::tagContinue',
            ),
            
'float_tags' => array('break' => 1'continue' => 1)
        ),
        
'while'      => array( // {while ...} {break} {continue} {/while}
            
'type'       => self::BLOCK_COMPILER,
            
'open'       => 'FenomCompiler::whileOpen',
            
'close'      => 'FenomCompiler::stdClose',
            
'tags'       => array(
                
'break'    => 'FenomCompiler::tagBreak',
                
'continue' => 'FenomCompiler::tagContinue',
            ),
            
'float_tags' => array('break' => 1'continue' => 1)
        ),
        
'include'    => array( // {include ...}
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagInclude'
        
),
        
'insert'     => array( // {include ...}
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagInsert'
        
),
        
'var'       => array( // {var ...}
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::setOpen',
            
'close' => 'FenomCompiler::setClose'
        
),
        
'set'       => array( // {set ...}
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::setOpen',
            
'close' => 'FenomCompiler::setClose'
        
),
        
'add'       => array( // {add ...}
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::setOpen',
            
'close' => 'FenomCompiler::setClose'
        
),
        
'block'      => array( // {block ...} {parent} {/block}
            
'type'       => self::BLOCK_COMPILER,
            
'open'       => 'FenomCompiler::tagBlockOpen',
            
'close'      => 'FenomCompiler::tagBlockClose',
            
'tags'       => array('parent' => 'FenomCompiler::tagParent'),
            
'float_tags' => array('parent' => 1)
        ),
        
'extends'    => array( // {extends ...}
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagExtends'
        
),
        
'use'        => array( // {use}
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagUse'
        
),
        
'filter'     => array( // {filter} ... {/filter}
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::filterOpen',
            
'close' => 'FenomCompiler::filterClose'
        
),
        
'macro'      => array(
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::macroOpen',
            
'close' => 'FenomCompiler::macroClose'
        
),
        
'import'     => array(
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagImport'
        
),
        
'cycle'      => array(
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagCycle'
        
),
        
'raw'        => array(
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagRaw'
        
),
        
'autoescape' => array( // deprecated
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::escapeOpen',
            
'close' => 'FenomCompiler::nope'
        
),
        
'escape' => array(
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::escapeOpen',
            
'close' => 'FenomCompiler::nope'
        
),
        
'strip' => array(
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::stripOpen',
            
'close' => 'FenomCompiler::nope'
        
),
        
'ignore' => array(
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => 'FenomCompiler::ignoreOpen',
            
'close' => 'FenomCompiler::nope'
        
),
        
'unset'  => array(
            
'type'   => self::INLINE_COMPILER,
            
'parser' => 'FenomCompiler::tagUnset'
        
)
    );

    
/**
     * List of tests
     * @see https://github.com/bzick/fenom/blob/develop/docs/operators.md#test-operator
     * @var array
     */
    
protected $_tests = array(
        
'integer'  => 'is_int(%s)',
        
'int'      => 'is_int(%s)',
        
'float'    => 'is_float(%s)',
        
'double'   => 'is_float(%s)',
        
'decimal'  => 'is_float(%s)',
        
'string'   => 'is_string(%s)',
        
'bool'     => 'is_bool(%s)',
        
'boolean'  => 'is_bool(%s)',
        
'number'   => 'is_numeric(%s)',
        
'numeric'  => 'is_numeric(%s)',
        
'scalar'   => 'is_scalar(%s)',
        
'object'   => 'is_object(%s)',
        
'callable' => 'is_callable(%s)',
        
'callback' => 'is_callable(%s)',
        
'array'    => 'is_array(%s)',
        
'iterable' => 'FenomModifier::isIterable(%s)',
        
'const'    => 'defined(%s)',
        
'template' => '$tpl->getStorage()->templateExists(%s)',
        
'empty'    => 'empty(%s)',
        
'set'      => 'isset(%s)',
        
'_empty'   => '!%s'// for none variable
        
'_set'     => '(%s !== null)'// for none variable
        
'odd'      => '(%s & 1)',
        
'even'     => '!(%s %% 2)',
        
'third'    => '!(%s %% 3)'
    
);

    
/**
     * Just factory
     *
     * @param string|FenomProviderInterface $source path to templates or custom provider
     * @param string $compile_dir path to compiled files
     * @param int|array $options
     * @throws InvalidArgumentException
     * @return Fenom
     */
    
public static function factory($source$compile_dir '/tmp'$options 0)
    {
        if (
is_string($source)) {
            
$provider = new FenomProvider($source);
        } elseif (
$source instanceof ProviderInterface) {
            
$provider $source;
        } else {
            throw new 
InvalidArgumentException("Source must be a valid path or provider object");
        }
        
$fenom = new static($provider);
        
/* @var Fenom $fenom */
        
$fenom->setCompileDir($compile_dir);
        if (
$options) {
            
$fenom->setOptions($options);
        }
        return 
$fenom;
    }

    
/**
     * @param FenomProviderInterface $provider
     */
    
public function __construct(FenomProviderInterface $provider)
    {
        
$this->_provider $provider;
    }

    
/**
     * Set compile directory
     *
     * @param string $dir directory to store compiled templates in
     * @throws LogicException
     * @return Fenom
     */
    
public function setCompileDir($dir)
    {
        if (!
is_writable($dir)) {
            throw new 
LogicException("Cache directory $dir is not writable");
        }
        
$this->_compile_dir $dir;
        return 
$this;
    }

    
/**
     *
     * @param callable $cb
     * @return self
     */
    
public function addPreFilter($cb)
    {
        
$this->pre_filters[] = $cb;
        return 
$this;
    }

    public function 
getPreFilters()
    {
        return 
$this->pre_filters;
    }

    
/**
     *
     * @param callable $cb
     * @return self
     */
    
public function addPostFilter($cb)
    {
        
$this->post_filters[] = $cb;
        return 
$this;
    }


    public function 
getPostFilters()
    {
        return 
$this->post_filters;
    }

    
/**
     * @param callable $cb
     * @return self
     */
    
public function addFilter($cb)
    {
        
$this->filters[] = $cb;
        return 
$this;
    }


    public function 
getFilters()
    {
        return 
$this->filters;
    }

    
/**
     * @param callable $cb
     * @return self
     */
    
public function addTagFilter($cb)
    {
        
$this->tag_filters[] = $cb;
        return 
$this;
    }


    public function 
getTagFilters()
    {
        return 
$this->tag_filters;
    }

    
/**
     * Add modifier
     *
     * @param string $modifier the modifier name
     * @param string $callback the modifier callback
     * @return Fenom
     */
    
public function addModifier($modifier$callback)
    {
        
$this->_modifiers[$modifier] = $callback;
        return 
$this;
    }

    
/**
     * Add inline tag compiler
     *
     * @param string $compiler
     * @param callable $parser
     * @return Fenom
     */
    
public function addCompiler($compiler$parser)
    {
        
$this->_actions[$compiler] = array(
            
'type'   => self::INLINE_COMPILER,
            
'parser' => $parser
        
);
        return 
$this;
    }

    
/**
     * @param string $compiler
     * @param string|object $storage
     * @return $this
     */
    
public function addCompilerSmart($compiler$storage)
    {
        if (
method_exists($storage"tag" $compiler)) {
            
$this->_actions[$compiler] = array(
                
'type'   => self::INLINE_COMPILER,
                
'parser' => array($storage"tag" $compiler)
            );
        }
        return 
$this;
    }

    
/**
     * Add block compiler
     *
     * @param string $compiler
     * @param callable $open_parser
     * @param callable|string $close_parser
     * @param array $tags
     * @return Fenom
     */
    
public function addBlockCompiler(
        
$compiler,
        
$open_parser,
        
$close_parser self::DEFAULT_CLOSE_COMPILER,
        array 
$tags = array()
    ) {
        
$this->_actions[$compiler] = array(
            
'type'  => self::BLOCK_COMPILER,
            
'open'  => $open_parser,
            
'close' => $close_parser ? : self::DEFAULT_CLOSE_COMPILER,
            
'tags'  => $tags,
        );
        return 
$this;
    }

    
/**
     * @param $compiler
     * @param $storage
     * @param array $tags
     * @param array $floats
     * @throws LogicException
     * @return Fenom
     */
    
public function addBlockCompilerSmart($compiler$storage, array $tags, array $floats = array())
    {
        
$c = array(
            
'type'       => self::BLOCK_COMPILER,
            
"tags"       => array(),
            
"float_tags" => array()
        );
        if (
method_exists($storage$compiler "Open")) {
            
$c["open"] = array($storage$compiler "Open");
        } else {
            throw new 
LogicException("Open compiler {$compiler}Open not found");
        }
        if (
method_exists($storage$compiler "Close")) {
            
$c["close"] = array($storage$compiler "Close");
        } else {
            throw new 
LogicException("Close compiler {$compiler}Close not found");
        }
        foreach (
$tags as $tag) {
            if (
method_exists($storage"tag" $tag)) {
                
$c["tags"][$tag] = array($storage"tag" $tag);
                if (
$floats && in_array($tag$floats)) {
                    
$c['float_tags'][$tag] = 1;
                }
            } else {
                throw new 
LogicException("Tag compiler $tag (tag{$compiler}) not found");
            }
        }
        
$this->_actions[$compiler] = $c;
        return 
$this;
    }

    
/**
     * @param string $function
     * @param callable $callback
     * @param callable|string $parser
     * @return Fenom
     */
    
public function addFunction($function$callback$parser self::DEFAULT_FUNC_PARSER)
    {
        
$this->_actions[$function] = array(
            
'type'     => self::INLINE_FUNCTION,
            
'parser'   => $parser,
            
'function' => $callback,
        );
        return 
$this;
    }

    
/**
     * @param string $function
     * @param callable $callback
     * @return Fenom
     */
    
public function addFunctionSmart($function$callback)
    {
        
$this->_actions[$function] = array(
            
'type'     => self::INLINE_FUNCTION,
            
'parser'   => self::SMART_FUNC_PARSER,
            
'function' => $callback,
        );
        return 
$this;
    }

    
/**
     * @param string $function
     * @param callable $callback
     * @param callable|string $parser_open
     * @param callable|string $parser_close
     * @return Fenom
     */
    
public function addBlockFunction(
        
$function,
        
$callback,
        
$parser_open self::DEFAULT_FUNC_OPEN,
        
$parser_close self::DEFAULT_FUNC_CLOSE
    
) {
        
$this->_actions[$function] = array(
            
'type'     => self::BLOCK_FUNCTION,
            
'open'     => $parser_open,
            
'close'    => $parser_close,
            
'function' => $callback,
        );
        return 
$this;
    }

    
/**
     * @param array $funcs
     * @return Fenom
     */
    
public function addAllowedFunctions(array $funcs)
    {
        
$this->_allowed_funcs $this->_allowed_funcs array_flip($funcs);
        return 
$this;
    }

    
/**
     * Add custom test
     * @param string $name test name
     * @param string $code test PHP code. Code may contains placeholder %s, which will be replaced by test-value. For example: is_callable(%s)
     */
    
public function addTest($name$code)
    {
        
$this->_tests[$name] = $code;
    }

    
/**
     * Get test code by name
     * @param string $name
     * @return string|bool
     */
    
public function getTest($name)
    {
        return isset(
$this->_tests[$name]) ? $this->_tests[$name] : false;
    }

    
/**
     * Return modifier function
     *
     * @param string $modifier
     * @param FenomTemplate $template
     * @return mixed
     */
    
public function getModifier($modifierTemplate $template null)
    {
        if (isset(
$this->_modifiers[$modifier])) {
            return 
$this->_modifiers[$modifier];
        } elseif (
$this->isAllowedFunction($modifier)) {
            return 
$modifier;
        } else {
            return 
$this->_loadModifier($modifier$template);
        }
    }

    
/**
     * Modifier autoloader
     * @param string $modifier
     * @param FenomTemplate $template
     * @return bool
     */
    
protected function _loadModifier($modifier$template)
    {
        return 
false;
    }

    
/**
     * Returns tag info
     *
     * @param string $tag
     * @param FenomTemplate $template
     * @return string|bool
     */
    
public function getTag($tagTemplate $template null)
    {
        if (isset(
$this->_actions[$tag])) {
            return 
$this->_actions[$tag];
        } else {
            return 
$this->_loadTag($tag$template);
        }
    }

    
/**
     * Tags autoloader
     * @param string $tag
     * @param FenomTemplate $template
     * @return bool
     */
    
protected function _loadTag($tag$template)
    {
        return 
false;
    }

    
/**
     * @param string $function
     * @return bool
     */
    
public function isAllowedFunction($function)
    {
        if (
$this->_options self::DENY_NATIVE_FUNCS) {
            return isset(
$this->_allowed_funcs[$function]);
        } else {
            return 
is_callable($function);
        }
    }

    
/**
     * @param string $tag
     * @return array
     */
    
public function getTagOwners($tag)
    {
        
$tags = array();
        foreach (
$this->_actions as $owner => $params) {
            if (isset(
$params["tags"][$tag])) {
                
$tags[] = $owner;
            }
        }
        return 
$tags;
    }

    
/**
     * Add source template provider by scheme
     *
     * @param string $scm scheme name
     * @param FenomProviderInterface $provider provider object
     * @param string $compile_path
     * @return $this
     */
    
public function addProvider($scmFenomProviderInterface $provider$compile_path null)
    {
        
$this->_providers[$scm] = $provider;
        if (
$compile_path) {
            
$this->_compiles[$scm] = $compile_path;
        }
        return 
$this;
    }

    
/**
     * Set options
     * @param int|array $options
     * @return $this
     */
    
public function setOptions($options)
    {
        if (
is_array($options)) {
            
$options self::_makeMask($optionsself::$_options_list$this->_options);
        }
        
$this->_storage = array();
        
$this->_options $options;
        return 
$this;
    }

    
/**
     * Get options as bits
     * @return int
     */
    
public function getOptions()
    {
        return 
$this->_options;
    }

    
/**
     * @param bool|string $scm
     * @return FenomProviderInterface
     * @throws InvalidArgumentException
     */
    
public function getProvider($scm false)
    {
        if (
$scm) {
            if (isset(
$this->_providers[$scm])) {
                return 
$this->_providers[$scm];
            } else {
                throw new 
InvalidArgumentException("Provider for '$scm' not found");
            }
        } else {
            return 
$this->_provider;
        }
    }

    
/**
     * Return empty template
     *
     * @return FenomTemplate
     */
    
public function getRawTemplate()
    {
        return new 
Template($this$this->_options);
    }

    
/**
     * Execute template and write result into stdout
     *
     * @param string $template name of template
     * @param array $vars array of data for template
     * @return FenomRender
     */
    
public function display($template, array $vars = array())
    {
        return 
$this->getTemplate($template)->display($vars);
    }

    
/**
     *
     * @param string $template name of template
     * @param array $vars array of data for template
     * @return mixed
     */
    
public function fetch($template, array $vars = array())
    {
        return 
$this->getTemplate($template)->fetch($vars);
    }

    
/**
     * Creates pipe-line of template's data to callback
     * @note Method not works correctly in old PHP 5.3.*
     * @param string $template name of the template
     * @param callable $callback template's data handler
     * @param array $vars
     * @param float $chunk amount of bytes of chunk
     * @return array
     */
    
public function pipe($template$callback, array $vars = array(), $chunk 1e6)
    {
        
ob_start($callback$chunkPHP_OUTPUT_HANDLER_STDFLAGS);
        
$data $this->getTemplate($template)->display($vars);
        
ob_end_flush();
        return 
$data;
    }

    
/**
     * Get template by name
     *
     * @param string $template template name with schema
     * @param int $options additional options and flags
     * @return FenomTemplate
     */
    
public function getTemplate($template$options 0)
    {
        
$options |= $this->_options;
        if (
is_array($template)) {
            
$key dechex($options) . "@" implode(","$template);
        } else {
            
$key dechex($options) . "@" $template;
        }
        if (isset(
$this->_storage[$key])) {
            
/** @var FenomTemplate $tpl */
            
$tpl $this->_storage[$key];
            if ((
$this->_options self::AUTO_RELOAD) && !$tpl->isValid()) {
                return 
$this->_storage[$key] = $this->compile($templatetrue$options);
            } else {
                return 
$tpl;
            }
        } elseif (
$this->_options self::FORCE_COMPILE) {

            return 
$this->compile($template$this->_options self::DISABLE_CACHE & ~self::FORCE_COMPILE$options);
        } else {
            return 
$this->_storage[$key] = $this->_load($template$options);
        }
    }

    
/**
     * Check if template exists
     * @param string $template
     * @return bool
     */
    
public function templateExists($template)
    {
        if (
$provider strstr($template":"true)) {
            if (isset(
$this->_providers[$provider])) {
                return 
$this->_providers[$provider]->templateExists(substr($templatestrlen($provider) + 1));
            }
        } else {
            return 
$this->_provider->templateExists($template);
        }
        return 
false;
    }

    
/**
     * Load template from cache or create cache if it doesn't exists.
     *
     * @param string $template
     * @param int $opts
     * @return FenomRender
     */
    
protected function _load($template$opts)
    {
        
$file_name $this->_getCacheName($template$opts);
        if (
is_file($this->_compile_dir "/" $file_name)) {
            
$fenom $this// used in template
            
$_tpl  = include($this->_compile_dir "/" $file_name);
            
/* @var FenomRender $_tpl */
            
if (!($this->_options self::AUTO_RELOAD) || ($this->_options self::AUTO_RELOAD) && $_tpl->isValid()) {
                return 
$_tpl;
            }
        }
        return 
$this->compile($templatetrue$opts);
    }

    
/**
     * Generate unique name of compiled template
     *
     * @param string $tpl
     * @param int $options
     * @return string
     */
    
private function _getCacheName($tpl$options)
    {
        if (
is_array($tpl)) {
            
$hash implode("."$tpl) . ":" $options;
            foreach (
$tpl as &$t) {
                
$t str_replace(":""_"basename($t));
            }
            return 
implode("~"$tpl) . "." sprintf("%x.%x.php"crc32($hash), strlen($hash));
        } else {
            
$hash $tpl ":" $options;
            return 
sprintf("%s.%x.%x.php"str_replace(":""_"basename($tpl)), crc32($hash), strlen($hash));
        }
    }

    
/**
     * Compile and save template
     *
     * @param string|array $tpl
     * @param bool $store store template on disk
     * @param int $options
     * @throws RuntimeException
     * @return FenomTemplate
     */
    
public function compile($tpl$store true$options 0)
    {
        
$options $this->_options $options;
        if (
is_string($tpl)) {
            
$template $this->getRawTemplate()->load($tpl);
        } else {
            
$template $this->getRawTemplate()->load($tpl[0], false);
            unset(
$tpl[0]);
            foreach (
$tpl as $t) {
                
$template->extend($t);
            }
        }
        if (
$store) {
            
$cache   $this->_getCacheName($tpl$options);
            
$tpl_tmp tempnam($this->_compile_dir$cache);
            
$tpl_fp  fopen($tpl_tmp"w");
            if (!
$tpl_fp) {
                throw new 
RuntimeException("Can't to open temporary file $tpl_tmp. Directory " $this->_compile_dir " is writable?");
            }
            
fwrite($tpl_fp$template->getTemplateCode());
            
fclose($tpl_fp);
            
$file_name $this->_compile_dir "/" $cache;
            if (!
rename($tpl_tmp$file_name)) {
                throw new 
RuntimeException("Can't to move $tpl_tmp to $file_name");
            }
        }
        return 
$template;
    }

    
/**
     * Flush internal memory template cache
     */
    
public function flush()
    {
        
$this->_storage = array();
    }

    
/**
     * Remove all compiled templates
     */
    
public function clearAllCompiles()
    {
        
FenomProvider::clean($this->_compile_dir);
    }

    
/**
     * Compile code to template
     *
     * @param string $code
     * @param string $name
     * @return FenomTemplate
     */
    
public function compileCode($code$name 'Runtime compile')
    {
        return 
$this->getRawTemplate()->source($name$code);
    }


    
/**
     * Create bit-mask from associative array use fully associative array possible keys with bit values
     * @static
     * @param array $values custom assoc array, ["a" => true, "b" => false]
     * @param array $options possible values, ["a" => 0b001, "b" => 0b010, "c" => 0b100]
     * @param int $mask the initial value of the mask
     * @return int result, ( $mask | a ) & ~b
     * @throws RuntimeException if key from custom assoc doesn't exists into possible values
     */
    
private static function _makeMask(array $values, array $options$mask 0)
    {
        foreach (
$values as $key => $value) {
            if (isset(
$options[$key])) {
                if (
$value) {
                    
$mask |= $options[$key];
                } else {
                    
$mask &= ~$options[$key];
                }
            } else {
                throw new 
RuntimeException("Undefined parameter $value");
            }
        }
        return 
$mask;
    }

    
/**
     * Register PSR-0 autoload
     * @param string $dir custom directory for autoloading, if NULL вЂ” autoload itself
     * @return bool
     */
    
public static function registerAutoload($dir null)
    {
        if (!
$dir) {
            
$dir __DIR__;
        }
        return 
spl_autoload_register(
            function (
$classname) use ($dir) {
                
$file $dir DIRECTORY_SEPARATOR str_replace('\', DIRECTORY_SEPARATOR, $classname) . '.php';
                if (is_file($file)) {
                    require_once $file;
                }
            }
        );
    }
}
Онлайн: 1
Реклама