| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320 |
- <?php
- // +----------------------------------------------------------------------
- // | ThinkPHP [ WE CAN DO IT JUST THINK ]
- // +----------------------------------------------------------------------
- // | Copyright (c) 2006~2019 http://thinkphp.cn All rights reserved.
- // +----------------------------------------------------------------------
- // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
- // +----------------------------------------------------------------------
- // | Author: liu21st <liu21st@gmail.com>
- // +----------------------------------------------------------------------
- declare (strict_types = 1);
- namespace think;
- use Exception;
- use Psr\SimpleCache\CacheInterface;
- /**
- * ThinkPHP分离出来的模板引擎
- * 支持XML标签和普通标签的模板解析
- * 编译型模板引擎 支持动态缓存
- */
- class Template
- {
- /**
- * 模板变量
- * @var array
- */
- protected $data = [];
- /**
- * 模板配置参数
- * @var array
- */
- protected $config = [
- 'view_path' => '', // 模板路径
- 'view_suffix' => 'html', // 默认模板文件后缀
- 'view_depr' => DIRECTORY_SEPARATOR,
- 'cache_path' => '',
- '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 $extend = [];
- /**
- * 模板包含信息
- * @var array
- */
- private $includeFile = [];
- /**
- * 模板存储对象
- * @var object
- */
- protected $storage;
- /**
- * 查询缓存对象
- * @var CacheInterface
- */
- protected $cache;
- /**
- * 架构函数
- * @access public
- * @param array $config
- */
- public function __construct(array $config = [])
- {
- $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';
- $class = false !== strpos($type, '\\') ? $type : '\\think\\template\\driver\\' . ucwords($type);
- $this->storage = new $class();
- }
- /**
- * 模板变量赋值
- * @access public
- * @param array $vars 模板变量
- * @return $this
- */
- public function assign(array $vars = [])
- {
- $this->data = array_merge($this->data, $vars);
- return $this;
- }
- /**
- * 模板引擎参数赋值
- * @access public
- * @param string $name
- * @param mixed $value
- */
- public function __set($name, $value)
- {
- $this->config[$name] = $value;
- }
- /**
- * 设置缓存对象
- * @access public
- * @param CacheInterface $cache 缓存对象
- * @return void
- */
- public function setCache(CacheInterface $cache): void
- {
- $this->cache = $cache;
- }
- /**
- * 模板引擎配置
- * @access public
- * @param array $config
- * @return $this
- */
- public function config(array $config)
- {
- $this->config = array_merge($this->config, $config);
- return $this;
- }
- /**
- * 获取模板引擎配置项
- * @access public
- * @param string $name
- * @return mixed
- */
- public function getConfig(string $name)
- {
- return $this->config[$name] ?? null;
- }
- /**
- * 模板变量获取
- * @access public
- * @param string $name 变量名
- * @return mixed
- */
- public function get(string $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 $rule 解析规则
- * @param callable $callback 解析规则
- * @return void
- */
- public function extend(string $rule, callable $callback = null): void
- {
- $this->extend[$rule] = $callback;
- }
- /**
- * 渲染模板文件
- * @access public
- * @param string $template 模板文件
- * @param array $vars 模板变量
- * @return void
- */
- public function fetch(string $template, array $vars = []): void
- {
- if ($vars) {
- $this->data = array_merge($this->data, $vars);
- }
- if (!empty($this->config['cache_id']) && $this->config['display_cache'] && $this->cache) {
- // 读取渲染缓存
- if ($this->cache->has($this->config['cache_id'])) {
- echo $this->cache->get($this->config['cache_id']);
- 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'] && $this->cache) {
- // 缓存页面输出
- $this->cache->set($this->config['cache_id'], $content, $this->config['cache_time']);
- }
- echo $content;
- }
- }
- /**
- * 检查编译缓存是否存在
- * @access public
- * @param string $cacheId 缓存的id
- * @return boolean
- */
- public function isCache(string $cacheId): bool
- {
- if ($cacheId && $this->cache && $this->config['display_cache']) {
- // 缓存页面输出
- return $this->cache->has($cacheId);
- }
- return false;
- }
- /**
- * 渲染模板内容
- * @access public
- * @param string $content 模板内容
- * @param array $vars 模板变量
- * @return void
- */
- public function display(string $content, array $vars = []): void
- {
- if ($vars) {
- $this->data = array_merge($this->data, $vars);
- }
- $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 $this
- */
- public function layout($name, string $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 bool
- */
- private function checkCache(string $cacheFile): bool
- {
- 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 private
- * @param string $content 模板内容
- * @param string $cacheFile 缓存文件名
- * @return void
- */
- private function compiler(string &$content, string $cacheFile): void
- {
- // 判断是否启用布局
- 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(string &$content): void
- {
- // 内容为空不解析
- 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 Exception
- */
- private function parsePhp(string &$content): void
- {
- // 短标签的情况要将<?标签用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(string &$content): void
- {
- // 读取模板中的布局标签
- 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(string &$content): void
- {
- $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(string &$content): void
- {
- $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(string &$content, bool $restore = false): void
- {
- $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(string &$content, bool $sort = false): array
- {
- $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(string &$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(string $tagLib, string &$content, bool $hide = false): void
- {
- 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(string $str, string $name = null): array
- {
- $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(string &$content): void
- {
- $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(string &$varStr): void
- {
- $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 (isset($this->extend[$first])) {
- $callback = $this->extend[$first];
- $parseStr = $callback($vars);
- } elseif ('$Request' == $first) {
- // 输出请求变量
- $parseStr = $this->parseRequestVar($vars);
- } elseif ('$Think' == $first) {
- // 所有以Think.打头的以特殊变量对待 无需模板赋值就可以输出
- $parseStr = $this->parseThinkVar($vars);
- } 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 string
- */
- public function parseVarFunction(string &$varStr, bool $autoescape = true): string
- {
- 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;
- }
- /**
- * 请求变量解析
- * 格式 以 $Request. 打头的变量属于请求变量
- * @access public
- * @param array $vars 变量数组
- * @return string
- */
- public function parseRequestVar(array $vars): string
- {
- $type = strtoupper(trim(array_shift($vars)));
- $param = implode('.', $vars);
- switch ($type) {
- case 'SERVER':
- $parseStr = '$_SERVER[\'' . $param . '\']';
- break;
- case 'GET':
- $parseStr = '$_GET[\'' . $param . '\']';
- break;
- case 'POST':
- $parseStr = '$_POST[\'' . $param . '\']';
- break;
- case 'COOKIE':
- $parseStr = '$_COOKIE[\'' . $param . '\']';
- break;
- case 'SESSION':
- $parseStr = '$_SESSION[\'' . $param . '\']';
- break;
- case 'ENV':
- $parseStr = '$_ENV[\'' . $param . '\']';
- break;
- case 'REQUEST':
- $parseStr = '$_REQUEST[\'' . $param . '\']';
- break;
- default:
- $parseStr = '\'\'';
- }
- return $parseStr;
- }
- /**
- * 特殊模板变量解析
- * 格式 以 $Think. 打头的变量属于特殊模板变量
- * @access public
- * @param array $vars 变量数组
- * @return string
- */
- public function parseThinkVar(array $vars): string
- {
- $type = strtoupper(trim(array_shift($vars)));
- $param = implode('.', $vars);
- switch ($type) {
- case 'CONST':
- $parseStr = strtoupper($param);
- break;
- case 'NOW':
- $parseStr = "date('Y-m-d g:i a',time())";
- break;
- case 'LDELIM':
- $parseStr = '\'' . ltrim($this->config['tpl_begin'], '\\') . '\'';
- break;
- case 'RDELIM':
- $parseStr = '\'' . ltrim($this->config['tpl_end'], '\\') . '\'';
- break;
- default:
- $parseStr = defined($type) ? $type : '\'\'';
- }
- return $parseStr;
- }
- /**
- * 分析加载的模板文件并读取内容 支持多个模板文件读取
- * @access private
- * @param string $templateName 模板文件名
- * @return string
- */
- private function parseTemplateName(string $templateName): string
- {
- $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
- */
- private function parseTemplateFile(string $template): string
- {
- if ('' == pathinfo($template, PATHINFO_EXTENSION)) {
- if (0 !== strpos($template, '/')) {
- $template = str_replace(['/', ':'], $this->config['view_depr'], $template);
- } else {
- $template = str_replace(['/', ':'], $this->config['view_depr'], substr($template, 1));
- }
- $template = $this->config['view_path'] . $template . '.' . ltrim($this->config['view_suffix'], '.');
- }
- if (is_file($template)) {
- // 记录模板文件的更新时间
- $this->includeFile[$template] = filemtime($template);
- return $template;
- }
- throw new Exception('template not exists:' . $template);
- }
- /**
- * 按标签生成正则
- * @access private
- * @param string $tagName 标签名
- * @return string
- */
- private function getRegex(string $tagName): string
- {
- $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['storage']);
- return $data;
- }
- }
|