<?php
// +----------------------------------------------------------------------
// | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
// +----------------------------------------------------------------------
// | Copyright (c) 2006-2014 http://thinkphp.cn All rights reserved.
// +----------------------------------------------------------------------
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
// +----------------------------------------------------------------------
// | Author: liu21st <liu21st@gmail.com>
// +----------------------------------------------------------------------
namespace Think;
/**
* ThinkPHP内置模板引擎类
* 支持XML标签和普通标签的模板解析
* 编译型模板引擎 支持动态缓存
* 模版 里面 最好的一个部分了
*/
class Template {

// 模板页面中引入的标签库列表
protected $tagLib = array();// 模版看来是把,模版的操作,跟标签操作进行分类
// 这个有点像是那个 mysql 里面的 表达式转义
// 当前模板文件
protected $templateFile = '';// 当前模版文件 处理 估计就是要引入进来,
// 这样对于class 来说 里面的 文件 跟 输入的接口进行了各种统一,
// 其实现在的 Php 编程已经上升了一个新的高度,那个就是 java类似的情况了
// 编程了 class编程为基础 function 进行辅助的状态呢。

// 总结: 标签 ---》 模版文件

// 模板变量
public $tVar = array(); // 模版变量 1
public $config = array(); // 模版变量 2
private $literal = array(); // 模版变量 3
private $block = array(); // 模版变量 4

/**
* 架构函数
* @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'));// 'TMPL_L_DELIM' => '{', // 模板引擎普通标签开始标记
$this->config['tmpl_end'] = $this->stripPreg(C('TMPL_R_DELIM'));//'TMPL_R_DELIM' => '}', // 模板引擎普通标签结束标记
$this->config['default_tmpl'] = C('TEMPLATE_NAME');// 默认模版名i在
$this->config['layout_item'] = C('TMPL_LAYOUT_ITEM');//'TMPL_LAYOUT_ITEM' => '{__CONTENT__}', // 布局模板的内容替换标识
}

private function stripPreg($str) {
return str_replace(
array('{','}','(',')','|','[',']','-','+','*','.','^','?'),
array('\{','\}','\(','\)','\|','\[','\]','\-','\+','\*','\.','\^','\?'),
$str);
}// 这个就是个大剂量 的str_replace













// 模板变量获取和设置
public function get($name) { // 获取模版变量数据
if(isset($this->tVar[$name]))
return $this->tVar[$name];
else
return false;
// 其实可以直接这样改写 一句话搞定,哈哈
// return isset($this->tVar[$name])?$this->tVar[$name]:false;
}

public function set($name,$value) {// 这个设置很简洁
$this->tVar[$name]= $value;
}

/**
* 加载模板
* @access public
* @param string $templateFile 模板文件
* @param array $templateVar 模板变量
* @param string $prefix 模板标识前缀
* @return void
* 加载模版
*/
public function fetch($templateFile,$templateVar,$prefix='') {
$this->tVar = $templateVar;// 模版变量数组 这个是当前模版的变量数组
// extract($vars, EXTR_OVERWRITE);
// 这个是一个大量 使用的函数,这个是比较不错的,就是在加载缓存文件之前,先把对应的变量,进行了赋值
// 其实人家给的 变量名字就是很明显的了,
$templateCacheFile = $this->loadTemplate($templateFile,$prefix);// 加载对应的模版文件 并且生成其缓存文件,
Storage::load($templateCacheFile,$this->tVar,null,'tpl');
// 这个干了这个事情:第一 :将模版数组进行赋值,第二,根据对应的模版文件生成对应的模版缓存文件,第三: 进行变量赋值,并且 include 模版。

} // 总结:一手抓变量,一手抓模版,融会贯通,哈哈哈

/**
* 加载主模板并缓存
* @access public
* @param string $templateFile 模板文件
* @param string $prefix 模板标识前缀
* @return string
* @throws ThinkExecption
*/
public function loadTemplate ($templateFile,$prefix='') {// 牛叉,居然还有模版前缀,
// $templateFile 是一个标准的 模版文件, 准确的来说应该是标准的 完整路径的模版文件
if(is_file($templateFile)) {
$this->templateFile = $templateFile;
// 读取模板文件内容
$tmplContent = file_get_contents($templateFile);
}else{
$tmplContent = $templateFile;
}
// 我去,如果是文件,并且可以找到,那么,就读取文件内容,否则,直接视作是文件内容啊,这个有点霸气啊

// 根据模版文件名定位缓存文件
$tmplCacheFile = $this->config['cache_path'].$prefix.md5($templateFile).$this->config['cache_suffix'];
// 个人感觉就是 md5加密的模版名字,果不其然啊,哈哈

// 判断是否启用布局
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'];
// 检查布局文件
if(!is_file($layoutFile)) {
E(L('_TEMPLATE_NOT_EXIST_').':'.$layoutFile);
}
$tmplContent = str_replace($this->config['layout_item'],$tmplContent,file_get_contents($layoutFile));
}
}// 如果启用的布局文件,就对布局文件里面的内容 进行替换。
// 编译模板内容
$tmplContent = $this->compiler($tmplContent);// 对内容,进行编译
Storage::put($tmplCacheFile,trim($tmplContent),'tpl');// 写入文件
return $tmplCacheFile;// 返回文件路径,即可。
}





























/**
* 编译模板文件内容
* @access protected
* @param mixed $tmplContent 模板内容
* @return string
* 这个是编译
* 干大活的,就是 就是 个总的 外包
*/
protected function compiler($tmplContent) {
//模板解析
$tmplContent = $this->parse($tmplContent); // 这个获取 模版的解析文件 {$a} ==> <?php echo $a;


// 还原被替换的Literal标签
// preg_replace_callback — 执行一个正则表达式搜索并且使用一个回调进行替换
$tmplContent = preg_replace_callback('/<!--###literal(\d+)###-->/is', array($this, 'restoreLiteral'), $tmplContent);


// 添加安全代码
$tmplContent = '<?php if (!defined(\'THINK_PATH\')) exit();?>'.$tmplContent;// 这里的 添加了 框架里面的安全代码
// 优化生成的php代码
$tmplContent = str_replace('?><?php','',$tmplContent); // 删除那种 在一起的页面

// 模版编译过滤标签
Hook::listen('template_filter',$tmplContent);// 监听,有什么意思吗,这个就是一个,记录的运行流程。
return strip_whitespace($tmplContent); // 删除了 里面的空格
}
//总结: 第一编译标签
// 正则进行替换
// 安全代码 优化php 代码,外加 监听模版编译,去掉空白

/**
* 模板解析入口
* 支持普通标签和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); // 解析模版里面的 这个 include 标签,变成php 里面的 include
// 检查PHP语法
$content = $this->parsePhp($content);// 解析模版里的 默认的php标签
// 首先替换literal标签内容
// 替换这种标签了,嘿嘿,搞不懂啦
$content = preg_replace_callback('/'.$begin.'literal'.$end.'(.*?)'.$begin.'\/literal'.$end.'/is', array($this, 'parseLiteral'),$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); // 标签整理 1
}
}
}
// 预先加载的标签库 无需在每个模板中使用taglib标签加载 但必须使用标签库XML前缀
if(C('TAGLIB_PRE_LOAD')) { // 这里的 一个解析了 特殊判读
$tagLibs = explode(',',C('TAGLIB_PRE_LOAD'));
foreach ($tagLibs as $tag){
$this->parseTagLib($tag,$content); // 标签整理 2
}
}
// 内置标签库 无需使用taglib标签导入就可以使用 并且不需使用标签库XML前缀
$tagLibs = explode(',',C('TAGLIB_BUILD_IN'));// 解析标签库
foreach ($tagLibs as $tag){
$this->parseTagLib($tag,$content,true); // 标签整理 3
}
//解析普通模板标签 {$tagName}
$content = preg_replace_callback('/('.$this->config['tmpl_begin'].')([^\d\w\s'.$this->config['tmpl_begin'].$this->config['tmpl_end'].'].+?)('.$this->config['tmpl_end'].')/is', array($this, 'parseTag'),$content);
return $content;
}// 把里面各种需要解析的东西,按分类进行 解析。
// 剩下的就是 那个 各种仔细的 解析

















// 检查PHP语法
protected function parsePhp($content) {
if(ini_get('short_open_tag')){// 如果开启了短标签,
// <?
// <?php
// <?php=
// <?php
// 开启短标签的情况要将<?标签用echo方式输出 否则无法正常输出xml标识
$content = preg_replace('/(<\?(?!php|=|$))/i', '<?php echo \'\\1\'; ?>'."\n", $content );
}
// PHP语法检查
// 'TMPL_DENY_PHP' => false, // 默认模板引擎是否禁用PHP原生代码
if(C('TMPL_DENY_PHP') && false !== strpos($content,'<?php')) {
E(L('_NOT_ALLOW_PHP_'));
}
// 如果禁止使用 Php 模版,并且 还发现了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标签
// 解析 include 里面的标签
protected function parseInclude($content, $extend = true) {
// 解析继承
if($extend)
$content = $this->parseExtend($content);// 处理了 个 拓展
// 解析布局
$content = $this->parseLayout($content); // 替换到 里面的 layout 标签
// 读取模板中的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,$extend),$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', array($this, 'parseBlock'),$content);
// 读取继承模板
$array = $this->parseXmlAttrs($matches[1]);
$content = $this->parseTemplateName($array['name']);
$content = $this->parseInclude($content, false); //对继承模板中的include进行分析
// 替换block标签
$content = $this->replaceBlock($content);
}else{
$content = preg_replace_callback('/'.$begin.'block\sname=[\'"](.+?)[\'"]\s*?'.$end.'(.*?)'.$begin.'\/block'.$end.'/is', function($match){return stripslashes($match[2]);}, $content);
}
return $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)
E(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(is_array($content)) $content = $content[1];
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) {
if(is_array($tag)) $tag = $tag[1];
// 还原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 = '') {
if(is_array($name)){
$content = $name[2];
$name = $name[1];
}
$this->block[$name] = $content;
return '';
}// 就是个解析了

/**
* 替换继承模板中的block标签
* @access private
* @param string $content 模板内容
* @return string
*/
private function replaceBlock($content){
static $parse = 0;
$begin = $this->config['taglib_begin'];
$end = $this->config['taglib_end'];
$reg = '/('.$begin.'block\sname=[\'"](.+?)[\'"]\s*?'.$end.')(.*?)'.$begin.'\/block'.$end.'/is';
if(is_string($content)){ // 这个是个字符串
do{
$content = preg_replace_callback($reg, array($this, 'replaceBlock'), $content);
} while ($parse && $parse--);
return $content;
} elseif(is_array($content)){ // 居然是个 数组
if(preg_match('/'.$begin.'block\sname=[\'"](.+?)[\'"]\s*?'.$end.'/is', $content[3])){ //存在嵌套,进一步解析
$parse = 1;
$content[3] = preg_replace_callback($reg, array($this, 'replaceBlock'), "{$content[3]}{$begin}/block{$end}");
return $content[1] . $content[3];
} else {
$name = $content[2];
$content = $content[3];
$content = isset($this->block[$name]) ? $this->block[$name] : $content;
return $content;
}
}
}// 就是个解析了

/**
* 搜索模板页面中包含的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 boolean $hide 是否隐藏标签库前缀
* @return string
*/
public function parseTagLib($tagLib,&$content,$hide=false) {
$begin = $this->config['taglib_begin'];// 标签开始 位置标记
$end = $this->config['taglib_end'];// 标签结束 位置标记
if(strpos($tagLib,'\\')){// 如果存在 标签库命名 说白了,就是 有了
// 支持指定标签库的命名空间
$className = $tagLib;// 转换了 存储了一下
$tagLib = substr($tagLib,strrpos($tagLib,'\\')+1);
}else{
$className = 'Think\\Template\TagLib\\'.ucwords($tagLib);
}// 找到要解析的标签库的名称
$tLib = \Think\Think::instance($className);// 实例化对应的类名
$that = $this;
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.']*)';
$this->tempVar = array($tagLib, $tag);

if (!$closeTag){ // 非闭合标签
$patterns = '/'.$begin.$parseTag.$n1.'\/(\s*?)'.$end.'/is';
$content = preg_replace_callback($patterns, function($matches) use($tLib,$tag,$that){
return $that->parseXmlTag($tLib,$tag,$matches[1],$matches[2]);
},$content);
}else{ // 闭合标签
$patterns = '/'.$begin.$parseTag.$n1.$end.'(.*?)'.$begin.'\/'.$parseTag.'(\s*?)'.$end.'/is';
for($i=0;$i<$level;$i++) {
$content=preg_replace_callback($patterns,function($matches) use($tLib,$tag,$that){
return $that->parseXmlTag($tLib,$tag,$matches[1],$matches[2]);
},$content);
}
}
}
}
}

/**
* 解析标签库的标签
* 需要调用对应的标签库文件解析类
* @access public
* @param object $tagLib 标签库对象实例
* @param string $tag 标签名
* @param string $attr 标签属性
* @param string $content 标签内容
* @return string|false
*/
public function parseXmlTag($tagLib,$tag,$attr,$content) {
if(ini_get('magic_quotes_sybase'))
$attr = str_replace('\"','\'',$attr); // 把单引号
$parse = '_'.$tag;
$content = trim($content);
$tags = $tagLib->parseXmlAttr($attr,$tag);
return $tagLib->$parse($tags,$content);
}// 解析掉 对应

/**
* 模板标签解析
* 格式: {TagName:args [|content] }
* @access public
* @param string $tagStr 标签内容
* @return string
*/
public function parseTag($tagStr){
if(is_array($tagStr)) $tagStr = $tagStr[2];// 如果 是个
//if (MAGIC_QUOTES_GPC) {
$tagStr = stripslashes($tagStr);// 删除反斜杠:
//}
$flag = substr($tagStr,0,1);// 去掉了 标志位1
$flag2 = substr($tagStr,1,1); // 标志位2
$name = substr($tagStr,1); // 从 第一位开始
if('$' == $flag && '.' != $flag2 && '(' != $flag2){ //解析模板变量 格式 {$varName}
// 类似于这种
return $this->parseVar($name); // 或者这种 普通的 标签之类的
}elseif('-' == $flag || '+'== $flag){ // 输出计算 +a
return '<?php echo '.$flag.$name.';?>';
}elseif(':' == $flag){ // 输出某个函数的结果 这个找到了 {:U()}
return '<?php echo '.$name.';?>';
}elseif('~' == $flag){ // 执行某个函数 {}
return '<?php '.$name.';?>';// 这里作为普通函数进行输出一下
}elseif(substr($tagStr,0,2)=='//' || (substr($tagStr,0,2)=='/*' && substr(rtrim($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); // trim() 函数移除字符串两侧的空白字符或其他预定义字符。
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.root
// 所有以Think.打头的以特殊变量对待 无需模板赋值就可以输出
$name = $this->parseThinkVar($var); // 就是解析完成的了
}elseif( false !== strpos($var,'.')) { // {$data.id}
//支持 {$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')); //'TMPL_DENY_FUNC_LIST' => 'echo,exit', // 模板引擎禁用函数
for($i=0;$i<$length ;$i++ ){
$args = explode('=',$varArray[$i],2); // 进行 输入 参数的 过滤
//模板函数过滤
$fun = trim($args[0]); // 这个是函数了
switch($fun) { // 这种的 自己东西
case 'default': // 特殊模板函数 哦哦想起来了这个,就是 那个 默认参数那个了 进行 三元运算符开始了
$name = '(isset('.$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(),$extend){
// 分析模板文件名并读取内容
$parseStr = $this->parseTemplateName($tmplPublicName);
// 这里的 外包了
// 替换变量
foreach ($vars as $key=>$val) {
$parseStr = str_replace('['.$key.']',$val,$parseStr);
}
// // 这里的 外包了 进行替换了
// 再次对包含文件进行模板分析
return $this->parseInclude($parseStr,$extend);
// 返回了
}

/**
* 分析加载的模板文件并读取内容 支持多个模板文件读取
* @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(empty($templateName)) continue;
if(false === strpos($templateName,$this->config['template_suffix'])) {
// 解析规则为 模块@主题/控制器/操作
$templateName = T($templateName);
}
// 获取模板文件内容
$parseStr .= file_get_contents($templateName);
}
return $parseStr;
} // 各种神器的控制
}