| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318 | <?php// +----------------------------------------------------------------------// | ThinkPHP [ WE CAN DO IT JUST THINK ]// +----------------------------------------------------------------------// | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.// +----------------------------------------------------------------------// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )// +----------------------------------------------------------------------// | Author: liu21st <liu21st@gmail.com>// +----------------------------------------------------------------------namespace think;use think\exception\TemplateNotFoundException;/** * ThinkPHP分离出来的模板引擎 * 支持XML标签和普通标签的模板解析 * 编译型模板引擎 支持动态缓存 */class Template{    protected $app;    /**     * 模板变量     * @var array     */    protected $data = [];    /**     * 模板配置参数     * @var array     */    protected $config = [        'view_path'          => '', // 模板路径        'view_base'          => '',        'view_suffix'        => 'html', // 默认模板文件后缀        'view_depr'          => DIRECTORY_SEPARATOR,        'cache_suffix'       => 'php', // 默认模板缓存后缀        'tpl_deny_func_list' => 'echo,exit', // 模板引擎禁用函数        'tpl_deny_php'       => false, // 默认模板引擎是否禁用PHP原生代码        'tpl_begin'          => '{', // 模板引擎普通标签开始标记        'tpl_end'            => '}', // 模板引擎普通标签结束标记        'strip_space'        => false, // 是否去除模板文件里面的html空格与换行        'tpl_cache'          => true, // 是否开启模板编译缓存,设为false则每次都会重新编译        'compile_type'       => 'file', // 模板编译类型        'cache_prefix'       => '', // 模板缓存前缀标识,可以动态改变        'cache_time'         => 0, // 模板缓存有效期 0 为永久,(以数字为值,单位:秒)        'layout_on'          => false, // 布局模板开关        'layout_name'        => 'layout', // 布局模板入口文件        'layout_item'        => '{__CONTENT__}', // 布局模板的内容替换标识        'taglib_begin'       => '{', // 标签库标签开始标记        'taglib_end'         => '}', // 标签库标签结束标记        'taglib_load'        => true, // 是否使用内置标签库之外的其它标签库,默认自动检测        'taglib_build_in'    => 'cx', // 内置标签库名称(标签使用不必指定标签库名称),以逗号分隔 注意解析顺序        'taglib_pre_load'    => '', // 需要额外加载的标签库(须指定标签库名称),多个以逗号分隔        'display_cache'      => false, // 模板渲染缓存        'cache_id'           => '', // 模板缓存ID        'tpl_replace_string' => [],        'tpl_var_identify'   => 'array', // .语法变量识别,array|object|'', 为空时自动识别        'default_filter'     => 'htmlentities', // 默认过滤方法 用于普通标签输出    ];    /**     * 保留内容信息     * @var array     */    private $literal = [];    /**     * 模板包含信息     * @var array     */    private $includeFile = [];    /**     * 模板存储对象     * @var object     */    protected $storage;    /**     * 架构函数     * @access public     * @param  array $config     */    public function __construct(App $app, array $config = [])    {        $this->app                  = $app;        $this->config['cache_path'] = $app->getRuntimePath() . 'temp/';        $this->config               = array_merge($this->config, $config);        $this->config['taglib_begin_origin'] = $this->config['taglib_begin'];        $this->config['taglib_end_origin']   = $this->config['taglib_end'];        $this->config['taglib_begin'] = preg_quote($this->config['taglib_begin'], '/');        $this->config['taglib_end']   = preg_quote($this->config['taglib_end'], '/');        $this->config['tpl_begin']    = preg_quote($this->config['tpl_begin'], '/');        $this->config['tpl_end']      = preg_quote($this->config['tpl_end'], '/');        // 初始化模板编译存储器        $type = $this->config['compile_type'] ? $this->config['compile_type'] : 'File';        $this->storage = Loader::factory($type, '\\think\\template\\driver\\', null);    }    public static function __make(Config $config)    {        return new static($config->pull('template'));    }    /**     * 模板变量赋值     * @access public     * @param  mixed $name     * @param  mixed $value     * @return void     */    public function assign($name, $value = '')    {        if (is_array($name)) {            $this->data = array_merge($this->data, $name);        } else {            $this->data[$name] = $value;        }    }    /**     * 模板引擎参数赋值     * @access public     * @param  mixed $name     * @param  mixed $value     */    public function __set($name, $value)    {        $this->config[$name] = $value;    }    /**     * 模板引擎配置项     * @access public     * @param  array|string $config     * @return void|array     */    public function config($config)    {        if (is_array($config)) {            $this->config = array_merge($this->config, $config);        } elseif (isset($this->config[$config])) {            return $this->config[$config];        }    }    /**     * 模板变量获取     * @access public     * @param  string $name 变量名     * @return mixed     */    public function get($name = '')    {        if ('' == $name) {            return $this->data;        }        $data = $this->data;        foreach (explode('.', $name) as $key => $val) {            if (isset($data[$val])) {                $data = $data[$val];            } else {                $data = null;                break;            }        }        return $data;    }    /**     * 渲染模板文件     * @access public     * @param  string    $template 模板文件     * @param  array     $vars 模板变量     * @param  array     $config 模板参数     * @return void     */    public function fetch($template, $vars = [], $config = [])    {        if ($vars) {            $this->data = $vars;        }        if ($config) {            $this->config($config);        }        $cache = $this->app['cache'];        if (!empty($this->config['cache_id']) && $this->config['display_cache']) {            // 读取渲染缓存            $cacheContent = $cache->get($this->config['cache_id']);            if (false !== $cacheContent) {                echo $cacheContent;                return;            }        }        $template = $this->parseTemplateFile($template);        if ($template) {            $cacheFile = $this->config['cache_path'] . $this->config['cache_prefix'] . md5($this->config['layout_on'] . $this->config['layout_name'] . $template) . '.' . ltrim($this->config['cache_suffix'], '.');            if (!$this->checkCache($cacheFile)) {                // 缓存无效 重新模板编译                $content = file_get_contents($template);                $this->compiler($content, $cacheFile);            }            // 页面缓存            ob_start();            ob_implicit_flush(0);            // 读取编译存储            $this->storage->read($cacheFile, $this->data);            // 获取并清空缓存            $content = ob_get_clean();            if (!empty($this->config['cache_id']) && $this->config['display_cache']) {                // 缓存页面输出                $cache->set($this->config['cache_id'], $content, $this->config['cache_time']);            }            echo $content;        }    }    /**     * 渲染模板内容     * @access public     * @param  string    $content 模板内容     * @param  array     $vars 模板变量     * @param  array     $config 模板参数     * @return void     */    public function display($content, $vars = [], $config = [])    {        if ($vars) {            $this->data = $vars;        }        if ($config) {            $this->config($config);        }        $cacheFile = $this->config['cache_path'] . $this->config['cache_prefix'] . md5($content) . '.' . ltrim($this->config['cache_suffix'], '.');        if (!$this->checkCache($cacheFile)) {            // 缓存无效 模板编译            $this->compiler($content, $cacheFile);        }        // 读取编译存储        $this->storage->read($cacheFile, $this->data);    }    /**     * 设置布局     * @access public     * @param  mixed     $name 布局模板名称 false 则关闭布局     * @param  string    $replace 布局模板内容替换标识     * @return object     */    public function layout($name, $replace = '')    {        if (false === $name) {            // 关闭布局            $this->config['layout_on'] = false;        } else {            // 开启布局            $this->config['layout_on'] = true;            // 名称必须为字符串            if (is_string($name)) {                $this->config['layout_name'] = $name;            }            if (!empty($replace)) {                $this->config['layout_item'] = $replace;            }        }        return $this;    }    /**     * 检查编译缓存是否有效     * 如果无效则需要重新编译     * @access private     * @param  string $cacheFile 缓存文件名     * @return boolean     */    private function checkCache($cacheFile)    {        if (!$this->config['tpl_cache'] || !is_file($cacheFile) || !$handle = @fopen($cacheFile, "r")) {            return false;        }        // 读取第一行        preg_match('/\/\*(.+?)\*\//', fgets($handle), $matches);        if (!isset($matches[1])) {            return false;        }        $includeFile = unserialize($matches[1]);        if (!is_array($includeFile)) {            return false;        }        // 检查模板文件是否有更新        foreach ($includeFile as $path => $time) {            if (is_file($path) && filemtime($path) > $time) {                // 模板文件如果有更新则缓存需要更新                return false;            }        }        // 检查编译存储是否有效        return $this->storage->check($cacheFile, $this->config['cache_time']);    }    /**     * 检查编译缓存是否存在     * @access public     * @param  string $cacheId 缓存的id     * @return boolean     */    public function isCache($cacheId)    {        if ($cacheId && $this->config['display_cache']) {            // 缓存页面输出            return $this->app['cache']->has($cacheId);        }        return false;    }    /**     * 编译模板文件内容     * @access private     * @param  string    $content 模板内容     * @param  string    $cacheFile 缓存文件名     * @return void     */    private function compiler(&$content, $cacheFile)    {        // 判断是否启用布局        if ($this->config['layout_on']) {            if (false !== strpos($content, '{__NOLAYOUT__}')) {                // 可以单独定义不使用布局                $content = str_replace('{__NOLAYOUT__}', '', $content);            } else {                // 读取布局模板                $layoutFile = $this->parseTemplateFile($this->config['layout_name']);                if ($layoutFile) {                    // 替换布局的主体内容                    $content = str_replace($this->config['layout_item'], $content, file_get_contents($layoutFile));                }            }        } else {            $content = str_replace('{__NOLAYOUT__}', '', $content);        }        // 模板解析        $this->parse($content);        if ($this->config['strip_space']) {            /* 去除html空格与换行 */            $find    = ['~>\s+<~', '~>(\s+\n|\r)~'];            $replace = ['><', '>'];            $content = preg_replace($find, $replace, $content);        }        // 优化生成的php代码        $content = preg_replace('/\?>\s*<\?php\s(?!echo\b|\bend)/s', '', $content);        // 模板过滤输出        $replace = $this->config['tpl_replace_string'];        $content = str_replace(array_keys($replace), array_values($replace), $content);        // 添加安全代码及模板引用记录        $content = '<?php /*' . serialize($this->includeFile) . '*/ ?>' . "\n" . $content;        // 编译存储        $this->storage->write($cacheFile, $content);        $this->includeFile = [];    }    /**     * 模板解析入口     * 支持普通标签和TagLib解析 支持自定义标签库     * @access public     * @param  string $content 要解析的模板内容     * @return void     */    public function parse(&$content)    {        // 内容为空不解析        if (empty($content)) {            return;        }        // 替换literal标签内容        $this->parseLiteral($content);        // 解析继承        $this->parseExtend($content);        // 解析布局        $this->parseLayout($content);        // 检查include语法        $this->parseInclude($content);        // 替换包含文件中literal标签内容        $this->parseLiteral($content);        // 检查PHP语法        $this->parsePhp($content);        // 获取需要引入的标签库列表        // 标签库只需要定义一次,允许引入多个一次        // 一般放在文件的最前面        // 格式:<taglib name="html,mytag..." />        // 当TAGLIB_LOAD配置为true时才会进行检测        if ($this->config['taglib_load']) {            $tagLibs = $this->getIncludeTagLib($content);            if (!empty($tagLibs)) {                // 对导入的TagLib进行解析                foreach ($tagLibs as $tagLibName) {                    $this->parseTagLib($tagLibName, $content);                }            }        }        // 预先加载的标签库 无需在每个模板中使用taglib标签加载 但必须使用标签库XML前缀        if ($this->config['taglib_pre_load']) {            $tagLibs = explode(',', $this->config['taglib_pre_load']);            foreach ($tagLibs as $tag) {                $this->parseTagLib($tag, $content);            }        }        // 内置标签库 无需使用taglib标签导入就可以使用 并且不需使用标签库XML前缀        $tagLibs = explode(',', $this->config['taglib_build_in']);        foreach ($tagLibs as $tag) {            $this->parseTagLib($tag, $content, true);        }        // 解析普通模板标签 {$tagName}        $this->parseTag($content);        // 还原被替换的Literal标签        $this->parseLiteral($content, true);    }    /**     * 检查PHP语法     * @access private     * @param  string $content 要解析的模板内容     * @return void     * @throws \think\Exception     */    private function parsePhp(&$content)    {        // 短标签的情况要将<?标签用echo方式输出 否则无法正常输出xml标识        $content = preg_replace('/(<\?(?!php|=|$))/i', '<?php echo \'\\1\'; ?>' . "\n", $content);        // PHP语法检查        if ($this->config['tpl_deny_php'] && false !== strpos($content, '<?php')) {            throw new Exception('not allow php tag');        }    }    /**     * 解析模板中的布局标签     * @access private     * @param  string $content 要解析的模板内容     * @return void     */    private function parseLayout(&$content)    {        // 读取模板中的布局标签        if (preg_match($this->getRegex('layout'), $content, $matches)) {            // 替换Layout标签            $content = str_replace($matches[0], '', $content);            // 解析Layout标签            $array = $this->parseAttr($matches[0]);            if (!$this->config['layout_on'] || $this->config['layout_name'] != $array['name']) {                // 读取布局模板                $layoutFile = $this->parseTemplateFile($array['name']);                if ($layoutFile) {                    $replace = isset($array['replace']) ? $array['replace'] : $this->config['layout_item'];                    // 替换布局的主体内容                    $content = str_replace($replace, $content, file_get_contents($layoutFile));                }            }        } else {            $content = str_replace('{__NOLAYOUT__}', '', $content);        }    }    /**     * 解析模板中的include标签     * @access private     * @param  string $content 要解析的模板内容     * @return void     */    private function parseInclude(&$content)    {        $regex = $this->getRegex('include');        $func  = function ($template) use (&$func, &$regex, &$content) {            if (preg_match_all($regex, $template, $matches, PREG_SET_ORDER)) {                foreach ($matches as $match) {                    $array = $this->parseAttr($match[0]);                    $file  = $array['file'];                    unset($array['file']);                    // 分析模板文件名并读取内容                    $parseStr = $this->parseTemplateName($file);                    foreach ($array as $k => $v) {                        // 以$开头字符串转换成模板变量                        if (0 === strpos($v, '$')) {                            $v = $this->get(substr($v, 1));                        }                        $parseStr = str_replace('[' . $k . ']', $v, $parseStr);                    }                    $content = str_replace($match[0], $parseStr, $content);                    // 再次对包含文件进行模板分析                    $func($parseStr);                }                unset($matches);            }        };        // 替换模板中的include标签        $func($content);    }    /**     * 解析模板中的extend标签     * @access private     * @param  string $content 要解析的模板内容     * @return void     */    private function parseExtend(&$content)    {        $regex  = $this->getRegex('extend');        $array  = $blocks  = $baseBlocks  = [];        $extend = '';        $func = function ($template) use (&$func, &$regex, &$array, &$extend, &$blocks, &$baseBlocks) {            if (preg_match($regex, $template, $matches)) {                if (!isset($array[$matches['name']])) {                    $array[$matches['name']] = 1;                    // 读取继承模板                    $extend = $this->parseTemplateName($matches['name']);                    // 递归检查继承                    $func($extend);                    // 取得block标签内容                    $blocks = array_merge($blocks, $this->parseBlock($template));                    return;                }            } else {                // 取得顶层模板block标签内容                $baseBlocks = $this->parseBlock($template, true);                if (empty($extend)) {                    // 无extend标签但有block标签的情况                    $extend = $template;                }            }        };        $func($content);        if (!empty($extend)) {            if ($baseBlocks) {                $children = [];                foreach ($baseBlocks as $name => $val) {                    $replace = $val['content'];                    if (!empty($children[$name])) {                        // 如果包含有子block标签                        foreach ($children[$name] as $key) {                            $replace = str_replace($baseBlocks[$key]['begin'] . $baseBlocks[$key]['content'] . $baseBlocks[$key]['end'], $blocks[$key]['content'], $replace);                        }                    }                    if (isset($blocks[$name])) {                        // 带有{__block__}表示与所继承模板的相应标签合并,而不是覆盖                        $replace = str_replace(['{__BLOCK__}', '{__block__}'], $replace, $blocks[$name]['content']);                        if (!empty($val['parent'])) {                            // 如果不是最顶层的block标签                            $parent = $val['parent'];                            if (isset($blocks[$parent])) {                                $blocks[$parent]['content'] = str_replace($blocks[$name]['begin'] . $blocks[$name]['content'] . $blocks[$name]['end'], $replace, $blocks[$parent]['content']);                            }                            $blocks[$name]['content'] = $replace;                            $children[$parent][]      = $name;                            continue;                        }                    } elseif (!empty($val['parent'])) {                        // 如果子标签没有被继承则用原值                        $children[$val['parent']][] = $name;                        $blocks[$name]              = $val;                    }                    if (!$val['parent']) {                        // 替换模板中的顶级block标签                        $extend = str_replace($val['begin'] . $val['content'] . $val['end'], $replace, $extend);                    }                }            }            $content = $extend;            unset($blocks, $baseBlocks);        }    }    /**     * 替换页面中的literal标签     * @access private     * @param  string   $content 模板内容     * @param  boolean  $restore 是否为还原     * @return void     */    private function parseLiteral(&$content, $restore = false)    {        $regex = $this->getRegex($restore ? 'restoreliteral' : 'literal');        if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER)) {            if (!$restore) {                $count = count($this->literal);                // 替换literal标签                foreach ($matches as $match) {                    $this->literal[] = substr($match[0], strlen($match[1]), -strlen($match[2]));                    $content         = str_replace($match[0], "<!--###literal{$count}###-->", $content);                    $count++;                }            } else {                // 还原literal标签                foreach ($matches as $match) {                    $content = str_replace($match[0], $this->literal[$match[1]], $content);                }                // 清空literal记录                $this->literal = [];            }            unset($matches);        }    }    /**     * 获取模板中的block标签     * @access private     * @param  string   $content 模板内容     * @param  boolean  $sort 是否排序     * @return array     */    private function parseBlock(&$content, $sort = false)    {        $regex  = $this->getRegex('block');        $result = [];        if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER | PREG_OFFSET_CAPTURE)) {            $right = $keys = [];            foreach ($matches as $match) {                if (empty($match['name'][0])) {                    if (count($right) > 0) {                        $tag    = array_pop($right);                        $start  = $tag['offset'] + strlen($tag['tag']);                        $length = $match[0][1] - $start;                        $result[$tag['name']] = [                            'begin'   => $tag['tag'],                            'content' => substr($content, $start, $length),                            'end'     => $match[0][0],                            'parent'  => count($right) ? end($right)['name'] : '',                        ];                        $keys[$tag['name']] = $match[0][1];                    }                } else {                    // 标签头压入栈                    $right[] = [                        'name'   => $match[2][0],                        'offset' => $match[0][1],                        'tag'    => $match[0][0],                    ];                }            }            unset($right, $matches);            if ($sort) {                // 按block标签结束符在模板中的位置排序                array_multisort($keys, $result);            }        }        return $result;    }    /**     * 搜索模板页面中包含的TagLib库     * 并返回列表     * @access private     * @param  string $content 模板内容     * @return array|null     */    private function getIncludeTagLib(&$content)    {        // 搜索是否有TagLib标签        if (preg_match($this->getRegex('taglib'), $content, $matches)) {            // 替换TagLib标签            $content = str_replace($matches[0], '', $content);            return explode(',', $matches['name']);        }    }    /**     * TagLib库解析     * @access public     * @param  string   $tagLib 要解析的标签库     * @param  string   $content 要解析的模板内容     * @param  boolean  $hide 是否隐藏标签库前缀     * @return void     */    public function parseTagLib($tagLib, &$content, $hide = false)    {        if (false !== strpos($tagLib, '\\')) {            // 支持指定标签库的命名空间            $className = $tagLib;            $tagLib    = substr($tagLib, strrpos($tagLib, '\\') + 1);        } else {            $className = '\\think\\template\\taglib\\' . ucwords($tagLib);        }        $tLib = new $className($this);        $tLib->parseTag($content, $hide ? '' : $tagLib);    }    /**     * 分析标签属性     * @access public     * @param  string   $str 属性字符串     * @param  string   $name 不为空时返回指定的属性名     * @return array     */    public function parseAttr($str, $name = null)    {        $regex = '/\s+(?>(?P<name>[\w-]+)\s*)=(?>\s*)([\"\'])(?P<value>(?:(?!\\2).)*)\\2/is';        $array = [];        if (preg_match_all($regex, $str, $matches, PREG_SET_ORDER)) {            foreach ($matches as $match) {                $array[$match['name']] = $match['value'];            }            unset($matches);        }        if (!empty($name) && isset($array[$name])) {            return $array[$name];        }        return $array;    }    /**     * 模板标签解析     * 格式: {TagName:args [|content] }     * @access private     * @param  string $content 要解析的模板内容     * @return void     */    private function parseTag(&$content)    {        $regex = $this->getRegex('tag');        if (preg_match_all($regex, $content, $matches, PREG_SET_ORDER)) {            foreach ($matches as $match) {                $str  = stripslashes($match[1]);                $flag = substr($str, 0, 1);                switch ($flag) {                    case '$':                        // 解析模板变量 格式 {$varName}                        // 是否带有?号                        if (false !== $pos = strpos($str, '?')) {                            $array = preg_split('/([!=]={1,2}|(?<!-)[><]={0,1})/', substr($str, 0, $pos), 2, PREG_SPLIT_DELIM_CAPTURE);                            $name  = $array[0];                            $this->parseVar($name);                            //$this->parseVarFunction($name);                            $str = trim(substr($str, $pos + 1));                            $this->parseVar($str);                            $first = substr($str, 0, 1);                            if (strpos($name, ')')) {                                // $name为对象或是自动识别,或者含有函数                                if (isset($array[1])) {                                    $this->parseVar($array[2]);                                    $name .= $array[1] . $array[2];                                }                                switch ($first) {                                    case '?':                                        $this->parseVarFunction($name);                                        $str = '<?php echo (' . $name . ') ? ' . $name . ' : ' . substr($str, 1) . '; ?>';                                        break;                                    case '=':                                        $str = '<?php if(' . $name . ') echo ' . substr($str, 1) . '; ?>';                                        break;                                    default:                                        $str = '<?php echo ' . $name . '?' . $str . '; ?>';                                }                            } else {                                if (isset($array[1])) {                                    $express = true;                                    $this->parseVar($array[2]);                                    $express = $name . $array[1] . $array[2];                                } else {                                    $express = false;                                }                                if (in_array($first, ['?', '=', ':'])) {                                    $str = trim(substr($str, 1));                                    if ('$' == substr($str, 0, 1)) {                                        $str = $this->parseVarFunction($str);                                    }                                }                                // $name为数组                                switch ($first) {                                    case '?':                                        // {$varname??'xxx'} $varname有定义则输出$varname,否则输出xxx                                        $str = '<?php echo ' . ($express ?: 'isset(' . $name . ')') . ' ? ' . $this->parseVarFunction($name) . ' : ' . $str . '; ?>';                                        break;                                    case '=':                                        // {$varname?='xxx'} $varname为真时才输出xxx                                        $str = '<?php if(' . ($express ?: '!empty(' . $name . ')') . ') echo ' . $str . '; ?>';                                        break;                                    case ':':                                        // {$varname?:'xxx'} $varname为真时输出$varname,否则输出xxx                                        $str = '<?php echo ' . ($express ?: '!empty(' . $name . ')') . ' ? ' . $this->parseVarFunction($name) . ' : ' . $str . '; ?>';                                        break;                                    default:                                        if (strpos($str, ':')) {                                            // {$varname ? 'a' : 'b'} $varname为真时输出a,否则输出b                                            $array = explode(':', $str, 2);                                            $array[0] = '$' == substr(trim($array[0]), 0, 1) ? $this->parseVarFunction($array[0]) : $array[0];                                            $array[1] = '$' == substr(trim($array[1]), 0, 1) ? $this->parseVarFunction($array[1]) : $array[1];                                            $str = implode(' : ', $array);                                        }                                        $str = '<?php echo ' . ($express ?: '!empty(' . $name . ')') . ' ? ' . $str . '; ?>';                                }                            }                        } else {                            $this->parseVar($str);                            $this->parseVarFunction($str);                            $str = '<?php echo ' . $str . '; ?>';                        }                        break;                    case ':':                        // 输出某个函数的结果                        $str = substr($str, 1);                        $this->parseVar($str);                        $str = '<?php echo ' . $str . '; ?>';                        break;                    case '~':                        // 执行某个函数                        $str = substr($str, 1);                        $this->parseVar($str);                        $str = '<?php ' . $str . '; ?>';                        break;                    case '-':                    case '+':                        // 输出计算                        $this->parseVar($str);                        $str = '<?php echo ' . $str . '; ?>';                        break;                    case '/':                        // 注释标签                        $flag2 = substr($str, 1, 1);                        if ('/' == $flag2 || ('*' == $flag2 && substr(rtrim($str), -2) == '*/')) {                            $str = '';                        }                        break;                    default:                        // 未识别的标签直接返回                        $str = $this->config['tpl_begin'] . $str . $this->config['tpl_end'];                        break;                }                $content = str_replace($match[0], $str, $content);            }            unset($matches);        }    }    /**     * 模板变量解析,支持使用函数     * 格式: {$varname|function1|function2=arg1,arg2}     * @access public     * @param  string $varStr 变量数据     * @return void     */    public function parseVar(&$varStr)    {        $varStr = trim($varStr);        if (preg_match_all('/\$[a-zA-Z_](?>\w*)(?:[:\.][0-9a-zA-Z_](?>\w*))+/', $varStr, $matches, PREG_OFFSET_CAPTURE)) {            static $_varParseList = [];            while ($matches[0]) {                $match = array_pop($matches[0]);                //如果已经解析过该变量字串,则直接返回变量值                if (isset($_varParseList[$match[0]])) {                    $parseStr = $_varParseList[$match[0]];                } else {                    if (strpos($match[0], '.')) {                        $vars  = explode('.', $match[0]);                        $first = array_shift($vars);                        if ('$Think' == $first) {                            // 所有以Think.打头的以特殊变量对待 无需模板赋值就可以输出                            $parseStr = $this->parseThinkVar($vars);                        } elseif ('$Request' == $first) {                            // 获取Request请求对象参数                            $method = array_shift($vars);                            if (!empty($vars)) {                                $params = implode('.', $vars);                                if ('true' != $params) {                                    $params = '\'' . $params . '\'';                                }                            } else {                                $params = '';                            }                            $parseStr = 'app(\'request\')->' . $method . '(' . $params . ')';                        } else {                            switch ($this->config['tpl_var_identify']) {                                case 'array': // 识别为数组                                    $parseStr = $first . '[\'' . implode('\'][\'', $vars) . '\']';                                    break;                                case 'obj': // 识别为对象                                    $parseStr = $first . '->' . implode('->', $vars);                                    break;                                default: // 自动判断数组或对象                                    $parseStr = '(is_array(' . $first . ')?' . $first . '[\'' . implode('\'][\'', $vars) . '\']:' . $first . '->' . implode('->', $vars) . ')';                            }                        }                    } else {                        $parseStr = str_replace(':', '->', $match[0]);                    }                    $_varParseList[$match[0]] = $parseStr;                }                $varStr = substr_replace($varStr, $parseStr, $match[1], strlen($match[0]));            }            unset($matches);        }    }    /**     * 对模板中使用了函数的变量进行解析     * 格式 {$varname|function1|function2=arg1,arg2}     * @access public     * @param  string    $varStr     变量字符串     * @param  bool      $autoescape 自动转义     * @return void     */    public function parseVarFunction(&$varStr, $autoescape = true)    {        if (!$autoescape && false === strpos($varStr, '|')) {            return $varStr;        } elseif ($autoescape && !preg_match('/\|(\s)?raw(\||\s)?/i', $varStr)) {            $varStr .= '|' . $this->config['default_filter'];        }        static $_varFunctionList = [];        $_key = md5($varStr);        //如果已经解析过该变量字串,则直接返回变量值        if (isset($_varFunctionList[$_key])) {            $varStr = $_varFunctionList[$_key];        } else {            $varArray = explode('|', $varStr);            // 取得变量名称            $name = trim(array_shift($varArray));            // 对变量使用函数            $length = count($varArray);            // 取得模板禁止使用函数列表            $template_deny_funs = explode(',', $this->config['tpl_deny_func_list']);            for ($i = 0; $i < $length; $i++) {                $args = explode('=', $varArray[$i], 2);                // 模板函数过滤                $fun = trim($args[0]);                if (in_array($fun, $template_deny_funs)) {                    continue;                }                switch (strtolower($fun)) {                    case 'raw':                        break;                    case 'date':                        $name = 'date(' . $args[1] . ',!is_numeric(' . $name . ')? strtotime(' . $name . ') : ' . $name . ')';                        break;                    case 'first':                        $name = 'current(' . $name . ')';                        break;                    case 'last':                        $name = 'end(' . $name . ')';                        break;                    case 'upper':                        $name = 'strtoupper(' . $name . ')';                        break;                    case 'lower':                        $name = 'strtolower(' . $name . ')';                        break;                    case 'format':                        $name = 'sprintf(' . $args[1] . ',' . $name . ')';                        break;                    case 'default': // 特殊模板函数                        if (false === strpos($name, '(')) {                            $name = '(isset(' . $name . ') && (' . $name . ' !== \'\')?' . $name . ':' . $args[1] . ')';                        } else {                            $name = '(' . $name . ' ?: ' . $args[1] . ')';                        }                        break;                    default: // 通用模板函数                        if (isset($args[1])) {                            if (strstr($args[1], '###')) {                                $args[1] = str_replace('###', $name, $args[1]);                                $name    = "$fun($args[1])";                            } else {                                $name = "$fun($name,$args[1])";                            }                        } else {                            if (!empty($args[0])) {                                $name = "$fun($name)";                            }                        }                }            }            $_varFunctionList[$_key] = $name;            $varStr                  = $name;        }        return $varStr;    }    /**     * 特殊模板变量解析     * 格式 以 $Think. 打头的变量属于特殊模板变量     * @access public     * @param  array $vars 变量数组     * @return string     */    public function parseThinkVar($vars)    {        $type  = strtoupper(trim(array_shift($vars)));        $param = implode('.', $vars);        if ($vars) {            switch ($type) {                case 'SERVER':                    $parseStr = 'app(\'request\')->server(\'' . $param . '\')';                    break;                case 'GET':                    $parseStr = 'app(\'request\')->get(\'' . $param . '\')';                    break;                case 'POST':                    $parseStr = 'app(\'request\')->post(\'' . $param . '\')';                    break;                case 'COOKIE':                    $parseStr = 'app(\'cookie\')->get(\'' . $param . '\')';                    break;                case 'SESSION':                    $parseStr = 'app(\'session\')->get(\'' . $param . '\')';                    break;                case 'ENV':                    $parseStr = 'app(\'request\')->env(\'' . $param . '\')';                    break;                case 'REQUEST':                    $parseStr = 'app(\'request\')->request(\'' . $param . '\')';                    break;                case 'CONST':                    $parseStr = strtoupper($param);                    break;                case 'LANG':                    $parseStr = 'app(\'lang\')->get(\'' . $param . '\')';                    break;                case 'CONFIG':                    $parseStr = 'app(\'config\')->get(\'' . $param . '\')';                    break;                default:                    $parseStr = '\'\'';                    break;            }        } else {            switch ($type) {                case 'NOW':                    $parseStr = "date('Y-m-d g:i a',time())";                    break;                case 'VERSION':                    $parseStr = 'app()->version()';                    break;                case 'LDELIM':                    $parseStr = '\'' . ltrim($this->config['tpl_begin'], '\\') . '\'';                    break;                case 'RDELIM':                    $parseStr = '\'' . ltrim($this->config['tpl_end'], '\\') . '\'';                    break;                default:                    if (defined($type)) {                        $parseStr = $type;                    } else {                        $parseStr = '';                    }            }        }        return $parseStr;    }    /**     * 分析加载的模板文件并读取内容 支持多个模板文件读取     * @access private     * @param  string $templateName 模板文件名     * @return string     */    private function parseTemplateName($templateName)    {        $array    = explode(',', $templateName);        $parseStr = '';        foreach ($array as $templateName) {            if (empty($templateName)) {                continue;            }            if (0 === strpos($templateName, '$')) {                //支持加载变量文件名                $templateName = $this->get(substr($templateName, 1));            }            $template = $this->parseTemplateFile($templateName);            if ($template) {                // 获取模板文件内容                $parseStr .= file_get_contents($template);            }        }        return $parseStr;    }    /**     * 解析模板文件名     * @access private     * @param  string $template 文件名     * @return string|false     */    private function parseTemplateFile($template)    {        if ('' == pathinfo($template, PATHINFO_EXTENSION)) {            if (strpos($template, '@')) {                list($module, $template) = explode('@', $template);            }            if (0 !== strpos($template, '/')) {                $template = str_replace(['/', ':'], $this->config['view_depr'], $template);            } else {                $template = str_replace(['/', ':'], $this->config['view_depr'], substr($template, 1));            }            if ($this->config['view_base']) {                $module = isset($module) ? $module : $this->app['request']->module();                $path   = $this->config['view_base'] . ($module ? $module . DIRECTORY_SEPARATOR : '');            } else {                $path = isset($module) ? $this->app->getAppPath() . $module . DIRECTORY_SEPARATOR . basename($this->config['view_path']) . DIRECTORY_SEPARATOR : $this->config['view_path'];            }            $template = $path . $template . '.' . ltrim($this->config['view_suffix'], '.');        }        if (is_file($template)) {            // 记录模板文件的更新时间            $this->includeFile[$template] = filemtime($template);            return $template;        }        throw new TemplateNotFoundException('template not exists:' . $template, $template);    }    /**     * 按标签生成正则     * @access private     * @param  string $tagName 标签名     * @return string     */    private function getRegex($tagName)    {        $regex = '';        if ('tag' == $tagName) {            $begin = $this->config['tpl_begin'];            $end   = $this->config['tpl_end'];            if (strlen(ltrim($begin, '\\')) == 1 && strlen(ltrim($end, '\\')) == 1) {                $regex = $begin . '((?:[\$]{1,2}[a-wA-w_]|[\:\~][\$a-wA-w_]|[+]{2}[\$][a-wA-w_]|[-]{2}[\$][a-wA-w_]|\/[\*\/])(?>[^' . $end . ']*))' . $end;            } else {                $regex = $begin . '((?:[\$]{1,2}[a-wA-w_]|[\:\~][\$a-wA-w_]|[+]{2}[\$][a-wA-w_]|[-]{2}[\$][a-wA-w_]|\/[\*\/])(?>(?:(?!' . $end . ').)*))' . $end;            }        } else {            $begin  = $this->config['taglib_begin'];            $end    = $this->config['taglib_end'];            $single = strlen(ltrim($begin, '\\')) == 1 && strlen(ltrim($end, '\\')) == 1 ? true : false;            switch ($tagName) {                case 'block':                    if ($single) {                        $regex = $begin . '(?:' . $tagName . '\b\s+(?>(?:(?!name=).)*)\bname=([\'\"])(?P<name>[\$\w\-\/\.]+)\\1(?>[^' . $end . ']*)|\/' . $tagName . ')' . $end;                    } else {                        $regex = $begin . '(?:' . $tagName . '\b\s+(?>(?:(?!name=).)*)\bname=([\'\"])(?P<name>[\$\w\-\/\.]+)\\1(?>(?:(?!' . $end . ').)*)|\/' . $tagName . ')' . $end;                    }                    break;                case 'literal':                    if ($single) {                        $regex = '(' . $begin . $tagName . '\b(?>[^' . $end . ']*)' . $end . ')';                        $regex .= '(?:(?>[^' . $begin . ']*)(?>(?!' . $begin . '(?>' . $tagName . '\b[^' . $end . ']*|\/' . $tagName . ')' . $end . ')' . $begin . '[^' . $begin . ']*)*)';                        $regex .= '(' . $begin . '\/' . $tagName . $end . ')';                    } else {                        $regex = '(' . $begin . $tagName . '\b(?>(?:(?!' . $end . ').)*)' . $end . ')';                        $regex .= '(?:(?>(?:(?!' . $begin . ').)*)(?>(?!' . $begin . '(?>' . $tagName . '\b(?>(?:(?!' . $end . ').)*)|\/' . $tagName . ')' . $end . ')' . $begin . '(?>(?:(?!' . $begin . ').)*))*)';                        $regex .= '(' . $begin . '\/' . $tagName . $end . ')';                    }                    break;                case 'restoreliteral':                    $regex = '<!--###literal(\d+)###-->';                    break;                case 'include':                    $name = 'file';                case 'taglib':                case 'layout':                case 'extend':                    if (empty($name)) {                        $name = 'name';                    }                    if ($single) {                        $regex = $begin . $tagName . '\b\s+(?>(?:(?!' . $name . '=).)*)\b' . $name . '=([\'\"])(?P<name>[\$\w\-\/\.\:@,\\\\]+)\\1(?>[^' . $end . ']*)' . $end;                    } else {                        $regex = $begin . $tagName . '\b\s+(?>(?:(?!' . $name . '=).)*)\b' . $name . '=([\'\"])(?P<name>[\$\w\-\/\.\:@,\\\\]+)\\1(?>(?:(?!' . $end . ').)*)' . $end;                    }                    break;            }        }        return '/' . $regex . '/is';    }    public function __debugInfo()    {        $data = get_object_vars($this);        unset($data['app'], $data['storage']);        return $data;    }}
 |