model = new ParserModel(); } public function _empty() { _404('您访问的地址有误,请核对后重试!'); } // 解析全局前置公共标签 public function parserBefore($content) { // 处理模板中不需要解析的标签 $content = $this->savePreLabel($content); $content = $this->parserSingleLabel($content); // 单标签解析 $content = $this->parserUserLabel($content); // 自定义标签 return $content; } // 解析全局后置公共标签 public function parserAfter($content) { // 默认页面信息替换 $content = str_replace('{pboot:pagetitle}', $this->config('other_title') ?: '{pboot:sitetitle}-{pboot:sitesubtitle}', $content); $content = str_replace('{pboot:pagekeywords}', '{pboot:sitekeywords}', $content); $content = str_replace('{pboot:pagedescription}', '{pboot:sitedescription}', $content); $content = str_replace('{pboot:keyword}', get('keyword', 'vars'), $content); // 当前搜索的关键字 $content = $this->parserSiteLabel($content); // 站点标签 $content = $this->parserCompanyLabel($content); // 公司标签 $content = $this->parserMemberLabel($content); // 会员标签 $content = $this->parserNavLabel($content); // 分类列表 $content = $this->parserCityListLabel($content); // 解析城市分站标签 $content = $this->parserSelectAllLabel($content); // CMS筛选全部标签解析 $content = $this->parserSelectLabel($content); // CMS筛选标签解析 $content = $this->parserSpecifySortLabel($content); // 指定分类 $content = $this->parserListLabel($content); // 指定列表 $content = $this->parserSpecifyContentLabel($content); // 指定内容 $content = $this->parserContentPicsLabel($content); // 内容多图 $content = $this->parserContentCheckboxLabel($content); // 内容多选调取 $content = $this->parserContentTagsLabel($content); // 内容tags调取 $content = $this->parserSlideLabel($content); // 幻灯片 $content = $this->parserLinkLabel($content); // 友情链接 $content = $this->parserMessageLabel($content); // 留言板 $content = $this->parserFormLabel($content); // 自定义表单 $content = $this->parserSubmitFormLabel($content); // 自定义表单提交 $content = $this->parserSqlListLabel($content); // 自定义SQL输出 $content = $this->parserQrcodeLabel($content); // 二维码生成 $content = $this->parserPageLabel($content); // CMS分页标签解析(需置后) $content = $this->parserIfLabel($content); // IF语句(需置最后) $content = $this->parserLoopLabel($content); // LOOP语句(需置后,不可放到if前面,否则有安全风险) $content = $this->restorePreLabel($content); // 还原不需要解析的内容 $content = $this->parserReplaceKeyword($content); // 页面关键词替换 $content = $this->parserTitleReplaceLabel($content); // 通用内容替换标签 这里是我新增的 $content = $this->parserBLock($content); //自定义方法2 $content = $this->parserForeachLabel($content); // 指定随意内容按条件遍 // 解析个人扩展标签,升级不覆盖 if (file_exists(APP_PATH . '/home/controller/ExtLabelController.php')) { if (class_exists('app\home\controller\ExtLabelController')) { $extlabel = new ExtLabelController(); $content = $extlabel->run($content); } } return $content; } // 解析指定随意内容遍历,支持设定分隔符 @pbhtml /* * foreach标签无法匹配成功 * 模板代码不要有php原生 因为无法与标签传值 * 复制link标签 加上city_id 的手法 就可以了 */ public function parserForeachLabel($content) { $pattern = '/{pboot:foreach(s+[^}]+)?}([sS]*?){/pboot:foreach}/'; $pattern2 = '/[foreach:([w]+)(s+[^]]+)?]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i<$count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $str = ''; $char = ','; if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'str': //var_dump($value); $str = $value; break; case 'char': if ($value) $char = $value; break; } } // 无数据直接替换为空并跳过 if (! $str) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; //$arr是数组 可以调用自定义函数或模型方法 一般直接用sql $arr = explode($char,$str); foreach ($arr as $value) { // 按查询图片条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j<$count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'name': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value), $one_html); break; } } $key ++; $out_html .= $one_html; if (isset($num) && $key > $num) { unset($num); break; } } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 通用内容替换标签 @mk-title_replace //$content这里用的时文章标题,直接是字符串 public function parserTitleReplaceLabel($content) { $pattern = '/\{pboot:titlereplace(\s+[^}]+)?\}/'; //先对$content 进行处理 包含 {xx} //{pboot:titlereplace title='{xx}心理咨询,{xx}心理医生,{xx}心理咨询中心机构【乐达心理】' /} //因为{xx} 正则问题 $content = str_replace('{xx}', '[xx]', $content); //var_dump($content);die; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { //var_dump($matches[0][$i]);die; $matches[0][$i] = str_replace('[xx]', '{xx}', $matches[0][$i]); $params = $this->parserParam($matches[0][$i]); $data = ''; foreach ($params as $key => $value) { switch ($key) { case 'title'://这里其实可以解析很多的,不只是title,可以根据case进行不同的解析 $data = $value; // 获取到的文章title //$data = titlereplace($data); //testreplace方法为自定义方法,在\apps\common\function.php里 //$data = str_replace("`", '', $data); //var_dump($data);die; $data = explode(',',$data); $data = $data[0]; break; } } if (!$data) { // 无内容不解析 continue; } //$content = str_replace($matches[0][$i], $data, $content); //$content {pboot:titlereplace title='广州心理咨询,广州心理医生,广州心理咨询中心机构【乐达心理】' /} //$data '广州 //$matches[0][$i] {pboot:titlereplace title='广州 //$content = str_replace("`/}", '', $content); $content = str_replace('[xx]', '{xx}', $content); //var_dump($content);die; $content = str_replace($matches[0][$i], $data, $content); } if((int)$count === 0){ $content = str_replace('[xx]', '{xx}', $content); } }else{ $content = str_replace('[xx]', '{xx}', $content); } return $content; } /* * {pboot:block id='{pboot:httpurl}' /} */ public function parserBLock($content){ $pattern = '/\{pboot:block(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[0][$i]); $data = ''; foreach ($params as $key => $value) { switch ($key) { case 'id': //var_dump($value); //$value里面存在{xx} 就需要特殊处理 //解析顺序不同 与eyoucms的模板机制 区别 $data = $this->model->getBlockContent($value); //return $data['value']; //print_r($data);exit(); break; /*$data = $this->model->getContent($value)->content; // 获取到的文章id去获取文章内容 $data = testreplace($data);*/ } } if (!$data) { // 无内容不解析 continue; } //$content = $data['value']; $content = str_replace($matches[0][$i], $data, $content); } } return $content; } // 保存保留内容 public function savePreLabel($content) { $pattern = '/\{pboot:pre}([\s\S]*?)\{\/pboot:pre\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $this->pre[] = $matches[1][$i]; end($this->pre); $content = str_replace($matches[0][$i], '#pre:' . key($this->pre) . '#', $content); } } return $content; } // 还原保留内容 public function restorePreLabel($content) { $pattern = '/\#pre:([0-9]+)\#/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $content = str_replace($matches[0][$i], $this->pre[$matches[1][$i]], $content); } } $content = str_replace('pboot@if', 'pboot:if', $content); // 还原系统解析if标签 return $content; } // 解析单标签 public function parserSingleLabel($content) { $content = str_replace('{pboot:ucenter}', Url::home('member/ucenter'), $content); // 用户中心 if (! ! $url = get("backurl")) { // 获取会跳地址 $content = str_replace('{pboot:login}', Url::home('member/login', null, "backurl=" . urlencode($url)), $content); // 登录地址 } else { $content = str_replace('{pboot:login}', Url::home('member/login'), $content); // 登录地址 } $content = str_replace('{pboot:register}', Url::home('member/register'), $content); // 注册地址 $content = str_replace('{pboot:retrieve}', Url::home('member/retrieve'), $content); //找回密码 $content = str_replace('{pboot:isregister}', Url::home('member/isRegister'), $content); // 检查是否注册地址 $content = str_replace('{pboot:umodify}', Url::home('member/umodify'), $content); // 修改资料地址 $content = str_replace('{pboot:logout}', Url::home('member/logout'), $content); // 推出登录 $content = str_replace('{pboot:upload}', Url::home('member/upload'), $content); // 上传资料 if (strpos($content, '{pboot:sendemail}')) { session('sendemail', true); // 避免非法外部提交 $content = str_replace('{pboot:sendemail}', Url::home('member/sendEmail'), $content); // 上传资料 } else { session('sendemail', false); } $content = str_replace('{pboot:islogin}', session('pboot_uid') ? 1 : 0, $content); // 是否登录 if (strpos($content, '{pboot:mustlogin}') !== false) { $content = str_replace('{pboot:mustlogin}', '', $content); if (! session('pboot_uid')) { // 没有经登录 if ($this->config('login_no_wait')) { location(Url::home('member/login', null, "backurl=" . urlencode(get_current_url()))); } else { error('您的权限不足,无法浏览本页面!', Url::home('member/login', null, "backurl=" . urlencode(get_current_url()))); } } } $content = str_replace('{pboot:msgaction}', Url::home('message'), $content); // 留言提交路径 $content = str_replace('{pboot:scaction}', Url::home('search'), $content); // 搜索提交路径 $content = str_replace('{pboot:msgcodestatus}', $this->config('message_check_code') === '0' ? 0 : 1, $content); // 是否开留言启验证码 $content = str_replace('{pboot:formcodestatus}', $this->config('form_check_code') === '0' ? 0 : 1, $content); // 是否开启表单验证码 $content = str_replace('{pboot:checkcode}', CORE_DIR . '/code.php', $content); // 验证码路径 $content = str_replace('{pboot:lgpath}', Url::get('home/Do/area'), $content); // 多语言切换前置路径,如{pboot:lgpath}?lg=cn $content = str_replace('{pboot:appid}', $this->config('api_appid'), $content); // API认证用户 $content = str_replace('{pboot:timestamp}', time(), $content); // 认证时间戳 $content = str_replace('{pboot:signature}', md5(md5($this->config('api_appid') . $this->config('api_secret') . time())), $content); // API认证密钥 $content = str_replace('{pboot:httpurl}', get_http_url(), $content); // 当前访问的域名地址 $content = str_replace('{pboot:pageurl}', get_current_url(), $content); // 当前页面的地址 $content = str_replace('{pboot:registercodestatus}', $this->config('register_check_code') === '0' ? 0 : ($this->config('register_check_code') ?: 1), $content); // 是否开启注册验证码 $content = str_replace('{pboot:logincodestatus}', $this->config('login_check_code') === '0' ? 0 : 1, $content); // 是否开启评论验证码 $content = str_replace('{pboot:commentcodestatus}', $this->config('comment_check_code') === '0' ? 0 : 1, $content); // 是否开启评论验证码 $content = str_replace('{pboot:commentaction}', Url::home('comment/add', null, "contentid={content:id}"), $content); // 评论提交路径 $content = str_replace('{pboot:mycommentpage}', Url::home('comment/my'), $content); // 我的评论 $content = str_replace('{pboot:registerstatus}', $this->config('register_status') === '0' ? 0 : 1, $content); // 是否开启注册 $content = str_replace('{pboot:loginstatus}', $this->config('login_status') === '0' ? 0 : 1, $content); // 是否开启登录 $content = str_replace('{pboot:commentstatus}', $this->config('comment_status') === '0' ? 0 : 1, $content); // 是否开启评论 // 记录蜘蛛爬行 if ($this->config('spiderlog') !== '0') { if ($this->config('tpl_html_cache')) { // 缓存时插入script,否则直接执行 $spidercode = ""; $content = preg_replace('/(<\/body>)/i', $spidercode . "\n$1", $content); } else { $spider = new SpiderController(URL); $spider->index(); } } // 生成分站首页 $city_suffix = $this->config('city_suffix')?true:false; //分站后缀 $content = str_replace('{zong}', Url::home('home/Index/', $city_suffix), $content); // {zong} $content = str_replace('{iscity}', cookie('city') ? 1 : 0, $content); // 用来判断当前是否是分站 return $content; } // 解析站点标签 public function parserSiteLabel($content) { $pattern = '/\{pboot:site([\w]+)(\s+[^}]+)?\}/'; $data = array(); // 页面自适应标题避免多横线 if (preg_match($pattern, $content)) { $data = $this->model->getSite(); if (! $data->subtitle) { $content = str_replace('{pboot:sitetitle}-{pboot:sitesubtitle}', '{pboot:sitetitle}', $content); } } if (preg_match_all($pattern, $content, $matches)) { $data = $data ?: $this->model->getSite(); $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { case 'index': $content = str_replace($matches[0][$i], Url::home('home/Index/'), $content); break; case 'path': $content = str_replace($matches[0][$i], SITE_DIR, $content); break; case 'enter': $content = str_replace($matches[0][$i], SITE_INDEX_DIR, $content); break; case 'logo': if (isset($data->logo) && $data->logo) { if (! preg_match('/^http/', $data->logo)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, SITE_DIR . $data->logo), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->logo), $content); } } else { $content = str_replace($matches[0][$i], STATIC_DIR . '/images/logo.png', $content); } break; case 'tplpath': $content = str_replace($matches[0][$i], APP_THEME_DIR, $content); break; case 'language': $content = str_replace($matches[0][$i], get_lg(), $content); break; case 'statistical': if (isset($data->statistical)) { $content = str_replace($matches[0][$i], decode_string($data->statistical), $content); } else { $content = str_replace($matches[0][$i], '', $content); } case 'copyright': if (isset($data->copyright)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, decode_string($data->copyright)), $content); } else { $content = str_replace($matches[0][$i], '', $content); } default: if (strpos(file_get_contents(CORE_PATH . base64_decode('L2Jhc2ljL0tlcm5lbC5waHA=')), base64_decode('S2VybmVs'))) exit(); if (isset($data->{$matches[1][$i]})) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->{$matches[1][$i]}), $content); } else { $content = str_replace($matches[0][$i], '', $content); } } } } return $content; } // 解析公司标签 public function parserCompanyLabel($content) { $pattern = '/\{pboot:company([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $data = $this->model->getCompany(); $count = count($matches[0]); // $city = cookie('city'); $city_info = $this->config('citys')[$city]; for ($i = 0; $i < $count; $i ++) { if (! $data) { // 无数据时直接替换为空 $content = str_replace($matches[0][$i], '', $content); continue; } $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { case 'weixin': if (isset($data->weixin) && $data->weixin) { if (! preg_match('/^http/', $data->weixin)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, SITE_DIR . $data->weixin), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->weixin), $content); } } else { $content = str_replace($matches[0][$i], '', $content); } break; case 'contact': case 'mobile': case 'phone': case 'fax': case 'email': case 'qq': case 'address': if( $city_info[$matches[1][$i]] ){ $data->{$matches[1][$i]} = $city_info[$matches[1][$i]]; } default: if (isset($data->{$matches[1][$i]})) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->{$matches[1][$i]}), $content); } } } } return $content; } // 解析自定义标签 public function parserUserLabel($content) { $pattern = '/\{label:([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $data = $this->model->getLabel(); $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { if (! $data) { // 无数据时直接替换为空 $content = str_replace($matches[0][$i], '', $content); continue; } $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { default: if (isset($data[$matches[1][$i]])) { if ($data[$matches[1][$i]]['type'] == 3 && $data[$matches[1][$i]]['value']) { if (! preg_match('/^http/', $data[$matches[1][$i]]['value'])) { $data[$matches[1][$i]]['value'] = $this->adjustLabelData($params, SITE_DIR . $data[$matches[1][$i]]['value']); } else { $data[$matches[1][$i]]['value'] = $this->adjustLabelData($params, $data[$matches[1][$i]]['value']); } } $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data[$matches[1][$i]]['value']), $content); } } } } return $content; } // 会员标签解析 private function parserMemberLabel($content) { $pattern = '/\{user:([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); $model = new MemberModel(); $data = $model->getUser(); for ($i = 0; $i < $count; $i ++) { // 无数据直接替换并跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { case 'password': // 密码不允许显示 $content = str_replace($matches[0][$i], '', $content); break; case 'registertime': $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->register_time), $content); break; case 'logincount': $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->login_count), $content); break; case 'lastloginip': $content = str_replace($matches[0][$i], $this->adjustLabelData($params, long2ip($data->last_login_ip)), $content); break; case 'lastlogintime': $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->last_login_time), $content); break; case 'headpic': if ($data->headpic) { if (! preg_match('/^http/', $data->headpic)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, SITE_DIR . $data->headpic), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->headpic), $content); } } else { $content = str_replace($matches[0][$i], SITE_DIR . '/apps/admin/view/default/images/logo.png', $content); } default: if (isset($data->{$matches[1][$i]})) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $data->{$matches[1][$i]}), $content); } else { $content = str_replace($matches[0][$i], '', $content); } } } } return $content; } // 解析栏目列表标签 public function parserNavLabel($content) { $pattern = '/\{pboot:nav(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:nav\}/'; $pattern2 = '/\[nav:([\w]+)(\s+[^]]+)?\]/'; $pattern3 = '/pboot:([0-9])+nav/'; if (preg_match_all($pattern, $content, $matches)) { $data = $this->model->getSortsTree(); $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 无数据时直接替换整体标签为空 if (! $data['tree']) { $content = str_replace($matches[0][$i], '', $content); continue; } // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $parent = 0; $num = 0; $scode = 0; $scode_arr = array(); foreach ($params as $key => $value) { switch ($key) { case 'parent': $parent = $value; break; case 'num': $num = $value; break; case 'scode': $scode = $value; $scode_arr = explode(',', $scode); break; } } if ($parent) { // 非顶级栏目起始,调用子栏目 $parent_arr = explode(',', $parent); $out_data = array(); foreach ($parent_arr as $vp) { if (isset($data['tree'][trim($vp)]['son'])) { $out_data = array_merge($out_data, $data['tree'][trim($vp)]['son']); } } } else { // 顶级栏目起始 $out_data = $data['top']; } // 读取指定数量 if ($num) { $out_data = array_slice($out_data, 0, $num); } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($out_data as $value) { // 按查询的数据条数循环 if ($scode_arr && ! in_array($value['scode'], $scode_arr)) { continue; } $one_html = $matches[2][$i]; if ($count2) { for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'link': if ($value['outlink']) { $one_html = str_replace($matches2[0][$j], $value['outlink'], $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->parserLink($value['type'], $value['urlname'], 'list', $value['scode'], $value['filename'], '', ''), $one_html); } break; case 'soncount': if (isset($data['tree'][$value['scode']]['son'])) { $one_html = str_replace($matches2[0][$j], count($data['tree'][$value['scode']]['son']), $one_html); } else { $one_html = str_replace($matches2[0][$j], 0, $one_html); } break; case 'rows': $one_html = str_replace($matches2[0][$j], $this->model->getSortRows($value['scode']), $one_html); break; case 'ico': if ($value['ico']) { if (! preg_match('/^http/', $value['ico'])) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value['ico']), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value['ico']), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; case 'pic': if ($value['pic']) { if (! preg_match('/^http/', $value['pic'])) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value['pic']), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value['pic']), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; default: if (isset($value[$matches2[1][$j]])) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value[$matches2[1][$j]]), $one_html); } } } } $key ++; $out_html .= $one_html; } // 无限极嵌套解析 if (preg_match($pattern3, $out_html, $matches3)) { $out_html = str_replace('pboot:' . $matches3[1] . 'nav', 'pboot:nav', $out_html); $out_html = str_replace('[' . $matches3[1] . 'nav:', '[nav:', $out_html); $out_html = $this->parserNavLabel($out_html); } // 执行内容替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析当前位置 public function parserPositionLabel($content, $scode, $page = null, $link = null) { $pattern = '/\{pboot:position(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); $data = $this->model->getPosition($scode); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[1][$i], false); // 保留对html标签的支持 $separator = ''; $separatoricon = ''; $indextext = ''; $indexicon = ''; // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'separator': $separator = $value; break; case 'separatoricon': $separatoricon = $value; break; case 'indextext': $indextext = $value; break; case 'indexicon': $indexicon = $value; break; } } // 已经设置图标,则图标优先,如果没有,则判断是否已经设置文字 if ($separatoricon) { $separator = ' '; } elseif (! $separator) { $separator = ' >> '; } if ($indexicon) { $indextext = ''; } elseif (! $indextext) { $indextext = '首页'; } $out_html = '' . $indextext . ''; if ($page && $scode == 0) { $out_html .= $separator . '' . $page . ''; } else { foreach ($data as $key => $value) { if ($value['outlink']) { $out_html .= $separator . '' . $value['name'] . ''; } else { $out_html .= $separator . '' . $value['name'] . ''; } } } // 执行内容替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析当前分类标签 public function parserSortLabel($content, $sort) { $pattern = '/\{sort:([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { case 'link': if ($sort->outlink) { $content = str_replace($matches[0][$i], $sort->outlink, $content); } else { $content = str_replace($matches[0][$i], $this->parserLink($sort->type, $sort->urlname, 'list', $sort->scode, $sort->filename, '', ''), $content); } break; case 'tcode': // 顶级栏目ID if (! isset($tcode)) $tcode = $this->model->getSortTopScode($sort->scode); $content = str_replace($matches[0][$i], $tcode, $content); break; case 'topname': if (! isset($tcode)) $tcode = $this->model->getSortTopScode($sort->scode); $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $this->model->getSortName($tcode)), $content); break; case 'toplink': if (! isset($tcode)) { $tcode = $this->model->getSortTopScode($sort->scode); } $top_sort = $this->model->getSort($tcode); if ($top_sort->outlink) { $toplink = $top_sort->outlink; } else { $toplink = $this->parserLink($top_sort->type, $top_sort->urlname, 'list', $top_sort->scode, $top_sort->filename, '', ''); } $content = str_replace($matches[0][$i], $toplink, $content); break; case 'parentname': if ($sort->pcode == 0) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->name), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->parentname), $content); } break; case 'parentlink': if ($sort->pcode == 0) { $parent_sort = $sort; } else { $parent_sort = $this->model->getSort($sort->pcode); } if ($parent_sort->outlink) { $parentlink = $top_sort->outlink; } else { $parentlink = $this->parserLink($parent_sort->type, $parent_sort->urlname, 'list', $parent_sort->scode, $parent_sort->filename, '', ''); } $content = str_replace($matches[0][$i], $parentlink, $content); break; case 'toprows': if (! isset($tcode)) $tcode = $this->model->getSortTopScode($sort->scode); $content = str_replace($matches[0][$i], $this->model->getSortRows($tcode), $content); break; case 'parentrows': if ($sort->pcode == 0) { $content = str_replace($matches[0][$i], $this->model->getSortRows($sort->scode), $content); } else { $content = str_replace($matches[0][$i], $this->model->getSortRows($sort->pcode), $content); } break; case 'rows': $content = str_replace($matches[0][$i], $this->model->getSortRows($sort->scode), $content); break; case 'ico': if ($sort->ico) { if (! preg_match('/^http/', $sort->ico)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, SITE_DIR . $sort->ico), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->ico), $content); } } else { $content = str_replace($matches[0][$i], '', $content); } break; case 'pic': if ($sort->pic) { if (! preg_match('/^http/', $sort->pic)) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, SITE_DIR . $sort->pic), $content); } else { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->pic), $content); } } else { $content = str_replace($matches[0][$i], '', $content); } break; case 'keywords': // 如果栏目关键字为空,则自动使用全局关键字 if ($sort->keywords) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->keywords), $content); } else { $content = str_replace($matches[0][$i], '{pboot:sitekeywords}', $content); } break; case 'description': // 如果栏目描述为空,则自动使用全局描述 if ($sort->description) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->description), $content); } else { $content = str_replace($matches[0][$i], '{pboot:sitedescription}', $content); } break; default: if (isset($sort->{$matches[1][$i]})) { $content = str_replace($matches[0][$i], $this->adjustLabelData($params, $sort->{$matches[1][$i]}), $content); } else { $content = str_replace($matches[0][$i], '', $content); } } } } return $content; } // 解析非列表页分类标签 public function parserSpecialPageSortLabel($content, $id, $page, $link) { $pattern = '/\{sort:([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[2][$i]); switch ($matches[1][$i]) { case 'tcode': // 顶级栏目ID $content = str_replace($matches[0][$i], $id, $content); break; case 'topname': $content = str_replace($this->adjustLabelData($params, $matches[0][$i]), $page, $content); break; case 'toplink': $content = str_replace($matches[0][$i], $link, $content); break; case 'pcode': // 父栏目ID $content = str_replace($matches[0][$i], $id, $content); break; case 'parentname': $content = str_replace($this->adjustLabelData($params, $matches[0][$i]), $page, $content); break; case 'parentlink': $content = str_replace($matches[0][$i], $link, $content); break; case 'scode': // 当前栏目ID $content = str_replace($matches[0][$i], $id, $content); break; case 'link': $content = str_replace($matches[0][$i], $link, $content); break; case 'name': // 当前分类名称 $content = str_replace($this->adjustLabelData($params, $matches[0][$i]), $page, $content); break; case 'keywords': // 当前分类关键字,使用全局 $content = str_replace($this->adjustLabelData($params, $matches[0][$i]), '{pboot:sitekeywords}', $content); break; case 'description': // 当前分类描述,使用全局 $content = str_replace($this->adjustLabelData($params, $matches[0][$i]), '{pboot:sitedescription}', $content); break; default: $content = str_replace($matches[0][$i], '', $content); } } } return $content; } // 解析指定分类标签 public function parserSpecifySortLabel($content) { $pattern = '/\{pboot:sort(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:sort\}/'; $pattern2 = '/\[sort:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $scode = - 1; // 跳过未指定scode的列表 if (! array_key_exists('scode', $params)) { continue; } if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 分离分类编码 foreach ($params as $key => $value) { switch ($key) { case 'scode': $scode = $value; break; } } if (! $scode) { $scode = - 1; } // 读取一个或多个栏目数据 $data = $this->model->getMultSort(escape_string($scode)); // 无数据直接跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'link': if ($value->outlink) { $one_html = str_replace($matches2[0][$j], $value->outlink, $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->parserLink($value->type, $value->urlname, 'list', $value->scode, $value->filename, '', ''), $one_html); } break; case 'ico': if ($value->ico) { if (! preg_match('/^http/', $value->ico)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->ico), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->ico), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; case 'pic': if ($value->pic) { if (! preg_match('/^http/', $value->pic)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->pic), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->pic), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; case 'rows': $one_html = str_replace($matches2[0][$j], $this->model->getSortRows($value->scode), $one_html); // 获取分类包含子类的内容数量 break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } } } $key ++; $out_html .= $one_html; } // 执行替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析筛选全部 public function parserSelectAllLabel($content) { $pattern = '/\{pboot:selectall(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[1][$i]); $text = '全部'; $field = ''; $class = ''; $active = ''; // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'field': $field = $value; break; case 'text': $text = $value; break; case 'class': $class = $value; break; case 'active': $active = $value; break; } } // 跳过不带field的标签 if (! $field) { continue; } $url_rule_type = $this->config('url_rule_type') ?: 3; $url_rule_suffix = $this->config('url_rule_suffix') ?: '.html'; $url_break_char = $this->config('url_break_char') ?: '_'; $url_rule_sort_suffix = '/'; // 附加后缀及参数 if ($url_rule_type == 1 || $url_rule_type == 2) { // 获取地址路径 $url = parse_url(URL); // 避免非根目录首页筛选问题 if (trim($url['path'], '/') == trim(SITE_DIR, '/')) { $url_rule_sort_suffix = '/'; } $path = preg_replace('/\/page\/[0-9]+/', '', $url['path']); // 去除路径方式分页 // 去后缀扩展 if (! ! $pos = strripos($path, $url_rule_suffix)) { $path = substr($path, 0, $pos); } // 去路径分页,回到首页 if (defined('CMS_PAGE_CUSTOM')) { $path = preg_replace('/(.*)' . $url_break_char . '[0-9]+$/', '$1', rtrim($path, '/')); } else { $path = preg_replace('/(.*)(' . $url_break_char . '[0-9]+)' . $url_break_char . '[0-9]+$/', '$1$2', rtrim($path, '/')); } // 拼接地址 $path .= $url_rule_sort_suffix . query_string('p,s,' . $field); } elseif ($url_rule_type == 3) { $output = array(); if (isset($_SERVER["QUERY_STRING"]) && ! ! $qs = $_SERVER["QUERY_STRING"]) { parse_str($qs, $output); unset($output['page']); // 去除字符串方式分页,回到第一页 unset($output['p']); // 去除保留参数 unset($output['s']); // 去除保留参数 unset($output[$field]); // 不筛选该字段 if ($output && ! current($output)) { $path_qs = key($output); // 第一个参数为路径信息,注意PHP数组会自动将点转换下划线 unset($output[$path_qs]); // 去除路径参数 $temp_suffix = substr($url_rule_suffix, 1); if (! ! $pos = strripos($path_qs, '_' . $temp_suffix)) { $path = substr($path_qs, 0, $pos); // 去扩展 } else { $path = $path_qs; } // 去除原分页参数 if (defined('CMS_PAGE_CUSTOM')) { $path = preg_replace('/(.*)' . $url_break_char . '[0-9]+$/', "$1", rtrim($path, '/')); } else { $path = preg_replace('/(.*)(' . $url_break_char . '[0-9]+)' . $url_break_char . '[0-9]+$/', "$1$2", rtrim($path, '/')); } $path = SITE_INDEX_DIR . '/?' . $path . $url_rule_sort_suffix; } else { $path = ''; } $qs = http_build_query($output); if ($path && $qs) { // 重组地址 $path = rtrim($path, '/') . '/&' . $qs; } elseif ($qs) { $path = SITE_INDEX_DIR . '/?' . $qs; } elseif (! $path) { $path = SITE_INDEX_DIR . '/'; } } else { $path = SITE_INDEX_DIR . '/'; } } // 如果有对本字段进行筛选,则不高亮 if (get($field, 'vars')) { $out_html = '' . $text . ''; } else { $out_html = '' . $text . ''; } // 执行内容替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析筛选标签 public function parserSelectLabel($content) { $pattern = '/\{pboot:select(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:select\}/'; $pattern2 = '/\[select:([\w]+)(\s+[^]]+)?\]/'; // 参数处理 if (preg_match($pattern, $content)) { $url_rule_type = $this->config('url_rule_type') ?: 3; $url_rule_suffix = $this->config('url_rule_suffix') ?: '.html'; $url_break_char = $this->config('url_break_char') ?: '_'; $url_rule_sort_suffix = '/'; // 附加后缀及参数 if ($url_rule_type == 1 || $url_rule_type == 2) { // 获取地址路径 $url = parse_url(URL); // 避免非根目录首页筛选问题 if (trim($url['path'], '/') == trim(SITE_DIR, '/')) { $url_rule_sort_suffix = '/'; } $path = preg_replace('/\/page\/[0-9]+/', '', $url['path']); // 去除路径方式分页,回到第一页 // 去后缀扩展 if (! ! $pos = strripos($path, $url_rule_suffix)) { $path = substr($path, 0, $pos); } // 去路径分页,回到首页 if (defined('CMS_PAGE_CUSTOM')) { $path = preg_replace('/(.*)' . $url_break_char . '[0-9]+$/', '$1', rtrim($path, '/')); } else { $path = preg_replace('/(.*)(' . $url_break_char . '[0-9]+)' . $url_break_char . '[0-9]+$/', '$1$2', rtrim($path, '/')); } } $output = array(); if (isset($_SERVER["QUERY_STRING"]) && ! ! $qs = $_SERVER["QUERY_STRING"]) { parse_str($qs, $output); unset($output['page']); // 去除字符串方式分页,回到第一页 unset($output['p']); // 去除保留参数 unset($output['s']); // 去除保留参数 if ($url_rule_type == 3 && $output && ! current($output)) { $path_qs = key($output); // 第一个参数为路径信息,注意PHP数组会自动将点转换下划线 unset($output[$path_qs]); // 去除路径参数 $temp_suffix = substr($url_rule_suffix, 1); if (! ! $pos = strripos($path_qs, '_' . $temp_suffix)) { $path = substr($path_qs, 0, $pos); // 去扩展 } else { $path = $path_qs; } // 去除原分页参数 if (defined('CMS_PAGE_CUSTOM')) { $path = preg_replace('/(.*)' . $url_break_char . '[0-9]+$/', "$1", rtrim($path, '/')); } else { $path = preg_replace('/(.*)(' . $url_break_char . '[0-9]+)' . $url_break_char . '[0-9]+$/', "$1$2", rtrim($path, '/')); } $path = SITE_INDEX_DIR . '/?' . $path; $not_index = true; } } $path = isset($path) ? $path . $url_rule_sort_suffix : SITE_INDEX_DIR . '/'; } // 执行匹配替换 if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $field = ''; // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'field': $field = $value; break; } } // 跳过不带field的标签 if (! $field) { continue; } // 读取数据 if (! ! $data = $this->model->getSelect(escape_string($field))) { $data = explode(',', $data); } else { $data = array(); } // 无数据直接替换为空并跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'value': $one_html = str_replace($matches2[0][$j], $value, $one_html); break; case 'current': $one_html = str_replace($matches2[0][$j], get($field, 'vars'), $one_html); break; case 'link': $qs = $output; // 需使用中间变量,避免多个链接相同问题 $qs[$field] = $value; $qs = http_build_query($qs); if ($url_rule_type == 3 && $not_index) { $link = rtrim($path, '/') . '/&' . $qs; } else { $link = $path . '?' . $qs; } $one_html = str_replace($matches2[0][$j], $link, $one_html); break; } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析内容列表标签 public function parserListLabel($content, $cscode = '') { $pattern = '/\{pboot:list(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:list\}/'; $pattern2 = '/\[list:([\w\+\-\*\/\%]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); // 未设置条数时使用默认15 $order = 'a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; // 默认排序 $filter = ''; // 过滤 $tags = ''; // tag标签 $fuzzy = true; // 设置过滤、tag、筛选是否模糊匹配 $ispics = ''; // 是否多图 $isico = ''; // 是否缩略图 $istop = ''; // 是否置顶 $isrecommend = ''; // 是否推荐 $isheadline = ''; // 是否头条 $start = 1; // 起始条数,默认第一条开始 $lfield = ''; // 查询字段限制 // 判断当前栏目和指定栏目 if ($cscode && ! array_key_exists('scode', $params)) { // 解析当前 $scode = $cscode; $page = true; // 如果未指定分类默认分页 } elseif (! $cscode && array_key_exists('scode', $params)) { // 解析指定 $scode = $params['scode']; $page = false; // 如果指定分类默认不分页 } else { continue; } if ($scode == '*') { $scode = ''; } // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'order': switch ($value) { case 'id': $order = 'a.id DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC'; break; case 'date': $order = 'a.date DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.id DESC'; break; case 'sorting': $order = 'a.sorting ASC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.date DESC,a.id DESC'; break; case 'istop': $order = 'a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'isrecommend': $order = 'a.isrecommend DESC,a.istop DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'isheadline': $order = 'a.isrecommend DESC,a.istop DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'visits': case 'likes': case 'oppose': $order = $value . ' DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'random': // 随机取数 $db_type = get_db_type(); if ($db_type == 'mysql') { $order = "RAND()"; } elseif ($db_type == 'sqlite') { $order = "RANDOM()"; } break; default: if ($value) { $orders = explode(',', $value); foreach ($orders as $k => $v) { if (strpos($v, 'ext_') === 0) { $orders[$k] = 'e.' . $v; } else { $orders[$k] = 'a.' . $v; } } $value = implode(',', $orders); $order = $value . ',a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; } } break; case 'filter': $filter = $value; break; case 'fuzzy': $fuzzy = $value; break; case 'tags': $tags = $value; break; case 'ispics': $ispics = $value; break; case 'isico': $isico = $value; break; case 'istop': $istop = $value; break; case 'isrecommend': $isrecommend = $value; break; case 'isheadline': $isheadline = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'lfield': $lfield = $value; break; } } // filter数据筛选 $where1 = array(); if ($filter) { $filter = explode('|', $filter); if (count($filter) == 2) { $filter_arr = explode(',', $filter[1]); if ($filter[0] == 'title') { $filter[0] = 'a.title'; } foreach ($filter_arr as $value) { if ($value) { if ($fuzzy) { $where1[] = $filter[0] . " like '%" . escape_string($value) . "%'"; } else { $where1[] = $filter[0] . "='" . escape_string($value) . "'"; } } } } } // tags数据参数筛选 $where2 = array(); if ($tags) { $tags_arr = explode(',', $tags); foreach ($tags_arr as $value) { if ($value) { if ($fuzzy) { $where2[] = "a.tags like '%" . escape_string($value) . "%'"; } else { $where2[] = "a.tags='" . escape_string($value) . "'"; } } } } // 重置存储条件 $where3 = array(); // 只对有分页的列表有效 if ($page) { // tags数据传值筛选 if (! ! $get_tag = get('tag', 'vars')) { if ($fuzzy) { $where2[] = "a.tags like '%" . $get_tag . "%'"; } else { $where2[] = "a.tags='" . $get_tag . "'"; } } // 扩展字段数据筛选 foreach ($_GET as $key => $value) { if (preg_match('/^ext_[\w\-]+$/', $key)) { // 其他字段不加入 $where3[$key] = get($key, 'vars'); } } } // 判断多图调节参数 if ($ispics !== '') { if ($ispics) { $where3[] = "a.pics<>''"; } else { $where3[] = "a.pics=''"; } } // 判断缩略图调节参数 if ($isico !== '') { if ($isico) { $where3[] = "a.ico<>''"; } else { $where3[] = "a.ico=''"; } } // 判断置顶调节参数 if ($istop !== '') { if ($istop) { $where3[] = "a.istop=1"; } else { $where3[] = "a.istop=0"; } } // 判断推荐调节参数 if ($isrecommend !== '') { if ($isrecommend) { $where3[] = "a.isrecommend=1"; } else { $where3[] = "a.isrecommend=0"; } } // 判断头条调节参数 if ($isheadline !== '') { if ($isheadline) { $where3[] = "a.isheadline=1"; } else { $where3[] = "a.isheadline=0"; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } if ($page) { if (isset($paging)) { error('请不要在一个页面使用多个具有分页的列表,您可将多余的使用page=0关闭分页!'); } else { $paging = true; $data = $this->model->getLists($scode, $num, $order, $where1, $where2, $where3, $fuzzy, $start, $lfield); } } else { $data = $this->model->getList($scode, $num, $order, $where1, $where2, $where3, $fuzzy, $start, $lfield); } // 无数据直接替换 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); $one_html = $this->parserList($matches2[1][$j], $matches2[0][$j], $one_html, $value, $params, $key); } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析当前内容标签 public function parserCurrentContentLabel($content, $sort, $data) { $pattern = '/\{content:([\w]+)(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 无数据直接替换并跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } $params = $this->parserParam($matches[2][$i]); $content = $this->parserContent($matches[1][$i], $matches[0][$i], $content, $data, $params, $sort); } } // 新增计数代码,非缓存方式,直接计数 if ($this->config('tpl_html_cache')) { if (! isset($this->var['addvisits'])) { $visits = ""; $content = preg_replace('/(<\/body>)/i', $visits . "\n$1", $content); $this->var['addvisits'] = true; } } else { $do = new DoModel(); $do->addVisits($data->id); } return $content; } // 解析指定内容标签,单页支持使用scode调用 public function parserSpecifyContentLabel($content) { $pattern = '/\{pboot:content(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:content\}/'; $pattern2 = '/\[content:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $id = - 1; $scode = - 1; // 跳过未指定id和scode的列表 if (array_key_exists('id', $params)) { $id = $params['id']; $data = $this->model->getContent(escape_string($id)); } elseif (array_key_exists('scode', $params)) { $scode = $params['scode']; $data = $this->model->getAbout(escape_string($scode)); } else { continue; } // 读取数据 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); $out_html = $this->parserContent($matches2[1][$j], $matches2[0][$j], $out_html, $data, $params, $scode); } // 执行替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析指定内容多图 public function parserContentPicsLabel($content) { $pattern = '/\{pboot:pics(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:pics\}/'; $pattern2 = '/\[pics:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $id = - 1; $field = "pics"; if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 跳过未指定id的列表 if (! array_key_exists('id', $params)) { continue; } // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'id': $id = $value; break; case 'num': $num = $value; break; case 'field': $field = $value; break; } } // 读取内容多图 if (! ! $rs = $this->model->getContentPics(escape_string($id), $field)) { $pics = explode(',', $rs->$field); $picstitle = explode(',', $rs->picstitle); } else { $pics = array(); $picstitle = array(); } // 无图直接替换为空并跳过 if (! $pics) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($pics as $vkey => $value) { // 按查询图片条数循环 $one_html = $matches[2][$i]; if (! $value) continue; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'src': if ($value) { if (! preg_match('/^http/', $value)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; case 'title': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, isset($picstitle[$vkey]) ? $picstitle[$vkey] : ''), $one_html); break; default: $one_html = str_replace($matches2[0][$j], '', $one_html); } } $key ++; $out_html .= $one_html; if (isset($num) && $key > $num) { unset($num); break; } } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析指定内容多选 public function parserContentCheckboxLabel($content) { $pattern = '/\{pboot:checkbox(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:checkbox\}/'; $pattern2 = '/\[checkbox:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $id = - 1; if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 跳过未指定id的调用 if (! array_key_exists('id', $params)) { continue; } // 跳过未指定field的调用 if (! array_key_exists('field', $params)) { continue; } // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'id': $id = $value; break; case 'field': $field = $value; break; } } // 读取内容多图 if (! ! $checkboxs = $this->model->getContentCheckbox(escape_string($id), escape_string($field))) { $data = explode(',', $checkboxs); } else { $data = array(); } // 无内容直接替换为空并跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'text': $one_html = str_replace($this->adjustLabelData($params, $matches2[0][$j]), $value, $one_html); break; } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析内容tags public function parserContentTagsLabel($content) { $pattern = '/\{pboot:tags(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:tags\}/'; $pattern2 = '/\[tags:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $id = ''; // 调取指定内容的tags $scode = ''; // 调取指定分类的tags $target = 'list'; // 标签跳转目标,可以是内容列表,也可以是独立tags.html页面 // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'id': $id = $value; break; case 'scode': $scode = $value; break; case 'num': $num = $value; break; case 'target': $target = $value; } } // 获取数据 $data = array(); if ($id) { // 获取单个内容的tags if (strpos($scode, ',') !== false) { error('模板中指定id输出tags时不允许scode指定多个栏目!'); } if (! ! $rs = $this->model->getContentTags(escape_string($id))) { if ($rs->tags) { $tags = explode(',', $rs->tags); $scode = $scode ?: $rs->scode; $sort = $this->model->getSort($scode); // 获取栏目信息 foreach ($tags as $key => $value) { $data[] = array( 'sort' => $sort, 'tags' => $value ); } } } } elseif ($scode) { // 获取指定栏目的tags $scodes = explode(',', $scode); // 多个栏目是分别获取 foreach ($scodes as $key => $value) { $sort = $this->model->getSort($value); // 获取栏目信息 if (! ! $rs = $this->model->getSortTags($value)) { $tags = implode(',', $rs); // 把栏目tags串起来 $tags = array_unique(explode(',', $tags)); // 再把所有tags组成数组并去重 foreach ($tags as $key2 => $value2) { if (! in_array($value2, array_column($data, 'tags'))) { // 避免重复输出 $data[] = array( 'sort' => $sort, 'tags' => $value2 ); } } } } } else { // 全部栏目时候强制标签页形式 $target = 'tag'; if (! ! $rs = $this->model->getSortTags('')) { $tags = implode(',', $rs); // 把栏目tags串起来 $tags = array_unique(explode(',', $tags)); // 再把所有tags组成数组并去重 foreach ($tags as $key2 => $value2) { if (! in_array($value2, array_column($data, 'tags'))) { // 避免重复输出 $data[] = array( 'tags' => $value2 ); } } } } // 无内容直接替换为空并跳过 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'text': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value['tags']), $one_html); break; case 'link': $url_rule_type = $this->config('url_rule_type') ?: 3; if ($target == 'tag') { if ($url_rule_type == 3) { $link = Url::home('tag=' . urlencode($value['tags']), false); } else { $link = Url::home('tag/' . urlencode($value['tags']), false); } } else { $link = $this->parserLink($value['sort']->type, $value['sort']->urlname, 'list', $value['sort']->scode, $value['sort']->filename, '', ''); if ($url_rule_type == 3) { $link = $link . '&tag=' . urlencode($value['tags']); } else { $link = $link . '?tag=' . urlencode($value['tags']); } } $one_html = str_replace($matches2[0][$j], $link, $one_html); break; } } $key ++; $out_html .= $one_html; if (isset($num) && $key > $num) { unset($num); break; } } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析幻灯片标签 public function parserSlideLabel($content) { $pattern = '/\{pboot:slide(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:slide\}/'; $pattern2 = '/\[slide:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $gid = 1; $num = 5; $start = 1; // 跳过未指定gid的标签 if (! array_key_exists('gid', $params)) { continue; } // 分离参数 foreach ($params as $key => $value) { switch ($key) { case 'gid': $gid = $value; break; case 'num': $num = $value; break; case 'start': $start = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getSlides(escape_string($gid), escape_string($num), $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'src': if ($value->pic) { if (! preg_match('/^http/', $value->pic)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->pic), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->pic), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析友情链接标签 public function parserLinkLabel($content) { $pattern = '/\{pboot:link(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:link\}/'; $pattern2 = '/\[link:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $gid = 1; $num = 10; $start = 1; // 跳过未指定gid的标签 if (! array_key_exists('gid', $params)) { continue; } foreach ($params as $key => $value) { switch ($key) { case 'gid': $gid = $value; break; case 'num': $num = $value; break; case 'start': $start = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getLinks(escape_string($gid), escape_string($num), $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'logo': if ($value->logo) { if (! preg_match('/^http/', $value->logo)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->logo), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->logo), $one_html); } } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析留言板标签 public function parserMessageLabel($content) { $pattern = '/\{pboot:message(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:message\}/'; $pattern2 = '/\[message:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); $page = true; $start = 1; $lg = ''; foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'lg': $lg = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getMessage(escape_string($num), $page, $start, $lg)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'ip': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, long2ip($value->user_ip)), $one_html); break; case 'os': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->user_os), $one_html); break; case 'bs': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->user_bs), $one_html); break; case 'askdate': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->create_time), $one_html); break; case 'replydate': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->update_time), $one_html); break; case 'headpic': if ($value->headpic) { if (! preg_match('/^http/', $value->headpic)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->headpic), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->headpic), $one_html); } } else { $one_html = str_replace($matches2[0][$j], SITE_DIR . '/apps/admin/view/default/images/logo.png', $one_html); } break; case 'nickname': if ($value->nickname) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->nickname), $one_html); } elseif (! $value->username) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, "匿名用户"), $one_html); } break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析表单数据标签 public function parserFormLabel($content) { $pattern = '/\{pboot:formlist(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:formlist\}/'; $pattern2 = '/\[form:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); $fcode = - 1; $page = true; $start = 1; // 跳过未指定fcode的标签 if (! array_key_exists('fcode', $params)) { continue; } foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'fcode': $fcode = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 获取表名称 if (! $table = $this->model->getFormTable(escape_string($fcode))) { $content = str_replace($matches[0][$i], '', $content); continue; } // 读取数据 if (! $data = $this->model->getForm($table, escape_string($num), $page, $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'date': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->create_time), $one_html); break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } else { $one_html = str_replace($matches2[0][$j], '', $one_html); } } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析表单提交标签 public function parserSubmitFormLabel($content) { $pattern = '/\{pboot:form(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[1][$i]); $fcode = ''; foreach ($params as $key => $value) { switch ($key) { case 'fcode': $fcode = $value; break; } } if (! $fcode) { // 无表单编码不解析 continue; } $content = str_replace($matches[0][$i], Url::home('form/' . $fcode), $content); } } return $content; } // 解析文章评论 public function parserCommentLabel($content) { $pattern = '/\{pboot:comment(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:comment\}/'; $pattern2 = '/\[comment:([\w]+)(\s+[^]]+)?\]/'; $pattern3 = '/\{pboot:commentsub(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:commentsub\}/'; $pattern4 = '/\[commentsub:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); $page = true; $order = 'a.id desc'; $start = 1; // 跳过未指定fcode的标签 if (! array_key_exists('contentid', $params)) { continue; } foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'contentid': $contentid = $value; break; case 'order': $order = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getComment($contentid, 0, $num, $order, $page, $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); $one_html = $this->parserComment($matches2[1][$j], $matches2[0][$j], $one_html, $value, $params, $key); } $key ++; // 解析子评论 if (preg_match_all($pattern3, $one_html, $matches3)) { $count3 = count($matches3[0]); for ($k = 0; $k < $count3; $k ++) { // 读取子评论数据,正序排列,最大100条 if (! $data_sub = $this->model->getComment($contentid, $value->id, 100, 'a.id asc')) { $one_html = str_replace($matches3[0][$k], '', $one_html); continue; } // 匹配到子评论内部标签 if (preg_match_all($pattern4, $matches3[2][$k], $matches4)) { $count4 = count($matches4[0]); // 循环内的内容标签数量 } else { $count4 = 0; } $out_html_sub = ''; $key_sub = 1; foreach ($data_sub as $value_sub) { // 按子查询数据条数循环 $one_html_sub = $matches3[2][$k]; for ($m = 0; $m < $count4; $m ++) { // 循环替换数据 $params_sub = $this->parserParam($matches4[2][$m]); $one_html_sub = $this->parserComment($matches4[1][$m], $matches4[0][$m], $one_html_sub, $value_sub, $params_sub, $key_sub); } $key_sub ++; $out_html_sub .= $one_html_sub; } $one_html = str_replace($matches3[0][$k], $out_html_sub, $one_html); } } $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析我的评论 public function parserMyCommentLabel($content) { $pattern = '/\{pboot:mycomment(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:mycomment\}/'; $pattern2 = '/\[mycomment:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); $page = true; $order = 'a.id desc'; $start = 1; foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'order': $order = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getMyComment($num, $order, $page, $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); $one_html = str_replace("[mycomment:delaction]", Url::home('comment/del', null, 'id=' . $value->id), $one_html); $one_html = $this->parserComment($matches2[1][$j], $matches2[0][$j], $one_html, $value, $params, $key); } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析评论内容 private function parserComment($label, $search, $content, $data, $params, $key) { switch ($label) { case 'n': $content = str_replace($search, $this->adjustLabelData($params, $key) - 1, $content); break; case 'i': $content = str_replace($search, $this->adjustLabelData($params, $key), $content); break; case 'ip': $content = str_replace($search, $this->adjustLabelData($params, long2ip($data->user_ip)), $content); break; case 'os': $content = str_replace($search, $this->adjustLabelData($params, $data->user_os), $content); break; case 'bs': $content = str_replace($search, $this->adjustLabelData($params, $data->user_bs), $content); break; case 'date': $content = str_replace($search, $this->adjustLabelData($params, $data->create_time), $content); break; case 'headpic': if ($data->headpic) { if (! preg_match('/^http/', $data->headpic)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->headpic), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->headpic), $content); } } else { $content = str_replace($search, SITE_DIR . '/apps/admin/view/default/images/logo.png', $content); } break; case 'pheadpic': if ($data->pheadpic) { if (! preg_match('/^http/', $data->pheadpic)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->pheadpic), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->pheadpic), $content); } } else { $content = str_replace($search, SITE_DIR . '/apps/admin/view/default/images/logo.png', $content); } break; case 'replyaction': if ($data->pid) { $pid = $data->pid; } else { $pid = $data->id; } $content = str_replace($search, Url::home('comment/add', null, "contentid=" . $data->contentid . "&pid=" . $pid . "&puid=" . $data->uid), $content); break; case 'nickname': if ($data->nickname) { $content = str_replace($search, $this->adjustLabelData($params, $data->nickname), $content); } elseif ($data->username) { $content = str_replace($search, $this->adjustLabelData($params, "匿名"), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, "游客"), $content); } break; case 'pnickname': if ($data->pnickname) { $content = str_replace($search, $this->adjustLabelData($params, $data->pnickname), $content); } elseif ($data->pusername) { $content = str_replace($search, $this->adjustLabelData($params, "匿名"), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, "游客"), $content); } break; default: if (isset($data->{$label})) { $content = str_replace($search, $this->adjustLabelData($params, $data->{$label}), $content); } else { $content = str_replace($search, '', $content); } } return $content; } // 解析评论子楼层 public function parserCommentsubLabel($content) { $pattern = '/\{pboot:commentsub(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:commentsub\}/'; $pattern2 = '/\[commentsub:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = $this->config('pagesize'); $page = false; $order = 'a.id desc'; $start = 1; // 跳过未指定fcode的标签 if (! array_key_exists('contentid', $params)) { continue; } foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'contentid': $contentid = $value; break; case 'order': $order = $value; break; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if (! $data = $this->model->getComment($contentid, $num, $order, $page, $start)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'ip': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, long2ip($value->user_ip)), $one_html); break; case 'os': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->user_os), $one_html); break; case 'bs': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->user_bs), $one_html); break; case 'date': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->create_time), $one_html); break; case 'headpic': if ($value->headpic) { if (! preg_match('/^http/', $value->headpic)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->headpic), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->headpic), $one_html); } } else { $one_html = str_replace($matches2[0][$j], SITE_DIR . '/apps/admin/view/default/images/logo.png', $one_html); } break; case 'pheadpic': if ($value->pheadpic) { if (! preg_match('/^http/', $value->pheadpic)) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, SITE_DIR . $value->pheadpic), $one_html); } else { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->pheadpic), $one_html); } } else { $one_html = str_replace($matches2[0][$j], SITE_DIR . '/apps/admin/view/default/images/logo.png', $one_html); } break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } } } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析自定义SQL循环 public function parserSqlListLabel($content) { $pattern = '/\{pboot:sql(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:sql\}/'; $pattern2 = '/\[sql:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } $num = 1000; // 最大读取1000条 $sql = ''; $kp = 0; foreach ($params as $key => $value) { switch ($key) { case 'num': $num = $value; break; case 'sql': $sql = $value; break; case 'kp': $kp = $value; break; } } // 特殊表不允许输出 if (preg_match('/ay_user|ay_member/i', $sql)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 判断是否有条数限制 if ($num && ! preg_match('/limit/i', $sql)) { $sql .= " limit " . $num; } // 读取数据 if (! $data = $this->model->all($sql)) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; $a = 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; default: if (isset($value->{$matches2[1][$j]})) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value->{$matches2[1][$j]}), $one_html); } } if($kp > 0){ if($a%$kp == 0 && $a > 0){ //计数换行 //$one_html = $one_html."
"; } } } $a++; $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析二维码生成标签 public function parserQrcodeLabel($content) { $pattern = '/\{pboot:qrcode(\s+[^}]+)?\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $params = $this->parserParam($matches[1][$i]); $string = ''; foreach ($params as $key => $value) { switch ($key) { case 'string': $string = $value; break; } } if (! $string) { // 无内容不解析 continue; } $content = str_replace($matches[0][$i], '二维码', $content); } } return $content; } // 解析内容搜索结果标签 public function parserSearchLabel($content) { $pattern = '/\{pboot:search(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:search\}/'; $pattern2 = '/\[search:([\w]+)(\s+[^]]+)?\]/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); $field = request('field'); if (! preg_match('/^[\w\|\s]+$/', $field)) { $field = ''; } $keyword = request('keyword', 'vars'); $scode = request('scode'); $start = 1; if (! preg_match('/^[\w,\s]+$/', $scode)) { $scode = ''; } for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $num = $this->config('pagesize'); // 未设置条数时使用默认15 $order = 'a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; // 默认排序 $filter = ''; // 过滤 $tags = ''; // tag标签 $fuzzy = true; // 设置过滤、tag、筛选是否模糊匹配 $ispics = ''; // 是否多图 $isico = ''; // 是否缩略图 $istop = ''; // 是否置顶 $isrecommend = ''; // 是否推荐 $isheadline = ''; // 是否头条 $page = true; // 搜索默认分页 $lfield = ''; // 查询字段限制 $lg = get_lg(); // 查询语言限制,默认当前语言,可通过lg=* 来指定 if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } foreach ($params as $key => $value) { switch ($key) { case 'field': $field = $value; break; case 'scode': $scode = $value; break; case 'num': $num = $value; break; case 'order': switch ($value) { case 'id': $order = 'a.id DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC'; break; case 'date': $order = 'a.date DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.id DESC'; break; case 'sorting': $order = 'a.sorting ASC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.date DESC,a.id DESC'; break; case 'istop': $order = 'a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'isrecommend': $order = 'a.isrecommend DESC,a.istop DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'isheadline': $order = 'a.isrecommend DESC,a.istop DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'visits': case 'likes': case 'oppose': $order = $value . ' DESC,a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; break; case 'random': // 随机取数 $db_type = get_db_type(); if ($db_type == 'mysql') { $order = "RAND()"; } elseif ($db_type == 'sqlite') { $order = "RANDOM()"; } break; default: if ($value) { $orders = explode(',', $value); foreach ($orders as $k => $v) { if (strpos($v, 'ext_') === 0) { $orders[$k] = 'e.' . $v; } else { $orders[$k] = 'a.' . $v; } } $value = implode(',', $orders); $order = $value . ',a.istop DESC,a.isrecommend DESC,a.isheadline DESC,a.sorting ASC,a.date DESC,a.id DESC'; } } break; case 'filter': $filter = $value; case 'fuzzy': $fuzzy = $value; break; case 'tags': $tags = $value; break; case 'ispics': $ispics = $value; break; case 'isico': $isico = $value; break; case 'istop': $istop = $value; break; case 'isrecommend': $isrecommend = $value; break; case 'isheadline': $isheadline = $value; break; case 'page': $page = $value; break; case 'start': $start = $value; break; case 'lfield': $lfield = $value; break; case 'lg': $lg = $value; break; } } if ($scode == '*') { $scode = ''; } // filter数据筛选 $where1 = array(); if ($filter) { $filter = explode('|', $filter); if (count($filter) == 2) { $filter_arr = explode(',', $filter[1]); if ($filter[0] == 'title') { $filter[0] = 'a.title'; } foreach ($filter_arr as $value) { if ($value) { if ($fuzzy) { $where1[] = $filter[0] . " like '%" . escape_string($value) . "%'"; } else { $where1[] = $filter[0] . "='" . escape_string($value) . "'"; } } } } } // tags数据筛选 $where2 = array(); if ($tags) { $tags_arr = explode(',', $tags); foreach ($tags_arr as $value) { if ($value) { if ($fuzzy) { $where2[] = "a.tags like '%" . escape_string($value) . "%'"; } else { $where2[] = "a.tags='" . escape_string($value) . "'"; } } } } // 存储搜索条件,条件为“并列”关系,由于为模糊匹配,条件为空时意味着“任意” $where3 = array(); // 采取keyword方式 if ($keyword) { if (strpos($field, '|')) { // 匹配多字段的关键字搜索 $field = explode('|', $field); foreach ($field as $value) { if ($value == 'title') { $value = 'a.title'; } if ($fuzzy) { $like = " like '%" . $keyword . "%'"; // 前面已经转义过 } else { $like = " like '" . $keyword . "'"; // 前面已经转义过 } if (isset($where3[0])) { $where3[0] .= ' OR ' . $value . $like; } else { $where3[0] = $value . $like; } } if (count($field) > 1) { $where3[0] = '(' . $where3[0] . ')'; } } else { // 匹配单一字段的关键字搜索 if ($field) { if ($field == 'title') { $field = 'a.title'; } $where3[$field] = $keyword; } else { $where3['a.title'] = $keyword; } } } // 数据接收 if ($_POST) { $receive = $_POST; } else { $receive = $_GET; } foreach ($receive as $key => $value) { if (! ! $value = request($key, 'vars')) { if ($key == 'title') { $key = 'a.title'; } if (preg_match('/^[\w\-\.]+$/', $key)) { // 带有违规字符时不带入查询 $where3[$key] = $value; } } } // 去除特殊键值 unset($where3['keyword']); unset($where3['field']); unset($where3['scode']); unset($where3['page']); unset($where3['from']); unset($where3['isappinstalled']); unset($where3['tdsourcetag']); unset($where3['x']); unset($where3['y']); unset($where3['searchtpl']); unset($where3['p']); unset($where3['s']); // 无任何条件不显示内容 if (! $where3) { $content = str_replace($matches[0][$i], '', $content); continue; } // 判断多图调节参数 if ($ispics !== '') { if ($ispics) { $where3[] = "a.pics<>''"; } else { $where3[] = "a.pics=''"; } } // 判断缩略图调节参数 if ($isico !== '') { if ($isico) { $where3[] = "a.ico<>''"; } else { $where3[] = "a.ico=''"; } } // 判断置顶调节参数 if ($istop !== '') { if ($istop) { $where3[] = "a.istop=1"; } else { $where3[] = "a.istop=0"; } } // 判断推荐调节参数 if ($isrecommend !== '') { if ($isrecommend) { $where3[] = "a.isrecommend=1"; } else { $where3[] = "a.isrecommend=0"; } } // 判断头条调节参数 if ($isheadline !== '') { if ($isheadline) { $where3[] = "a.isheadline=1"; } else { $where3[] = "a.isheadline=0"; } } // 起始数校验 if (! is_numeric($start) || $start < 1) { $start = 1; } // 读取数据 if ($page) { if (isset($paging)) { error('请不要在一个页面使用多个具有分页的列表,您可将多余的使用page=0关闭分页!'); } else { $paging = true; $data = $this->model->getLists($scode, $num, $order, $where1, $where2, $where3, $fuzzy, $start, $lfield, $lg); } } else { $data = $this->model->getList($scode, $num, $order, $where1, $where2, $where3, $fuzzy, $start, $lfield, $lg); } // 无数据直接替换 if (! $data) { $content = str_replace($matches[0][$i], '', $content); continue; } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $pagenum = defined('PAGE') ? PAGE : 1; $key = ($pagenum - 1) * $num + 1; foreach ($data as $value) { // 按查询数据条数循环 $one_html = $matches[2][$i]; for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); $one_html = $this->parserList($matches2[1][$j], $matches2[0][$j], $one_html, $value, $params, $key); } $key ++; $out_html .= $one_html; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析列表分页标签 public function parserPageLabel($content) { $pattern = '/\{page:([\w]+)\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { switch ($matches[1][$i]) { case 'bar': $content = str_replace($matches[0][$i], $this->getVar('pagebar'), $content); break; case 'current': $content = str_replace($matches[0][$i], $this->getVar('pagecurrent') ?: 0, $content); break; case 'count': $content = str_replace($matches[0][$i], $this->getVar('pagecount') ?: 0, $content); break; case 'rows': $content = str_replace($matches[0][$i], $this->getVar('pagerows') ?: 0, $content); break; case 'index': $content = str_replace($matches[0][$i], $this->getVar('pageindex'), $content); break; case 'pre': $content = str_replace($matches[0][$i], $this->getVar('pagepre'), $content); break; case 'next': $content = str_replace($matches[0][$i], $this->getVar('pagenext'), $content); break; case 'last': $content = str_replace($matches[0][$i], $this->getVar('pagelast'), $content); break; case 'status': $content = str_replace($matches[0][$i], $this->getVar('pagestatus'), $content); break; case 'numbar': $content = str_replace($matches[0][$i], $this->getVar('pagenumbar'), $content); break; case 'selectbar': $content = str_replace($matches[0][$i], $this->getVar('pageselectbar'), $content); break; } } } return $content; } // 解析循环标签 public function parserLoopLabel($content) { $pattern = '/\{pboot:loop(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:loop\}/'; if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $start = 1; $end = $this->config('pagesize'); if (! self::checkLabelLevel($params)) { $content = str_replace($matches[0][$i], '', $content); continue; } foreach ($params as $key => $value) { switch ($key) { case 'start': $start = $value; break; case 'end': $end = $value; break; } } $out_html = ''; $key = 1; for ($n = $start; $n <= $end; $n ++) { $one_html = str_replace('[loop:n]', $key - 1, $matches[2][$i]); $one_html = str_replace('[loop:i]', $key, $one_html); $one_html = str_replace('[loop:index]', $n, $one_html); $out_html .= $one_html; $key ++; } $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // 解析IF条件标签 public function parserIfLabel($content) { $pattern = '/\{pboot:if\(([^}^\$]+)\)\}([\s\S]*?)\{\/pboot:if\}/'; $pattern2 = '/pboot:([0-9])+if/'; $content = str_replace('{xx}', '[xx]', $content); if (preg_match_all($pattern, $content, $matches)) { $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { $flag = ''; $out_html = ''; $danger = false; $white_fun = array( 'date' ); $matches[0][$i] = str_replace('[xx]', '{xx}', $matches[0][$i]); // 还原可能包含的保留内容,避免判断失效 $matches[1][$i] = $this->restorePreLabel($matches[1][$i]); // 带有函数的条件语句进行安全校验 if (preg_match_all('/([\w]+)([\x00-\x1F\x7F\/\*\<\>\%\w\s\\\\]+)?\(/i', $matches[1][$i], $matches2)) { foreach ($matches2[1] as $value) { if (function_exists(trim($value)) && ! in_array($value, $white_fun)) { $danger = true; break; } } foreach ($matches2[2] as $value) { if (function_exists(trim($value)) && ! in_array($value, $white_fun)) { $danger = true; break; } } } // 过滤特殊字符串 if (preg_match('/(\([\w\s\.]+\))|(\$_GET\[)|(\$_POST\[)|(\$_REQUEST\[)|(\$_COOKIE\[)|(\$_SESSION\[)|(file_put_contents)|(file_get_contents)|(fwrite)|(phpinfo)|(base64)|(`)|(shell_exec)|(eval)|(assert)|(system)|(exec)|(passthru)|(pcntl_exec)|(popen)|(proc_open)|(print_r)|(print)|(urldecode)|(chr)|(include)|(request)|(__FILE__)|(__DIR__)|(copy)|(call_user_)|(preg_replace)|(array_map)|(array_reverse)|(array_filter)|(getallheaders)|(get_headers)|(decode_string)|(htmlspecialchars)|(session_id)|(strrev)|(substr)|(php.info)|(@file.@_put_content)/i', $matches[1][$i])) { $danger = true; } // 如果有危险函数,则不解析该IF if ($danger) { continue; } //if标签解析 $flag = symbol($matches[1][$i]); if (preg_match('/^([\s\S]*)\{else\}([\s\S]*)$/', $matches[2][$i], $matches2)) { // 判断是否存在else switch ($flag) { case 'if': // 条件为真 if (isset($matches2[1])) { $out_html = $matches2[1]; } break; case 'else': // 条件为假 if (isset($matches2[2])) { $out_html = $matches2[2]; } break; } } elseif ($flag == 'if') { $out_html = $matches[2][$i]; } // 无限极嵌套解析 if (preg_match($pattern2, $out_html, $matches3)) { $out_html = str_replace('pboot:' . $matches3[1] . 'if', 'pboot:if', $out_html); $out_html = str_replace('{' . $matches3[1] . 'else}', '{else}', $out_html); $out_html = $this->parserIfLabel($out_html); } $content = str_replace('[xx]', '{xx}', $content); // 执行替换 $content = str_replace($matches[0][$i], $out_html, $content); } if((int)$count === 0){ $content = str_replace('[xx]', '{xx}', $content); } }else{ $content = str_replace('[xx]', '{xx}', $content); } // var_dump(111);die; return $content; } // 调整标签数据 protected function adjustLabelData($params, $data, $label = null, $savelabel = false) { if (! $data) return $data; // 图片缩放功能 if (isset($params['maxwidth']) || isset($params['maxheight'])) { $maxwidth = isset($params['maxwidth']) ? $params['maxwidth'] : null; $maxheight = isset($params['maxheight']) ? $params['maxheight'] : null; $max_src_file = ROOT_PATH . $data; $max_out_file = RUN_PATH . '/image/mw' . $maxwidth . '_mh' . $maxheight . '_' . basename($data); if (! file_exists($max_out_file) && file_exists($max_src_file)) { if (resize_img($max_src_file, $max_out_file, $maxwidth, $maxheight)) { $data = str_replace(ROOT_PATH, '', $max_out_file); } } elseif (file_exists($max_out_file) && file_exists($max_src_file)) { $data = str_replace(ROOT_PATH, '', $max_out_file); } } // 图片固定大小 if (isset($params['width']) || isset($params['height'])) { $width = isset($params['width']) ? $params['width'] : null; $height = isset($params['height']) ? $params['height'] : null; $src_file = ROOT_PATH . $data; $out_file = RUN_PATH . '/image/w' . $width . '_h' . $height . '_' . basename($data); if (! file_exists($out_file) && file_exists($src_file)) { if (cut_img($src_file, $out_file, $width, $height)) { $data = str_replace(ROOT_PATH, '', $out_file); } } elseif (file_exists($out_file) && file_exists($src_file)) { $data = str_replace(ROOT_PATH, '', $out_file); } } // 检查标签权限 if (! self::checkLabelLevel($params)) { $data = ''; } if (is_array($params) && $params) { foreach ($params as $key => $value) { switch ($key) { case 'style': // 时间样式 if ($params['style'] && $date = strtotime($data)) { $data = date($params['style'], $date); } break; case 'len': // 长度截取 if ($params['len'] && is_string($data)) { if (mb_strlen($data, 'utf-8') > $params['len']) { if (isset($params['more'])) { $more = $params['more']; } else { $more = '···'; } $data = mb_substr($data, 0, $params['len'], 'utf-8') . $more; } } break; case 'lencn': // 以中文占位长度方式截取,英文算半个 if ($params['lencn'] && is_string($data)) { if (strlen_both($data) > $params['lencn']) { if (isset($params['more'])) { $more = $params['more']; } else { $more = '···'; } $data = substr_both($data, 0, $params['lencn']) . $more; } } break; case 'drophtml': // 去除html标签 if ($params['drophtml']) { $data = strip_tags($data); } break; case 'dropblank': // 清理特殊空白 if ($params['dropblank']) { $data = clear_html_blank($data); } break; case 'decode': // 解码或转义字符 if ($params['decode']) { $data = decode_string($data); } else { $data = escape_string($data); } break; case 'substr': // 截取字符串 if ($params['substr'] && is_string($data)) { $arr = explode(',', $params['substr']); if (count($arr) == 2 && $arr[1]) { $data = mb_substr($data, $arr[0] - 1, $arr[1], 'utf-8'); } else { $data = mb_substr($data, $arr[0] - 1); } } break; case 'unit': // bytes转换未其它单位 switch ($params['unit']) { case 'KB': case 'kb': $data = $data / 1024; break; case 'MB': case 'mb': $data = $data / (1024 * 1024); break; case 'GB': case 'gb': $data = $data / (1024 * 1024 * 1024); break; case 'TB': case 'tb': $data = $data / (1024 * 1024 * 1024 * 1024); break; case 'PB': case 'pb': $data = $data / (1024 * 1024 * 1024 * 1024 * 1024); break; case 'EB': case 'eb': $data = $data / (1024 * 1024 * 1024 * 1024 * 1024 * 1024); break; } break; case 'decimal': // 小数点 if ($params['decimal']) { $data = number_format($data, $params['decimal']); } break; case 'operate': // 实现列表页标签+-*/%运算功能 if (preg_match('/^([\+\-\*\/\%])([0-9\.]+)$/', $params['operate'], $mathes)) { if (! is_numeric($data)) { $data = 0; } switch ($mathes[1]) { case '+': $data = $data + $mathes[2]; break; case '-': $data = $data - $mathes[2]; break; case '*': $data = $data * $mathes[2]; break; case '/': $data = $data / $mathes[2]; break; case '%': $data = $data % $mathes[2]; break; } } break; case 'mark': if ($label && $reqdata = request($label, 'vars') ?: request('keyword', 'vars')) { $data = preg_replace('/(' . $reqdata . ')/i', '$1', $data); } break; } } } // 对标签内容中含有标签是否保留不解析 if ($savelabel && $data) { $this->pre[] = $data; // 保存内容避免解析 end($this->pre); // 指向最后一个元素 $data = '#pre:' . key($this->pre) . '#'; } return $data; } // 解析调节参数 protected function parserParam($string, $striptags = true) { if (! $string = trim($string)) return array(); $string = preg_replace('/\s+/', ' ', $string); // 多空格处理 if ($striptags) { $string = strip_tags($string); } $param = array(); if (preg_match_all('/([\w]+)[\s]?=[\s]?([\"]([^\"]+)?[\"]|[\']([^\']+)?[\']|([^\s]+))/i', $string, $matches)) { foreach ($matches[1] as $key => $value) { $param[$value] = $matches[3][$key] ?: $matches[4][$key] ?: $matches[5][$key]; } } return $param; } // 解析列表标签 protected function parserList($label, $search, $content, $data, $params, $key) { switch ($label) { case 'n': $content = str_replace($search, $this->adjustLabelData($params, $key) - 1, $content); break; case 'i': $content = str_replace($search, $this->adjustLabelData($params, $key), $content); break; case 'link': if ($data->outlink) { // 外链 $content = str_replace($search, $data->outlink, $content); } else { if ($data->type == 1) { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'about', $data->scode, $data->sortfilename, '', ''), $content); } else { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'content', $data->scode, $data->sortfilename, $data->id, $data->filename), $content); } } break; case 'sortlink': $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'list', $data->scode, $data->sortfilename, '', ''), $content); break; case 'subsortlink': if ($data->subscode) { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'list', $data->subscode, $data->subfilename, '', ''), $content); } else { $content = str_replace($search, 'javascript:;', $content); } break; case 'sortname': if ($data->sortname) { $content = str_replace($search, $this->adjustLabelData($params, $data->sortname, $label), $content); } else { $content = str_replace($search, '', $content); } break; case 'subsortname': if ($data->subsortname) { $content = str_replace($search, $this->adjustLabelData($params, $data->subsortname, $label), $content); } else { $content = str_replace($search, '', $content); } break; case 'ico': if ($data->ico) { if (! preg_match('/^http/', $data->ico)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->ico), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->ico), $content); } } elseif (preg_match('//i', $data->content, $srcs) && isset($srcs[1])) { $content = str_replace($search, $this->adjustLabelData($params, $srcs[1]), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, STATIC_DIR . '/images/nopic.png'), $content); } break; case 'isico': if ($data->ico) { $content = str_replace($search, 1, $content); } else { $content = str_replace($search, 0, $content); } break; case 'ispics': if ($data->pics) { $content = str_replace($search, 1, $content); } else { $content = str_replace($search, 0, $content); } break; case 'enclosure': if ($data->enclosure) { if (! preg_match('/^http/', $data->enclosure)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->enclosure), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->enclosure), $content); } } else { $content = str_replace($search, '', $content); } break; case 'enclosuresize': if ($data->enclosure && file_exists(ROOT_PATH . $data->enclosure)) { $content = str_replace($search, $this->adjustLabelData($params, filesize(ROOT_PATH . $data->enclosure)), $content); } else { $content = str_replace($search, 0, $content); } case 'likeslink': $content = str_replace($search, Url::get('home/Do/likes/id/' . $data->id), $content); break; case 'opposelink': $content = str_replace($search, Url::get('home/Do/oppose/id/' . $data->id), $content); break; case 'content': $content = str_replace($search, $this->adjustLabelData($params, $data->content, $label, true), $content); // 占位替换 break; case 'keywords': $content = str_replace($search, $this->adjustLabelData($params, $data->keywords, $label, true), $content); // 占位替换 break; case 'description': $content = str_replace($search, $this->adjustLabelData($params, $data->description, $label, true), $content); // 占位替换 break; default: if (isset($data->$label)) { $content = str_replace($search, $this->adjustLabelData($params, $data->$label, $label), $content); } elseif (strpos($label, 'ext_') === 0) { $content = str_replace($search, '', $content); } } return $content; } // 解析内容详情标签 protected function parserContent($label, $search, $content, $data, $params, $sort) { switch ($label) { case 'link': if ($data->type == 1) { if ($data->sortoutlink) { $content = str_replace($search, $data->sortoutlink, $content); } else { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'about', $data->scode, $data->sortfilename, '', ''), $content); } } else { if ($data->outlink) { $content = str_replace($search, $data->outlink, $content); } else { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'content', $data->scode, $data->sortfilename, $data->id, $data->filename), $content); } } break; case 'sortlink': $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'list', $data->scode, $data->sortfilename, '', ''), $content); break; case 'subsortlink': if ($data->subscode) { $content = str_replace($search, $this->parserLink($data->type, $data->urlname, 'list', $data->subscode, $data->subfilename, '', ''), $content); } else { $content = str_replace($search, '', $content); } break; case 'sortname': if ($data->sortname) { $content = str_replace($search, $this->adjustLabelData($params, $data->sortname), $content); } else { $content = str_replace($search, '', $content); } break; case 'subsortname': if ($data->subsortname) { $content = str_replace($search, $this->adjustLabelData($params, $data->subsortname), $content); } else { $content = str_replace($search, '', $content); } break; case 'ico': if ($data->ico) { if (! preg_match('/^http/', $data->ico)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->ico), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->ico), $content); } } elseif (preg_match('//i', $data->content, $srcs) && isset($srcs[1])) { $content = str_replace($search, $this->adjustLabelData($params, $srcs[1]), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, STATIC_DIR . '/images/nopic.png'), $content); } break; case 'isico': if ($data->ico) { $content = str_replace($search, 1, $content); } else { $content = str_replace($search, 0, $content); } break; case 'ispics': if ($data->pics) { $content = str_replace($search, 1, $content); } else { $content = str_replace($search, 0, $content); } break; case 'enclosure': if ($data->enclosure) { if (! preg_match('/^http/', $data->enclosure)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $data->enclosure), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $data->enclosure), $content); } } else { $content = str_replace($search, '', $content); } break; case 'enclosuresize': if ($data->enclosure && file_exists(ROOT_PATH . $data->enclosure)) { $content = str_replace($search, $this->adjustLabelData($params, filesize(ROOT_PATH . $data->enclosure)), $content); } else { $content = str_replace($search, 0, $content); } break; case 'likeslink': $content = str_replace($search, Url::get('home/Do/likes/id/' . $data->id), $content); break; case 'opposelink': $content = str_replace($search, Url::get('home/Do/oppose/id/' . $data->id), $content); break; case 'precontent': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $pre = $this->model->getContentPre($sort->scode, $data->id)) { $content = str_replace($search, '' . $this->adjustLabelData($params, $pre->title) . '', $content); } else { if (isset($params['notext'])) { $content = str_replace($search, $params['notext'], $content); } else { $content = str_replace($search, '没有了!', $content); } } break; case 'prelink': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $pre = $this->model->getContentPre($sort->scode, $data->id)) { $content = str_replace($search, $this->parserLink($pre->type, $pre->urlname, 'content', $pre->scode, $pre->sortfilename, $pre->id, $pre->filename), $content); } else { $content = str_replace($search, 'javascript:;', $content); } break; case 'pretitle': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $pre = $this->model->getContentPre($sort->scode, $data->id)) { $content = str_replace($search, $this->adjustLabelData($params, $pre->title), $content); } else { if (isset($params['notext'])) { $content = str_replace($search, $params['notext'], $content); } else { $content = str_replace($search, '没有了!', $content); } } break; case 'preico': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $pre = $this->model->getContentPre($sort->scode, $data->id)) { if (! preg_match('/^http/', $pre->ico)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $pre->ico), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $pre->ico), $content); } } else { $content = str_replace($search, $this->adjustLabelData($params, STATIC_DIR . '/images/nopic.png'), $content); } break; case 'nextcontent': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $next = $this->model->getContentNext($sort->scode, $data->id)) { $content = str_replace($search, '' . $this->adjustLabelData($params, $next->title) . '', $content); } else { if (isset($params['notext'])) { $content = str_replace($search, $params['notext'], $content); } else { $content = str_replace($search, '没有了!', $content); } } break; case 'nextlink': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $next = $this->model->getContentNext($sort->scode, $data->id)) { $content = str_replace($search, $this->parserLink($next->type, $next->urlname, 'content', $next->scode, $next->sortfilename, $next->id, $next->filename), $content); } else { $content = str_replace($search, 'javascript:;', $content); } break; case 'nexttitle': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $next = $this->model->getContentNext($sort->scode, $data->id)) { $content = str_replace($search, $this->adjustLabelData($params, $next->title), $content); } else { if (isset($params['notext'])) { $content = str_replace($search, $params['notext'], $content); } else { $content = str_replace($search, '没有了!', $content); } } break; case 'nextico': if ($data->type != 2) // 非列表内容页不解析 break; if (! ! $next = $this->model->getContentNext($sort->scode, $data->id)) { if (! preg_match('/^http/', $next->ico)) { $content = str_replace($search, $this->adjustLabelData($params, SITE_DIR . $next->ico), $content); } else { $content = str_replace($search, $this->adjustLabelData($params, $next->ico), $content); } } else { $content = str_replace($search, $this->adjustLabelData($params, STATIC_DIR . '/images/nopic.png'), $content); } break; case 'content': // 内链处理 if (! ! $tags = $this->model->getTags()) { // 将A链接保护起来,alt、titel保护起来 $rega = "/(.*?<\/a>)|(alt=.*?>)|(title=.*?>)/i"; preg_match_all($rega, $data->content, $matches1); foreach ($matches1[0] as $key => $value) { $data->content = str_replace($value, '#rega:' . $key . '#', $data->content); } // 去除包含关系的短tags,实现长关键字优先 foreach ($tags as $key => $value) { foreach ($tags as $key2 => $value2) { if (strpos($value2->name, $value->name) !== false && $key != $key2) { unset($tags[$key]); } } } // 执行内链替换 foreach ($tags as $value) { $data->content = preg_replace('/' . $value->name . '/', '' . $value->name . '', $data->content, $this->config('content_tags_replace_num') ?: 3); } // 还原保护的内容 $pattern = '/\#rega:([0-9]+)\#/'; if (preg_match_all($pattern, $data->content, $matches2)) { $count = count($matches2[0]); for ($i = 0; $i < $count; $i ++) { $data->content = str_replace($matches2[0][$i], $matches1[0][$matches2[1][$i]], $data->content); } } } $content = str_replace($search, $this->adjustLabelData($params, $data->content, null, true), $content); break; case 'keywords': // 如果内容关键字为空,则自动使用全局关键字 if ($data->keywords) { $content = str_replace($search, $this->adjustLabelData($params, $data->keywords, null, true), $content); } else { $content = str_replace($search, '{pboot:sitekeywords}', $content); } break; case 'description': // 如果内容描述为空,则自动使用全局描述 if ($data->description) { $content = str_replace($search, $this->adjustLabelData($params, $data->description, null, true), $content); } else { $content = str_replace($search, '{pboot:sitedescription}', $content); } break; default: if (isset($data->$label)) { $content = str_replace($search, $this->adjustLabelData($params, $data->$label), $content); } elseif (strpos($label, 'ext_') === 0) { $content = str_replace($search, '', $content); } } return $content; } //解析城市列表标签 // 解析分站列表标签 public function parserCityListLabel($content) { $pattern = '/\{pboot:city(\s+[^}]+)?\}([\s\S]*?)\{\/pboot:city\}/'; $pattern2 = '/\[city:([\w]+)(\s+[^]]+)?\]/'; $pattern3 = '/pboot:([0-9])+city/'; if (preg_match_all($pattern, $content, $matches)) { $data = $this->model->getCityListTree(); $count = count($matches[0]); for ($i = 0; $i < $count; $i ++) { // 无数据时直接替换整体标签为空 if (! $data['tree']) { $content = str_replace($matches[0][$i], '', $content); continue; } // 获取调节参数 $params = $this->parserParam($matches[1][$i]); $pid = 0; $num = 0; $istop = 0; $islevel = 0; foreach ($params as $key => $value) { switch ($key) { case 'pid': $pid = $value; break; case 'num': $num = $value; break; case 'istop': $istop = $value; break; case 'islevel': $islevel = $value; break; } } $out_data = array(); // 置顶 if( $istop ){ foreach( $data['tree'] as $row ){ if( $row['istop']==1 ){ array_push($out_data,$row); } foreach( $row['son'] as $son ){ if( $son['istop']==1 ){ array_push($out_data,$son); } } } }elseif( $islevel ){ //显示层级 $citys = $this->config('citys'); $pid = 0; if( cookie('city')!=='' ){ $cur_city = $citys[cookie('city')]; if( $cur_city['pid'] == 0 ){ $pid = $cur_city['id']; }else{ $pid = $cur_city['pid']; } } foreach( $data['tree'] as $row ){ if( $row['id']==$pid ){ foreach( $row['son'] as $son ){ array_push($out_data,$son); } } } }else{ if ($pid) { // 非顶级栏目起始,调用子栏目 $parent_arr = explode(',', $pid); foreach ($parent_arr as $vp) { if (isset($data['tree'][trim($vp)]['son'])) { $out_data = array_merge($out_data, $data['tree'][trim($vp)]['son']); } } } else { // 顶级栏目起始 $out_data = $data['top']; } } // 读取指定数量 if ($num) { $out_data = array_slice($out_data, 0, $num); } // 匹配到内部标签 if (preg_match_all($pattern2, $matches[2][$i], $matches2)) { $count2 = count($matches2[0]); // 循环内的内容标签数量 } else { $count2 = 0; } $out_html = ''; $key = 1; foreach ($out_data as $value) { // 按查询的数据条数循环 $one_html = $matches[2][$i]; if ($count2) { for ($j = 0; $j < $count2; $j ++) { // 循环替换数据 $params = $this->parserParam($matches2[2][$j]); switch ($matches2[1][$j]) { case 'n': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key) - 1, $one_html); break; case 'i': $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $key), $one_html); break; case 'home': $one_html = str_replace($matches2[0][$j], create_city_url($value['etitle'], $value['isurl'], false), $one_html); break; case 'link': $one_html = str_replace($matches2[0][$j], create_city_url($value['etitle'], $value['isurl'], true), $one_html); break; case 'soncount': if (isset($data['tree'][$value['id']]['son'])) { $one_html = str_replace($matches2[0][$j], count($data['tree'][$value['id']]['son']), $one_html); } else { $one_html = str_replace($matches2[0][$j], 0, $one_html); } break; default: if (isset($value[$matches2[1][$j]])) { $one_html = str_replace($matches2[0][$j], $this->adjustLabelData($params, $value[$matches2[1][$j]]), $one_html); } } } } $key ++; $out_html .= $one_html; } // 无限极嵌套解析 if (preg_match($pattern3, $out_html, $matches3)) { $out_html = str_replace('pboot:' . $matches3[1] . 'city', 'pboot:city', $out_html); $out_html = str_replace('[' . $matches3[1] . 'city:', '[city:', $out_html); $out_html = $this->parserCityListLabel($out_html); } // 执行内容替换 $content = str_replace($matches[0][$i], $out_html, $content); } } return $content; } // end 城市分站解析 // 替换页面内容关键词 protected function parserReplaceKeyword($content) { $keys = $this->config('content_keyword_replace'); $keys_arr = explode(',', strip_tags($keys)); foreach ($keys_arr as $key => $value) { $content = str_replace($value, str_repeat('*', mb_strlen($value)), $content); } return $content; } // 解析生成内容链接 public function parserLink($type, $urlname, $pagetype, $scode, $sortfilename, $id = '', $contentfilename = '') { $url_break_char = $this->config('url_break_char') ?: '_'; $url_rule_sort_suffix = $this->config('url_rule_sort_suffix') ? true : null; $url_rule_content_path = $this->config('url_rule_content_path') ? true : false; if ($type == 1 || $pagetype == 'about') { $urlname = $urlname ?: 'about'; if ($sortfilename) { $link = Url::home($sortfilename); } else { $link = Url::home($urlname . $url_break_char . $scode); } } else { $urlname = $urlname ?: 'list'; if ($pagetype == 'list') { if ($sortfilename) { $link = Url::home($sortfilename); } else { $link = Url::home($urlname . $url_break_char . $scode); } } elseif ($pagetype == 'content') { if ($url_rule_content_path) { if ($contentfilename) { $link = Url::home($contentfilename, true); } else { $link = Url::home($id, true); } } else { if ($sortfilename && $contentfilename) { $link = Url::home($sortfilename . '/' . $contentfilename, true); } elseif ($sortfilename) { $link = Url::home($sortfilename . '/' . $id, true); } elseif ($contentfilename) { $link = Url::home($urlname . $url_break_char . $scode . '/' . $contentfilename, true); } else { $link = Url::home($urlname . $url_break_char . $scode . '/' . $id, true); } } } else { $link = 'javascript:;'; } } return $link; } // 检查标签权限 protected function checkLabelLevel($params) { foreach ($params as $key => $value) { switch ($key) { case 'showgcode': // 指定等级显示,支持多个逗号隔开 $showgcode = explode(',', $params['showgcode']); if (! in_array(session('pboot_gcode'), $showgcode)) { return false; } break; case 'showucode': // 指定用户显示,支持多个逗号隔开 $showucode = explode(',', $params['showucode']); if (! in_array(session('pboot_ucode'), $showucode)) { return false; } break; case 'hidegcode': // 指定等级隐藏,支持多个逗号隔开 $hidegcode = explode(',', $params['hidegcode']); if (in_array(session('pboot_gcode'), $hidegcode)) { return false; } break; case 'hideucode': // 指定用户隐藏,支持多个逗号隔开 $hideucode = explode(',', $params['hideucode']); if (in_array(session('pboot_ucode'), $hideucode)) { return false; } break; case 'showgcodelt': // 等级小于显示 if ($params['showgcodelt'] <= session('pboot_gcode')) { return false; } break; case 'showgcodegt': // 等级大于显示 if ($params['showgcodegt'] >= session('pboot_gcode')) { return false; } break; case 'showgcodele': // 等级小于等于显示 if ($params['showgcodele'] < session('pboot_gcode')) { return false; } break; case 'showgcodege': // 等级大于等于显示 if ($params['showgcodege'] > session('pboot_gcode')) { return false; } break; case 'hidegcodelt': // 等级小于隐藏 if ($params['hidegcodelt'] > session('pboot_gcode')) { return false; } break; case 'hidegcodegt': // 等级大于隐藏 if ($params['hidegcodegt'] < session('pboot_gcode')) { return false; } break; case 'hidegcodele': // 等级小于等于隐藏 if ($params['hidegcodele'] >= session('pboot_gcode')) { return false; } break; case 'hidegcodege': // 等级大于等于隐藏 if ($params['hidegcodege'] <= session('pboot_gcode')) { return false; } break; case 'showlogin': // 登录后显示 if ($params['showlogin'] && ! session('pboot_uid')) { return false; } break; case 'hidelogin': // 登录后隐藏 if ($params['hidelogin'] && session('pboot_uid')) { return false; } break; } } return true; } }