很多程序員以爲現在時興的golang才是王者(2023更新: 今年基本涼了)。擺一個殘酷的現實告訴你,ThinkPHP3.x系列框架因太容易使用, 在2022年國内中小企業,甚至是某些大型國企商業開發應用的出場概率大到你不敢想象。好多二線三線城市的小廠還在用它搞開發。不信?那你接到的項目一定很高耑嘍!筆者這邊好多大佬頭疼於TP的老項目, 但他們沒有想重構成golang或者rust。最起碼把php升級一下版本好伐!OK問題來了, 遷移了項目之後, 十有八九, 前耑顯示是空白的。何解?
於是有的人折騰index.php, 加入各種debug專用配置, 妄圖找到錯誤的點在哪裡, 然後逐個擊破:
ini_set('display_errors', '1');
error_reporting(E_ALL);
define("APP_DEBUG", true);
define('APP_ERROR_HANDLE',false);
define("SHOW_ERROR", true);然並卵。排除代碼質量的問題先不說,根本性的問題都來源於框架的一個過時的模板正則語法解析類, 解決了就ok了, 但是改那個正則又很惡心。
解決方案
找到框架裡一個文档名叫 ThinkTemplate.class.php 的模板正則處理類。把以下代碼拷進去。注意,改的其實都是preg_match相關的語句, 你可以搜索preg_match關鍵字試試。
<?php
/**
* ThinkPHP内置模板引擎類
* 支持XML標簽和普通標簽的模板解析
* 編譯型模板引擎 支持動態緩存
* @category Think
* @package Think
* @subpackage Template
* @author liu21st <liu21st@gmail.com>
*/
class ThinkTemplate
{
// 模板頁面中引入的標簽庫列表
protected $tagLib = array();
// 當前模板文档
protected $templateFile = '';
// 模板變量
public $tVar = array();
public $config = array();
private $literal = array();
private $block = array();
/**
* 架構函數
* @access public
*/
public function __construct()
{
$this->config['cache_path'] = C('CACHE_PATH');
$this->config['template_suffix'] = C('TMPL_TEMPLATE_SUFFIX');
$this->config['cache_suffix'] = C('TMPL_CACHFILE_SUFFIX');
$this->config['tmpl_cache'] = C('TMPL_CACHE_ON');
$this->config['cache_time'] = C('TMPL_CACHE_TIME');
$this->config['taglib_begin'] = $this->stripPreg(C('TAGLIB_BEGIN'));
$this->config['taglib_end'] = $this->stripPreg(C('TAGLIB_END'));
$this->config['tmpl_begin'] = $this->stripPreg(C('TMPL_L_DELIM'));
$this->config['tmpl_end'] = $this->stripPreg(C('TMPL_R_DELIM'));
$this->config['default_tmpl'] = C('TEMPLATE_NAME');
$this->config['layout_item'] = C('TMPL_LAYOUT_ITEM');
}
private function stripPreg($str)
{
return str_replace(
array('{', '}', '(', ')', '|', '[', ']', '-', '+', '*', '.', '^', '?'),
array('\{', '\}', '\(', '\)', '\|', '\[', '\]', '\-', '\+', '\*', '\.', '\^', '\?'),
$str
);
}
// 模板變量獲取和設置
public function get($name)
{
if (isset($this->tVar[$name]))
return $this->tVar[$name];
else
return false;
}
public function set($name, $value)
{
$this->tVar[$name] = $value;
}
/**
* 加載模板
* @access public
* @param string $tmplTemplateFile 模板文档
* @param array $templateVar 模板變量
* @param string $prefix 模板標識前綴
* @return void
*/
public function fetch($templateFile, $templateVar, $prefix = '')
{
$this->tVar = $templateVar;
$templateCacheFile = $this->loadTemplate($templateFile, $prefix);
// 模板陣列變量分解成爲獨立變量
extract($templateVar, EXTR_OVERWRITE);
//載入模版緩存文档
include $templateCacheFile;
}
/**
* 加載主模板並緩存
* @access public
* @param string $tmplTemplateFile 模板文档
* @param string $prefix 模板標識前綴
* @return string
* @throws ThinkExecption
*/
public function loadTemplate($tmplTemplateFile, $prefix = '')
{
if (is_file($tmplTemplateFile)) {
$this->templateFile = $tmplTemplateFile;
// 讀取模板文档内容
$tmplContent = file_get_contents($tmplTemplateFile);
} else {
$tmplContent = $tmplTemplateFile;
}
// 根據模版文档名定位緩存文档
$tmplCacheFile = $this->config['cache_path'] . $prefix . md5($tmplTemplateFile) . $this->config['cache_suffix'];
// 判斷是否啓用布局
if (C('LAYOUT_ON')) {
if (false !== strpos($tmplContent, '{__NOLAYOUT__}')) { // 可以單獨定義不使用布局
$tmplContent = str_replace('{__NOLAYOUT__}', '', $tmplContent);
} else { // 替換布局的主體内容
$layoutFile = THEME_PATH . C('LAYOUT_NAME') . $this->config['template_suffix'];
$tmplContent = str_replace($this->config['layout_item'], $tmplContent, file_get_contents($layoutFile));
}
}
// 編譯模板内容
$tmplContent = $this->compiler($tmplContent);
// 檢測模板目錄
$dir = dirname($tmplCacheFile);
if (!is_dir($dir))
mkdir($dir, 0755, true);
//重寫Cache文档
if (false === file_put_contents($tmplCacheFile, trim($tmplContent)))
throw_exception(L('_CACHE_WRITE_ERROR_') . ':' . $tmplCacheFile);
return $tmplCacheFile;
}
/**
* 編譯模板文档内容
* @access protected
* @param mixed $tmplContent 模板内容
* @return string
*/
protected function compiler($tmplContent)
{
// 模板解析
$tmplContent = $this->parse($tmplContent);
// 還原被替換的Literal標簽
$tmplContent = preg_replace_callback('/<!--###literal(\d+)###-->/is', function ($match) {
return $this->restoreLiteral($match[1]);
}, $tmplContent);
// 添加安全代碼
$tmplContent = '<?php if (!defined(\'THINK_PATH\')) exit();?>' . $tmplContent;
if (C('TMPL_STRIP_SPACE')) {
/* 去除html空格與換行 */
$find = array('~>\s+<~', '~>(\s+\n|\r)~');
$replace = array('><', '>');
$tmplContent = preg_replace($find, $replace, $tmplContent);
}
// 優化生成的php代碼
$tmplContent = str_replace('?><?php', '', $tmplContent);
return strip_whitespace($tmplContent);
}
/**
* 模板解析入口
* 支持普通標簽和TagLib解析 支持自定義標簽庫
* @access public
* @param string $content 要解析的模板内容
* @return string
*/
public function parse($content)
{
// 内容爲空不解析
if (empty($content)) return '';
$begin = $this->config['taglib_begin'];
$end = $this->config['taglib_end'];
// 檢查include語法
$content = $this->parseInclude($content);
// 檢查PHP語法
$content = $this->parsePhp($content);
// 首先替換literal標簽内容
$content = preg_replace_callback('/' . $begin . 'literal' . $end . '(.*?)' . $begin . '\/literal' . $end . '/is', function ($match) {
$this->parseLiteral($match[1]);
}, $content);
// 獲取需要引入的標簽庫列表
// 標簽庫只需要定義一次,允許引入多個一次
// 一般放在文档的最前面
// 格式:<taglib name="html,mytag..." />
// 當TAGLIB_LOAD配置爲true時才會進行檢測
if (C('TAGLIB_LOAD')) {
$this->getIncludeTagLib($content);
if (!empty($this->tagLib)) {
// 對導入的TagLib進行解析
foreach ($this->tagLib as $tagLibName) {
$this->parseTagLib($tagLibName, $content);
}
}
}
// 預先加載的標簽庫 無需在每個模板中使用taglib標簽加載 但必須使用標簽庫XML前綴
if (C('TAGLIB_PRE_LOAD')) {
$tagLibs = explode(',', C('TAGLIB_PRE_LOAD'));
foreach ($tagLibs as $tag) {
$this->parseTagLib($tag, $content);
}
}
// 内置標簽庫 無需使用taglib標簽導入就可以使用 並且不需使用標簽庫XML前綴
$tagLibs = explode(',', C('TAGLIB_BUILD_IN'));
foreach ($tagLibs as $tag) {
$this->parseTagLib($tag, $content, true);
}
// 解析普通模板標簽 {tagName}
return preg_replace_callback('/(' . $this->config['tmpl_begin'] . ')([^\d\s' . $this->config['tmpl_begin'] . $this->config['tmpl_end'] . '].+?)(' . $this->config['tmpl_end'] . ')/is', function ($match) {
return $this->parseTag($match[2]);
}, $content);
}
// 檢查PHP語法
protected function parsePhp($content)
{
if (ini_get('short_open_tag')) {
// 開啓短標簽的情況要將<?標簽用echo方式輸出 否則無法正常輸出xml標識
$content = preg_replace('/(<\?(?!php|=|$))/i', '<?php echo \'\\1\'; ?>' . "\n", $content);
}
// PHP語法檢查
if (C('TMPL_DENY_PHP') && false !== strpos($content, '<?php')) {
throw_exception(L('_NOT_ALLOW_PHP_'));
}
return $content;
}
// 解析模板中的布局標簽
protected function parseLayout($content)
{
// 讀取模板中的布局標簽
$find = preg_match('/' . $this->config['taglib_begin'] . 'layout\s(.+?)\s*?\/' . $this->config['taglib_end'] . '/is', $content, $matches);
if ($find) {
// 替換Layout標簽
$content = str_replace($matches[0], '', $content);
// 解析Layout標簽
$array = $this->parseXmlAttrs($matches[1]);
if (!C('LAYOUT_ON') || C('LAYOUT_NAME') != $array['name']) {
// 讀取布局模板
$layoutFile = THEME_PATH . $array['name'] . $this->config['template_suffix'];
$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);
}
return $content;
}
// 解析模板中的include標簽
protected function parseInclude($content)
{
// 解析繼承
$content = $this->parseExtend($content);
// 解析布局
$content = $this->parseLayout($content);
// 讀取模板中的include標簽
$find = preg_match_all('/' . $this->config['taglib_begin'] . 'include\s(.+?)\s*?\/' . $this->config['taglib_end'] . '/is', $content, $matches);
if ($find) {
for ($i = 0; $i < $find; $i++) {
$include = $matches[1][$i];
$array = $this->parseXmlAttrs($include);
$file = $array['file'];
unset($array['file']);
$content = str_replace($matches[0][$i], $this->parseIncludeItem($file, $array), $content);
}
}
return $content;
}
// 解析模板中的extend標簽
protected function parseExtend($content)
{
$begin = $this->config['taglib_begin'];
$end = $this->config['taglib_end'];
// 讀取模板中的繼承標簽
$find = preg_match('/' . $begin . 'extend\s(.+?)\s*?\/' . $end . '/is', $content, $matches);
if ($find) {
// 替換extend標簽
$content = str_replace($matches[0], '', $content);
// 記錄頁面中的block標簽
preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
return $this->parseBlock($match[1], $match[2]);
}, $content);
// 讀取繼承模板
$array = $this->parseXmlAttrs($matches[1]);
$content = $this->parseTemplateName($array['name']);
// 替換block標簽
return preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
return $this->replaceBlock($match[1], $match[2]);
}, $content);
} else {
return preg_replace_callback('/' . $begin . 'block\sname=(.+?)\s*?' . $end . '(.*?)' . $begin . '\/block' . $end . '/is', function ($match) {
return stripslashes($match[1]);
}, $content);
}
}
/**
* 分析XML屬性
* @access private
* @param string $attrs XML屬性字符串
* @return array
*/
private function parseXmlAttrs($attrs)
{
$xml = '<tpl><tag ' . $attrs . ' /></tpl>';
$xml = simplexml_load_string($xml);
if (!$xml)
throw_exception(L('_XML_TAG_ERROR_'));
$xml = (array)($xml->tag->attributes());
$array = array_change_key_case($xml['@attributes']);
return $array;
}
/**
* 替換頁面中的literal標簽
* @access private
* @param string $content 模板内容
* @return string|false
*/
private function parseLiteral($content)
{
if (trim($content) == '') return '';
$content = stripslashes($content);
$i = count($this->literal);
$parseStr = "<!--###literal{$i}###-->";
$this->literal[$i] = $content;
return $parseStr;
}
/**
* 還原被替換的literal標簽
* @access private
* @param string $tag literal標簽序號
* @return string|false
*/
private function restoreLiteral($tag)
{
// 還原literal標簽
$parseStr = $this->literal[$tag];
// 銷毀literal記錄
unset($this->literal[$tag]);
return $parseStr;
}
/**
* 記錄當前頁面中的block標簽
* @access private
* @param string $name block名稱
* @param string $content 模板内容
* @return string
*/
private function parseBlock($name, $content)
{
$this->block[$name] = $content;
return '';
}
/**
* 替換繼承模板中的block標簽
* @access private
* @param string $name block名稱
* @param string $content 模板内容
* @return string
*/
private function replaceBlock($name, $content)
{
// 替換block標簽 沒有重新定義則使用原來的
$replace = isset($this->block[$name]) ? $this->block[$name] : $content;
return stripslashes($replace);
}
/**
* 搜索模板頁面中包含的TagLib庫
* 並返回列表
* @access public
* @param string $content 模板内容
* @return string|false
*/
public function getIncludeTagLib(&$content)
{
// 搜索是否有TagLib標簽
$find = preg_match('/' . $this->config['taglib_begin'] . 'taglib\s(.+?)(\s*?)\/' . $this->config['taglib_end'] . '\W/is', $content, $matches);
if ($find) {
// 替換TagLib標簽
$content = str_replace($matches[0], '', $content);
// 解析TagLib標簽
$array = $this->parseXmlAttrs($matches[1]);
$this->tagLib = explode(',', $array['name']);
}
return;
}
/**
* TagLib庫解析
* @access public
* @param string $tagLib 要解析的標簽庫
* @param string $content 要解析的模板内容
* @param boolen $hide 是否隱藏標簽庫前綴
* @return string
*/
public function parseTagLib($tagLib, &$content, $hide = false)
{
$begin = $this->config['taglib_begin'];
$end = $this->config['taglib_end'];
$className = 'TagLib' . ucwords($tagLib);
$tLib = Think::instance($className);
foreach ($tLib->getTags() as $name => $val) {
$tags = array($name);
if (isset($val['alias'])) { // 別名設置
$tags = explode(',', $val['alias']);
$tags[] = $name;
}
$level = isset($val['level']) ? $val['level'] : 1;
$closeTag = isset($val['close']) ? $val['close'] : true;
foreach ($tags as $tag) {
$parseTag = !$hide ? $tagLib . ':' . $tag : $tag; // 實際要解析的標簽名稱
if (!method_exists($tLib, '_' . $tag)) {
// 別名可以無需定義解析方法
$tag = $name;
}
$n1 = empty($val['attr']) ? '(\s*?)' : '\s([^' . $end . ']*)';
if (!$closeTag) {
$patterns = '/' . $begin . $parseTag . $n1 . '\/(\s*?)' . $end . '/is';
$content = preg_replace_callback($patterns, function ($match) use ($tagLib, $tag) {
return $this->parseXmlTag($tagLib, $tag, $match[1], '');
}, $content);
} else {
$patterns = '/' . $begin . $parseTag . $n1 . $end . '(.*?)' . $begin . '\/' . $parseTag . '(\s*?)' . $end . '/is';
for ($i = 0; $i < $level; $i++) {
$content = preg_replace_callback($patterns, function ($match) use ($tagLib, $tag) {
return $this->parseXmlTag($tagLib, $tag, $match[1], $match[2]);
}, $content);
}
}
}
}
}
/**
* 解析標簽庫的標簽
* 需要調用對應的標簽庫文档解析類
* @access public
* @param string $tagLib 標簽庫名稱
* @param string $tag 標簽名
* @param string $attr 標簽屬性
* @param string $content 標簽内容
* @return string|false
*/
public function parseXmlTag($tagLib, $tag, $attr, $content)
{
// if (MAGIC_QUOTES_GPC) {
$attr = stripslashes($attr);
$content = stripslashes($content);
// }
if (ini_get('magic_quotes_sybase'))
$attr = str_replace('\"', '\'', $attr);
$tLib = Think::instance('TagLib' . ucwords(strtolower($tagLib)));
$parse = '_' . $tag;
$content = trim($content);
return $tLib->$parse($attr, $content);
}
/**
* 模板標簽解析
* 格式: {TagName:args [|content] }
* @access public
* @param string $tagStr 標簽内容
* @return string
*/
public function parseTag($tagStr)
{
// if (MAGIC_QUOTES_GPC) {
$tagStr = stripslashes($tagStr);
// }
// 還原非模板標簽
if (preg_match('/^[\s|\d]/is', $tagStr))
// 過濾空格和數字打頭的標簽
return C('TMPL_L_DELIM') . $tagStr . C('TMPL_R_DELIM');
$flag = substr($tagStr, 0, 1);
$flag2 = substr($tagStr, 1, 1);
$name = substr($tagStr, 1);
if ('$' == $flag && '.' != $flag2 && '(' != $flag2) { //解析模板變量 格式 {$varName}
return $this->parseVar($name);
} elseif ('-' == $flag || '+' == $flag) { // 輸出計算
return '<?php echo ' . $flag . $name . ';?>';
} elseif (':' == $flag) { // 輸出某個函數的結果
return '<?php echo ' . $name . ';?>';
} elseif ('~' == $flag) { // 執行某個函數
return '<?php ' . $name . ';?>';
} elseif (substr($tagStr, 0, 2) == '//' || (substr($tagStr, 0, 2) == '/*' && substr($tagStr, -2) == '*/')) {
// 注釋標簽
return '';
}
// 未識別的標簽直接返回
return C('TMPL_L_DELIM') . $tagStr . C('TMPL_R_DELIM');
}
/**
* 模板變量解析,支持使用函數
* 格式: {$varname|function1|function2=arg1,arg2}
* @access public
* @param string $varStr 變量數據
* @return string
*/
public function parseVar($varStr)
{
$varStr = trim($varStr);
static $_varParseList = array();
//如果已經解析過該變量字串,則直接返回變量值
if (isset($_varParseList[$varStr])) return $_varParseList[$varStr];
$parseStr = '';
$varExists = true;
if (!empty($varStr)) {
$varArray = explode('|', $varStr);
//取得變量名稱
$var = array_shift($varArray);
if ('Think.' == substr($var, 0, 6)) {
// 所有以Think.打頭的以特殊變量對待 無需模板賦值就可以輸出
$name = $this->parseThinkVar($var);
} elseif (false !== strpos($var, '.')) {
//支持 {$var.property}
$vars = explode('.', $var);
$var = array_shift($vars);
switch (strtolower(C('TMPL_VAR_IDENTIFY'))) {
case 'array': // 識別爲數組
$name = '$' . $var;
foreach ($vars as $key => $val)
$name .= '["' . $val . '"]';
break;
case 'obj': // 識別爲對象
$name = '$' . $var;
foreach ($vars as $key => $val)
$name .= '->' . $val;
break;
default: // 自動判斷數組或對象 只支持二維
$name = 'is_array($' . $var . ')?$' . $var . '["' . $vars[0] . '"]:$' . $var . '->' . $vars[0];
}
} elseif (false !== strpos($var, '[')) {
// 支持 {$var['key']} 方式輸出數組
$name = "$" . $var;
preg_match('/(.+?)\[(.+?)\]/is', $var, $match);
$var = $match[1];
} elseif (false !== strpos($var, ':') && false === strpos($var, '(') && false === strpos($var, '::') && false === strpos($var, '?')) {
// 支持 {$var:property} 方式輸出對象的屬性
$vars = explode(':', $var);
$var = str_replace(':', '->', $var);
$name = "$" . $var;
$var = $vars[0];
} else {
$name = "$$var";
}
//對變量使用函數
if (count($varArray) > 0)
$name = $this->parseVarFunction($name, $varArray);
$parseStr = '<?php echo (' . $name . '); ?>';
}
$_varParseList[$varStr] = $parseStr;
return $parseStr;
}
/**
* 對模板變量使用函數
* 格式 {$varname|function1|function2=arg1,arg2}
* @access public
* @param string $name 變量名
* @param array $varArray 函數列表
* @return string
*/
public function parseVarFunction($name, $varArray)
{
// 對變量使用函數
$length = count($varArray);
// 取得模板禁止使用函數列表
$template_deny_funs = explode(',', C('TMPL_DENY_FUNC_LIST'));
for ($i = 0; $i < $length; $i++) {
$args = explode('=', $varArray[$i], 2);
// 模板函數過濾
$fun = strtolower(trim($args[0]));
switch ($fun) {
case 'default': // 特殊模板函數
$name = '(' . $name . ')?(' . $name . '):' . $args[1];
break;
default: // 通用模板函數
if (!in_array($fun, $template_deny_funs)) {
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)";
}
}
}
}
return $name;
}
/**
* 特殊模板變量解析
* 格式 以 $Think. 打頭的變量屬於特殊模板變量
* @access public
* @param string $varStr 變量字符串
* @return string
*/
public function parseThinkVar($varStr)
{
$vars = explode('.', $varStr);
$vars[1] = strtoupper(trim($vars[1]));
$parseStr = '';
if (count($vars) >= 3) {
$vars[2] = trim($vars[2]);
switch ($vars[1]) {
case 'SERVER':
$parseStr = '$_SERVER[\'' . strtoupper($vars[2]) . '\']';
break;
case 'GET':
$parseStr = '$_GET[\'' . $vars[2] . '\']';
break;
case 'POST':
$parseStr = '$_POST[\'' . $vars[2] . '\']';
break;
case 'COOKIE':
if (isset($vars[3])) {
$parseStr = '$_COOKIE[\'' . $vars[2] . '\'][\'' . $vars[3] . '\']';
} else {
$parseStr = 'cookie(\'' . $vars[2] . '\')';
}
break;
case 'SESSION':
if (isset($vars[3])) {
$parseStr = '$_SESSION[\'' . $vars[2] . '\'][\'' . $vars[3] . '\']';
} else {
$parseStr = 'session(\'' . $vars[2] . '\')';
}
break;
case 'ENV':
$parseStr = '$_ENV[\'' . strtoupper($vars[2]) . '\']';
break;
case 'REQUEST':
$parseStr = '$_REQUEST[\'' . $vars[2] . '\']';
break;
case 'CONST':
$parseStr = strtoupper($vars[2]);
break;
case 'LANG':
$parseStr = 'L("' . $vars[2] . '")';
break;
case 'CONFIG':
if (isset($vars[3])) {
$vars[2] .= '.' . $vars[3];
}
$parseStr = 'C("' . $vars[2] . '")';
break;
default:
break;
}
} else if (count($vars) == 2) {
switch ($vars[1]) {
case 'NOW':
$parseStr = "date('Y-m-d g:i a',time())";
break;
case 'VERSION':
$parseStr = 'THINK_VERSION';
break;
case 'TEMPLATE':
$parseStr = "'" . $this->templateFile . "'"; //'C("TEMPLATE_NAME")';
break;
case 'LDELIM':
$parseStr = 'C("TMPL_L_DELIM")';
break;
case 'RDELIM':
$parseStr = 'C("TMPL_R_DELIM")';
break;
default:
if (defined($vars[1]))
$parseStr = $vars[1];
}
}
return $parseStr;
}
/**
* 加載公共模板並緩存 和當前模板在同一路徑,否則使用相對路徑
* @access private
* @param string $tmplPublicName 公共模板文档名
* @param array $vars 要傳遞的變量列表
* @return string
*/
private function parseIncludeItem($tmplPublicName, $vars = array())
{
// 分析模板文档名並讀取内容
$parseStr = $this->parseTemplateName($tmplPublicName);
// 替換變量
foreach ($vars as $key => $val) {
$parseStr = str_replace('[' . $key . ']', $val, $parseStr);
}
// 再次對包含文档進行模板分析
return $this->parseInclude($parseStr);
}
/**
* 分析加載的模板文档並讀取内容 支持多個模板文档讀取
* @access private
* @param string $tmplPublicName 模板文档名
* @return string
*/
private function parseTemplateName($templateName)
{
if (substr($templateName, 0, 1) == '$')
// 支持加載變量文档名
$templateName = $this->get(substr($templateName, 1));
$array = explode(',', $templateName);
$parseStr = '';
foreach ($array as $templateName) {
if (false === strpos($templateName, $this->config['template_suffix'])) {
// 解析槼則爲 模板主題:模塊:操作 不支持 跨項目和跨分組調用
$path = explode(':', $templateName);
$action = array_pop($path);
$module = !empty($path) ? array_pop($path) : MODULE_NAME;
if (!empty($path) && THEME_NAME) { // 設置模板主題
$path = dirname(THEME_PATH) . '/' . array_pop($path) . '/';
} else {
$path = THEME_PATH;
}
$templateName = $path . $module . C('TMPL_FILE_DEPR') . $action . $this->config['template_suffix'];
}
// 獲取模板文档内容
$parseStr .= file_get_contents($templateName);
}
return $parseStr;
}
}至於其他問題, 你還有改不了的嗎? 請留言。
2 樓 IP 112.97.***.22 的嘉賓 说道 : 很久前
3 樓 IP 222.168.***.253 的嘉賓 说道 : 很久前