Keine Beschreibung
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

Special.php 41KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海口快推科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace app\admin\controller;
  14. use think\Page;
  15. use think\Db;
  16. use think\Config;
  17. class Special extends Base
  18. {
  19. public $nid = 'special';
  20. public $channeltype = '';
  21. public function _initialize()
  22. {
  23. parent::_initialize();
  24. $channeltype_list = config('global.channeltype_list');
  25. $this->channeltype = $channeltype_list[$this->nid];
  26. empty($this->channeltype) && $this->channeltype = 7;
  27. $this->assign('nid', $this->nid);
  28. $this->assign('channeltype', $this->channeltype);
  29. // 返回页面
  30. $paramTypeid = input('param.typeid/d', 0);
  31. $this->callback_url = url('Special/index', ['lang' => $this->admin_lang, 'typeid' => $paramTypeid]);
  32. $this->assign('callback_url', $this->callback_url);
  33. }
  34. //专题列表
  35. public function index()
  36. {
  37. $assign_data = $condition = [];
  38. // 获取到所有GET参数
  39. $param = input('param.');
  40. $typeid = input('typeid/d', 0);
  41. // 搜索、筛选查询条件处理
  42. foreach (['keywords', 'typeid', 'flag', 'is_release','province_id','city_id','area_id'] as $key) {
  43. if ($key == 'typeid' && empty($param['typeid'])) {
  44. $typeids = Db::name('arctype')->where('current_channel', $this->channeltype)->column('id');
  45. $condition['a.typeid'] = array('IN', $typeids);
  46. }
  47. if (isset($param[$key]) && $param[$key] !== '') {
  48. if ($key == 'keywords') {
  49. $keywords = $param[$key];
  50. $condition['a.title'] = array('LIKE', "%{$param[$key]}%");
  51. } else if ($key == 'typeid') {
  52. $typeid = $param[$key];
  53. $hasRow = model('Arctype')->getHasChildren($typeid);
  54. $typeids = get_arr_column($hasRow, 'id');
  55. // 权限控制 by 小虎哥
  56. $admin_info = session('admin_info');
  57. if (0 < intval($admin_info['role_id'])) {
  58. $auth_role_info = $admin_info['auth_role_info'];
  59. if (!empty($typeid) && !empty($auth_role_info) && !empty($auth_role_info['permission']['arctype'])) {
  60. $typeids = array_intersect($typeids, $auth_role_info['permission']['arctype']);
  61. }
  62. }
  63. $condition['a.typeid'] = array('IN', $typeids);
  64. } else if ($key == 'flag') {
  65. if ('is_release' == $param[$key]) {
  66. $condition['a.users_id'] = array('gt', 0);
  67. } else {
  68. $FlagNew = $param[$key];
  69. $condition['a.'.$param[$key]] = array('eq', 1);
  70. }
  71. } else if (in_array($key, ['province_id','city_id','area_id'])) {
  72. if (!empty($param['area_id'])) {
  73. $condition['a.area_id'] = $param['area_id'];
  74. } else if (!empty($param['city_id'])) {
  75. $condition['a.city_id'] = $param['city_id'];
  76. } else if (!empty($param['province_id'])) {
  77. $condition['a.province_id'] = $param['province_id'];
  78. }
  79. } else {
  80. $condition['a.'.$key] = array('eq', $param[$key]);
  81. }
  82. }
  83. }
  84. // 权限控制 by 小虎哥
  85. $admin_info = session('admin_info');
  86. if (0 < intval($admin_info['role_id'])) {
  87. $auth_role_info = $admin_info['auth_role_info'];
  88. if (!empty($auth_role_info) && isset($auth_role_info['only_oneself']) && 1 == $auth_role_info['only_oneself']) {
  89. $condition['a.admin_id'] = $admin_info['admin_id'];
  90. }
  91. }
  92. // 时间检索条件
  93. $begin = strtotime(input('add_time_begin'));
  94. $end = strtotime(input('add_time_end'));
  95. if ($begin > 0 && $end > 0) {
  96. $condition['a.add_time'] = array('between', "$begin, $end");
  97. } else if ($begin > 0) {
  98. $condition['a.add_time'] = array('egt', $begin);
  99. } else if ($end > 0) {
  100. $condition['a.add_time'] = array('elt', $end);
  101. }
  102. // 必要条件
  103. $condition['a.channel'] = array('eq', $this->channeltype);
  104. $condition['a.lang'] = array('eq', $this->admin_lang);
  105. $condition['a.is_del'] = array('eq', 0);
  106. $conditionNew = "(a.users_id = 0 OR (a.users_id > 0 AND a.arcrank >= 0))";
  107. // 自定义排序
  108. $orderby = input('param.orderby/s');
  109. $orderway = input('param.orderway/s');
  110. if (!empty($orderby) && !empty($orderway)) {
  111. $orderby = "a.{$orderby} {$orderway}, a.aid desc";
  112. } else {
  113. $orderby = "a.aid desc";
  114. }
  115. // 数据查询,搜索出主键ID的值
  116. $SqlQuery = Db::name('archives')->alias('a')->where($condition)->where($conditionNew)->fetchSql()->count('aid');
  117. $count = Db::name('sql_cache_table')->where(['sql_md5'=>md5($SqlQuery)])->getField('sql_result');
  118. $count = ($count < 0) ? 0 : $count;
  119. if (empty($count)) {
  120. $count = Db::name('archives')->alias('a')->where($condition)->where($conditionNew)->count('aid');
  121. /*添加查询执行语句到mysql缓存表*/
  122. $SqlCacheTable = [
  123. 'sql_name' => '|special|' . $this->channeltype . '|',
  124. 'sql_result' => $count,
  125. 'sql_md5' => md5($SqlQuery),
  126. 'sql_query' => $SqlQuery,
  127. 'add_time' => getTime(),
  128. 'update_time' => getTime(),
  129. ];
  130. if (!empty($FlagNew)) $SqlCacheTable['sql_name'] = $SqlCacheTable['sql_name'] . $FlagNew . '|';
  131. if (!empty($typeid)) $SqlCacheTable['sql_name'] = $SqlCacheTable['sql_name'] . $typeid . '|';
  132. if (!empty($keywords)) $SqlCacheTable['sql_name'] = '|special|keywords|';
  133. Db::name('sql_cache_table')->insertGetId($SqlCacheTable);
  134. /*END*/
  135. }
  136. $Page = new Page($count, config('paginate.list_rows'));
  137. $list = [];
  138. if (!empty($count)) {
  139. $limit = $count > config('paginate.list_rows') ? $Page->firstRow.','.$Page->listRows : $count;
  140. $list = Db::name('archives')
  141. ->field("a.aid")
  142. ->alias('a')
  143. ->where($condition)
  144. ->where($conditionNew)
  145. ->order($orderby)
  146. ->limit($limit)
  147. ->getAllWithIndex('aid');
  148. if (!empty($list)) {
  149. $aids = array_keys($list);
  150. $fields = "b.*, a.*, a.aid as aid";
  151. $row = Db::name('archives')
  152. ->field($fields)
  153. ->alias('a')
  154. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  155. ->where('a.aid', 'in', $aids)
  156. ->getAllWithIndex('aid');
  157. foreach ($list as $key => $val) {
  158. $row[$val['aid']]['arcurl'] = get_arcurl($row[$val['aid']]);
  159. $row[$val['aid']]['litpic'] = handle_subdir_pic($row[$val['aid']]['litpic']);
  160. $list[$key] = $row[$val['aid']];
  161. }
  162. }
  163. }
  164. $show = $Page->show();
  165. $assign_data['page'] = $show;
  166. $assign_data['list'] = $list;
  167. $assign_data['pager'] = $Page;
  168. $assign_data['typeid'] = $typeid;
  169. $assign_data['tab'] = input('param.tab/d', 3);// 选项卡
  170. $assign_data['seo_pseudo'] = tpCache('global.seo_pseudo');// 前台URL模式
  171. $assign_data['archives_flags'] = model('ArchivesFlag')->getList();// 文档属性
  172. $assign_data['arctype_info'] = $typeid > 0 ? Db::name('arctype')->field('typename')->find($typeid) : [];// 当前栏目信息
  173. $this->assign($assign_data);
  174. return $this->fetch();
  175. }
  176. /**
  177. * 添加
  178. */
  179. public function add()
  180. {
  181. $admin_info = session('admin_info');
  182. $auth_role_info = $admin_info['auth_role_info'];
  183. $this->assign('auth_role_info', $auth_role_info);
  184. $this->assign('admin_info', $admin_info);
  185. if (IS_POST) {
  186. $post = input('post.');
  187. model('Archives')->editor_auto_210607($post);
  188. /* 处理TAG标签 */
  189. if (!empty($post['tags_new'])) {
  190. $post['tags'] = !empty($post['tags']) ? $post['tags'] . ',' . $post['tags_new'] : $post['tags_new'];
  191. unset($post['tags_new']);
  192. }
  193. $post['tags'] = explode(',', $post['tags']);
  194. $post['tags'] = array_unique($post['tags']);
  195. $post['tags'] = implode(',', $post['tags']);
  196. /* END */
  197. $content = empty($post['addonFieldExt']['content']) ? '' : htmlspecialchars_decode($post['addonFieldExt']['content']);
  198. // 根据标题自动提取相关的关键字
  199. $seo_keywords = $post['seo_keywords'];
  200. if (!empty($seo_keywords)) {
  201. $seo_keywords = str_replace(',', ',', $seo_keywords);
  202. } else {
  203. // $seo_keywords = get_split_word($post['title'], $content);
  204. }
  205. // 自动获取内容第一张图片作为封面图
  206. $is_remote = !empty($post['is_remote']) ? $post['is_remote'] : 0;
  207. $litpic = '';
  208. if ($is_remote == 1) {
  209. $litpic = $post['litpic_remote'];
  210. } else {
  211. $litpic = $post['litpic_local'];
  212. }
  213. if (empty($litpic)) {
  214. $litpic = get_html_first_imgurl($content);
  215. }
  216. $post['litpic'] = $litpic;
  217. /*是否有封面图*/
  218. if (empty($post['litpic'])) {
  219. $is_litpic = 0; // 无封面图
  220. } else {
  221. $is_litpic = 1; // 有封面图
  222. }
  223. // SEO描述
  224. $seo_description = '';
  225. if (empty($post['seo_description']) && !empty($content)) {
  226. $seo_description = @msubstr(checkStrHtml($content), 0, config('global.arc_seo_description_length'), false);
  227. } else {
  228. $seo_description = $post['seo_description'];
  229. }
  230. // 外部链接跳转
  231. $jumplinks = '';
  232. $is_jump = isset($post['is_jump']) ? $post['is_jump'] : 0;
  233. if (intval($is_jump) > 0) {
  234. $jumplinks = $post['jumplinks'];
  235. }
  236. // 模板文件,如果文档模板名与栏目指定的一致,默认就为空。让它跟随栏目的指定而变
  237. if ($post['type_tempview'] == $post['tempview']) {
  238. unset($post['type_tempview']);
  239. unset($post['tempview']);
  240. }
  241. //处理自定义文件名,仅由字母数字下划线和短横杆组成,大写强制转换为小写
  242. $htmlfilename = trim($post['htmlfilename']);
  243. if (!empty($htmlfilename)) {
  244. $htmlfilename = preg_replace("/[^\x{4e00}-\x{9fa5}\w\-]+/u", "-", $htmlfilename);
  245. // $htmlfilename = strtolower($htmlfilename);
  246. //判断是否存在相同的自定义文件名
  247. $map = [
  248. 'htmlfilename' => $htmlfilename,
  249. 'lang' => $this->admin_lang,
  250. ];
  251. if (!empty($post['typeid'])) {
  252. $map['typeid'] = array('eq', $post['typeid']);
  253. }
  254. $filenameCount = Db::name('archives')->where($map)->count();
  255. if (!empty($filenameCount)) {
  256. $this->error("同栏目下,自定义文件名已存在!");
  257. } else if (preg_match('/^(\d+)$/i', $htmlfilename)) {
  258. $this->error("自定义文件名不能纯数字,会与文档ID冲突!");
  259. }
  260. } else {
  261. // 处理外贸链接
  262. if (is_dir('./weapp/Waimao/')) {
  263. $waimaoLogic = new \weapp\Waimao\logic\WaimaoLogic;
  264. $waimaoLogic->get_new_htmlfilename($htmlfilename, $post, 'add', $this->globalConfig);
  265. }
  266. }
  267. $post['htmlfilename'] = $htmlfilename;
  268. //做自动通过审核判断
  269. if ($admin_info['role_id'] > 0 && $auth_role_info['check_oneself'] < 1) {
  270. $post['arcrank'] = -1;
  271. }
  272. // 副栏目
  273. if (isset($post['stypeid'])) {
  274. $post['stypeid'] = preg_replace('/([^\d\,\,]+)/i', ',', $post['stypeid']);
  275. $post['stypeid'] = str_replace(',', ',', $post['stypeid']);
  276. $post['stypeid'] = trim($post['stypeid'], ',');
  277. $post['stypeid'] = str_replace(",{$post['typeid']},", ',', ",{$post['stypeid']},");
  278. $post['stypeid'] = trim($post['stypeid'], ',');
  279. }
  280. // --存储数据
  281. $newData = array(
  282. 'title'=> trim($post['title']),
  283. 'typeid'=> empty($post['typeid']) ? 0 : $post['typeid'],
  284. 'channel' => $this->channeltype,
  285. 'is_b' => empty($post['is_b']) ? 0 : $post['is_b'],
  286. 'is_head' => empty($post['is_head']) ? 0 : $post['is_head'],
  287. 'is_special' => empty($post['is_special']) ? 0 : $post['is_special'],
  288. 'is_recom' => empty($post['is_recom']) ? 0 : $post['is_recom'],
  289. 'is_roll' => empty($post['is_roll']) ? 0 : $post['is_roll'],
  290. 'is_slide' => empty($post['is_slide']) ? 0 : $post['is_slide'],
  291. 'is_diyattr' => empty($post['is_diyattr']) ? 0 : $post['is_diyattr'],
  292. 'editor_remote_img_local'=> empty($post['editor_remote_img_local']) ? 0 : $post['editor_remote_img_local'],
  293. 'editor_img_clear_link' => empty($post['editor_img_clear_link']) ? 0 : $post['editor_img_clear_link'],
  294. 'is_jump' => $is_jump,
  295. 'is_litpic' => $is_litpic,
  296. 'jumplinks' => $jumplinks,
  297. 'origin' => empty($post['origin']) ? '网络' : $post['origin'],
  298. 'seo_keywords' => $seo_keywords,
  299. 'seo_description' => $seo_description,
  300. 'admin_id' => session('admin_info.admin_id'),
  301. 'lang' => $this->admin_lang,
  302. 'sort_order' => 100,
  303. 'crossed_price' => empty($post['crossed_price']) ? 0 : floatval($post['crossed_price']),
  304. 'add_time' => strtotime($post['add_time']),
  305. 'update_time' => strtotime($post['add_time']),
  306. );
  307. $data = array_merge($post, $newData);
  308. $aid = Db::name('archives')->insertGetId($data);
  309. $_POST['aid'] = $aid;
  310. if ($aid) {
  311. // ---------后置操作
  312. model('Special')->afterSave($aid, $data, 'add');
  313. // 添加查询执行语句到mysql缓存表
  314. model('SqlCacheTable')->InsertSqlCacheTable();
  315. // ---------end
  316. adminLog('新增专题:'.$data['title']);
  317. // 生成静态页面代码
  318. $successData = [
  319. 'aid' => $aid,
  320. 'tid' => $post['typeid'],
  321. ];
  322. $this->success("操作成功!", null, $successData);
  323. exit;
  324. }
  325. $this->error("操作失败!");
  326. exit;
  327. }
  328. $typeid = input('param.typeid/d', 0);
  329. $assign_data['typeid'] = $typeid; // 栏目ID
  330. // 栏目信息
  331. $arctypeInfo = Db::name('arctype')->find($typeid);
  332. /*允许发布文档列表的栏目*/
  333. $arctype_html = allow_release_arctype($typeid, array($this->channeltype));
  334. $assign_data['arctype_html'] = $arctype_html;
  335. /*--end*/
  336. // 阅读权限
  337. $arcrank_list = get_arcrank_list();
  338. $assign_data['arcrank_list'] = $arcrank_list;
  339. /*模板列表*/
  340. $archivesLogic = new \app\admin\logic\ArchivesLogic;
  341. $templateList = $archivesLogic->getTemplateList($this->nid);
  342. $this->assign('templateList', $templateList);
  343. /*--end*/
  344. /*默认模板文件*/
  345. $tempview = 'view_'.$this->nid.'.'.config('template.view_suffix');
  346. !empty($arctypeInfo['tempview']) && $tempview = $arctypeInfo['tempview'];
  347. $this->assign('tempview', $tempview);
  348. /*--end*/
  349. // URL模式
  350. $tpcache = config('tpcache');
  351. $assign_data['seo_pseudo'] = !empty($tpcache['seo_pseudo']) ? $tpcache['seo_pseudo'] : 1;
  352. /*--end*/
  353. /*节点允许发布文档列表的栏目*/
  354. $allow_release_channel = config('global.allow_release_channel');
  355. $key_tmp = array_search('7', $allow_release_channel);
  356. if (is_numeric($key_tmp) && 0 <= $key_tmp) {
  357. unset($allow_release_channel[$key_tmp]);
  358. }
  359. $node_select_html = allow_release_arctype(0, $allow_release_channel);
  360. $assign_data['node_select_html'] = $node_select_html;
  361. /*--end*/
  362. /* 节点模型列表 */
  363. $allow_release_channel = config('global.allow_release_channel');
  364. $node_channeltype_list = \think\Cache::get('extra_global_channeltype');
  365. foreach ($node_channeltype_list as $key => $val) {
  366. if (!in_array($val['id'], $allow_release_channel)) {
  367. unset($node_channeltype_list[$val['id']]);
  368. }
  369. }
  370. unset($node_channeltype_list[7]);
  371. $assign_data['node_channeltype_list'] = $node_channeltype_list;
  372. /*--end*/
  373. // 文档默认浏览量
  374. $globalConfig = tpCache('global');
  375. if (isset($globalConfig['other_arcclick']) && 0 <= $globalConfig['other_arcclick']) {
  376. $arcclick_arr = explode("|", $globalConfig['other_arcclick']);
  377. if (count($arcclick_arr) > 1) {
  378. $assign_data['rand_arcclick'] = mt_rand($arcclick_arr[0], $arcclick_arr[1]);
  379. } else {
  380. $assign_data['rand_arcclick'] = intval($arcclick_arr[0]);
  381. }
  382. }else{
  383. $arcclick_config['other_arcclick'] = '500|1000';
  384. tpCache('other', $arcclick_config);
  385. $assign_data['rand_arcclick'] = mt_rand(500, 1000);
  386. }
  387. /*文档属性*/
  388. $assign_data['archives_flags'] = model('ArchivesFlag')->getList();
  389. $channelRow = Db::name('channeltype')->where('id', $this->channeltype)->find();
  390. $assign_data['channelRow'] = $channelRow;
  391. // 来源列表
  392. $system_originlist = tpSetting('system.system_originlist');
  393. $system_originlist = json_decode($system_originlist, true);
  394. $system_originlist = !empty($system_originlist) ? $system_originlist : [];
  395. $assign_data['system_originlist_str'] = implode(PHP_EOL, $system_originlist);
  396. $assign_data['system_originlist_0'] = !empty($system_originlist) ? $system_originlist[0] : "";
  397. // 多站点,当用站点域名访问后台,发布文档自动选择当前所属区域
  398. model('Citysite')->auto_location_select($assign_data);
  399. $this->assign($assign_data);
  400. return $this->fetch();
  401. }
  402. /**
  403. * 编辑
  404. */
  405. public function edit()
  406. {
  407. $admin_info = session('admin_info');
  408. $auth_role_info = $admin_info['auth_role_info'];
  409. $this->assign('auth_role_info', $auth_role_info);
  410. $this->assign('admin_info', $admin_info);
  411. if (IS_POST) {
  412. $post = input('post.');
  413. model('Archives')->editor_auto_210607($post);
  414. $post['aid'] = intval($post['aid']);
  415. /* 处理TAG标签 */
  416. if (!empty($post['tags_new'])) {
  417. $post['tags'] = !empty($post['tags']) ? $post['tags'] . ',' . $post['tags_new'] : $post['tags_new'];
  418. unset($post['tags_new']);
  419. }
  420. $post['tags'] = explode(',', $post['tags']);
  421. $post['tags'] = array_unique($post['tags']);
  422. $post['tags'] = implode(',', $post['tags']);
  423. /* END */
  424. $typeid = input('post.typeid/d', 0);
  425. $content = empty($post['addonFieldExt']['content']) ? '' : htmlspecialchars_decode($post['addonFieldExt']['content']);
  426. // 根据标题自动提取相关的关键字
  427. $seo_keywords = $post['seo_keywords'];
  428. if (!empty($seo_keywords)) {
  429. $seo_keywords = str_replace(',', ',', $seo_keywords);
  430. } else {
  431. // $seo_keywords = get_split_word($post['title'], $content);
  432. }
  433. // 自动获取内容第一张图片作为封面图
  434. $is_remote = !empty($post['is_remote']) ? $post['is_remote'] : 0;
  435. $litpic = '';
  436. if ($is_remote == 1) {
  437. $litpic = $post['litpic_remote'];
  438. } else {
  439. $litpic = $post['litpic_local'];
  440. }
  441. if (empty($litpic)) {
  442. $litpic = get_html_first_imgurl($content);
  443. }
  444. $post['litpic'] = $litpic;
  445. /*是否有封面图*/
  446. if (empty($post['litpic'])) {
  447. $is_litpic = 0; // 无封面图
  448. } else {
  449. $is_litpic = !empty($post['is_litpic']) ? $post['is_litpic'] : 0; // 有封面图
  450. }
  451. // 勾选后SEO描述将随正文内容更新
  452. $basic_update_seo_description = empty($post['basic_update_seo_description']) ? 0 : 1;
  453. if (is_language()) {
  454. $langRow = \think\Db::name('language')->order('id asc')
  455. ->cache(true, EYOUCMS_CACHE_TIME, 'language')
  456. ->select();
  457. foreach ($langRow as $key => $val) {
  458. tpCache('basic', ['basic_update_seo_description'=>$basic_update_seo_description], $val['mark']);
  459. }
  460. } else {
  461. tpCache('basic', ['basic_update_seo_description'=>$basic_update_seo_description]);
  462. }
  463. /*--end*/
  464. // SEO描述
  465. $seo_description = '';
  466. if (!empty($basic_update_seo_description) || empty($post['seo_description'])) {
  467. $seo_description = @msubstr(checkStrHtml($content), 0, config('global.arc_seo_description_length'), false);
  468. } else {
  469. $seo_description = $post['seo_description'];
  470. }
  471. // --外部链接
  472. $jumplinks = '';
  473. $is_jump = isset($post['is_jump']) ? $post['is_jump'] : 0;
  474. if (intval($is_jump) > 0) {
  475. $jumplinks = $post['jumplinks'];
  476. }
  477. // 模板文件,如果文档模板名与栏目指定的一致,默认就为空。让它跟随栏目的指定而变
  478. if ($post['type_tempview'] == $post['tempview']) {
  479. unset($post['type_tempview']);
  480. unset($post['tempview']);
  481. }
  482. // 同步栏目切换模型之后的文档模型
  483. $channel = Db::name('arctype')->where(['id'=>$typeid])->getField('current_channel');
  484. //处理自定义文件名,仅由字母数字下划线和短横杆组成,大写强制转换为小写
  485. $htmlfilename = trim($post['htmlfilename']);
  486. if (!empty($htmlfilename)) {
  487. $htmlfilename = preg_replace("/[^\x{4e00}-\x{9fa5}\w\-]+/u", "-", $htmlfilename);
  488. // $htmlfilename = strtolower($htmlfilename);
  489. //判断是否存在相同的自定义文件名
  490. $map = [
  491. 'aid' => ['NEQ', $post['aid']],
  492. 'htmlfilename' => $htmlfilename,
  493. 'lang' => $this->admin_lang,
  494. ];
  495. if (!empty($post['typeid'])) {
  496. $map['typeid'] = array('eq', $post['typeid']);
  497. }
  498. $filenameCount = Db::name('archives')->where($map)->count();
  499. if (!empty($filenameCount)) {
  500. $this->error("同栏目下,自定义文件名已存在!");
  501. } else if (preg_match('/^(\d+)$/i', $htmlfilename)) {
  502. $this->error("自定义文件名不能纯数字,会与文档ID冲突!");
  503. }
  504. } else {
  505. // 处理外贸链接
  506. if (is_dir('./weapp/Waimao/')) {
  507. $waimaoLogic = new \weapp\Waimao\logic\WaimaoLogic;
  508. $waimaoLogic->get_new_htmlfilename($htmlfilename, $post, 'edit', $this->globalConfig);
  509. }
  510. }
  511. $post['htmlfilename'] = $htmlfilename;
  512. //做未通过审核文档不允许修改文档状态操作
  513. if ($admin_info['role_id'] > 0 && $auth_role_info['check_oneself'] < 1) {
  514. $old_archives_arcrank = Db::name('archives')->where(['aid' => $post['aid']])->getField("arcrank");
  515. if ($old_archives_arcrank < 0) {
  516. unset($post['arcrank']);
  517. }
  518. }
  519. // 副栏目
  520. if (isset($post['stypeid'])) {
  521. $post['stypeid'] = preg_replace('/([^\d\,\,]+)/i', ',', $post['stypeid']);
  522. $post['stypeid'] = str_replace(',', ',', $post['stypeid']);
  523. $post['stypeid'] = trim($post['stypeid'], ',');
  524. $post['stypeid'] = str_replace(",{$typeid},", ',', ",{$post['stypeid']},");
  525. $post['stypeid'] = trim($post['stypeid'], ',');
  526. }
  527. // --存储数据
  528. $newData = array(
  529. 'title'=> trim($post['title']),
  530. 'typeid'=> $typeid,
  531. 'channel' => $channel,
  532. 'is_b' => empty($post['is_b']) ? 0 : $post['is_b'],
  533. 'is_head' => empty($post['is_head']) ? 0 : $post['is_head'],
  534. 'is_special' => empty($post['is_special']) ? 0 : $post['is_special'],
  535. 'is_recom' => empty($post['is_recom']) ? 0 : $post['is_recom'],
  536. 'is_roll' => empty($post['is_roll']) ? 0 : $post['is_roll'],
  537. 'is_slide' => empty($post['is_slide']) ? 0 : $post['is_slide'],
  538. 'is_diyattr' => empty($post['is_diyattr']) ? 0 : $post['is_diyattr'],
  539. 'editor_remote_img_local'=> empty($post['editor_remote_img_local']) ? 0 : $post['editor_remote_img_local'],
  540. 'editor_img_clear_link' => empty($post['editor_img_clear_link']) ? 0 : $post['editor_img_clear_link'],
  541. 'is_jump' => $is_jump,
  542. 'is_litpic' => $is_litpic,
  543. 'jumplinks' => $jumplinks,
  544. 'seo_keywords' => $seo_keywords,
  545. 'seo_description' => $seo_description,
  546. 'crossed_price' => empty($post['crossed_price']) ? 0 : floatval($post['crossed_price']),
  547. 'add_time' => strtotime($post['add_time']),
  548. 'update_time' => getTime(),
  549. );
  550. $data = array_merge($post, $newData);
  551. $r = Db::name('archives')->where([
  552. 'aid' => $data['aid'],
  553. 'lang' => $this->admin_lang,
  554. ])->update($data);
  555. if ($r) {
  556. // ---------后置操作
  557. model('Special')->afterSave($data['aid'], $data, 'edit');
  558. // ---------end
  559. adminLog('编辑专题:'.$data['title']);
  560. // 生成静态页面代码
  561. $successData = [
  562. 'aid' => $data['aid'],
  563. 'tid' => $typeid,
  564. ];
  565. $this->success("操作成功!", null, $successData);
  566. exit;
  567. }
  568. $this->error("操作失败!");
  569. exit;
  570. }
  571. $assign_data = array();
  572. $id = input('id/d');
  573. $info = model('Special')->getInfo($id, null, false);
  574. if (empty($info)) {
  575. $this->error('数据不存在,请联系管理员!');
  576. exit;
  577. }
  578. /*兼容采集没有归属栏目的文档*/
  579. if (empty($info['channel'])) {
  580. $channelRow = Db::name('channeltype')->field('id as channel')
  581. ->where('id',$this->channeltype)
  582. ->find();
  583. $info = array_merge($info, $channelRow);
  584. }
  585. /*--end*/
  586. $typeid = $info['typeid'];
  587. $assign_data['typeid'] = $typeid;
  588. // 副栏目
  589. $stypeid_arr = [];
  590. if (!empty($info['stypeid'])) {
  591. $info['stypeid'] = trim($info['stypeid'], ',');
  592. $stypeid_arr = Db::name('arctype')->field('id,typename')->where(['id'=>['IN', $info['stypeid']],'is_del'=>0])->select();
  593. }
  594. $assign_data['stypeid_arr'] = $stypeid_arr;
  595. // 栏目信息
  596. $arctypeInfo = Db::name('arctype')->find($typeid);
  597. $info['channel'] = $arctypeInfo['current_channel'];
  598. if (is_http_url($info['litpic'])) {
  599. $info['is_remote'] = 1;
  600. $info['litpic_remote'] = handle_subdir_pic($info['litpic']);
  601. } else {
  602. $info['is_remote'] = 0;
  603. $info['litpic_local'] = handle_subdir_pic($info['litpic']);
  604. }
  605. // SEO描述
  606. // if (!empty($info['seo_description'])) {
  607. // $info['seo_description'] = @msubstr(checkStrHtml($info['seo_description']), 0, config('global.arc_seo_description_length'), false);
  608. // }
  609. $assign_data['field'] = $info;
  610. /*允许发布文档列表的栏目,文档所在模型以栏目所在模型为主,兼容切换模型之后的数据编辑*/
  611. $arctype_html = allow_release_arctype($typeid, array($info['channel']));
  612. $assign_data['arctype_html'] = $arctype_html;
  613. /*--end*/
  614. // 阅读权限
  615. $arcrank_list = get_arcrank_list();
  616. $assign_data['arcrank_list'] = $arcrank_list;
  617. /*模板列表*/
  618. $archivesLogic = new \app\admin\logic\ArchivesLogic;
  619. $templateList = $archivesLogic->getTemplateList($this->nid);
  620. $this->assign('templateList', $templateList);
  621. /*--end*/
  622. /*默认模板文件*/
  623. $tempview = $info['tempview'];
  624. empty($tempview) && $tempview = $arctypeInfo['tempview'];
  625. $this->assign('tempview', $tempview);
  626. /*--end*/
  627. // URL模式
  628. $tpcache = config('tpcache');
  629. $assign_data['seo_pseudo'] = !empty($tpcache['seo_pseudo']) ? $tpcache['seo_pseudo'] : 1;
  630. /*--end*/
  631. /*节点允许发布文档列表的栏目*/
  632. $allow_release_channel = config('global.allow_release_channel');
  633. $key_tmp = array_search('7', $allow_release_channel);
  634. if (is_numeric($key_tmp) && 0 <= $key_tmp) {
  635. unset($allow_release_channel[$key_tmp]);
  636. }
  637. $node_select_html = allow_release_arctype(0, $allow_release_channel);
  638. $assign_data['node_select_html'] = $node_select_html;
  639. /*--end*/
  640. /* 节点模型列表 */
  641. $allow_release_channel = config('global.allow_release_channel');
  642. $node_channeltype_list = \think\Cache::get('extra_global_channeltype');
  643. foreach ($node_channeltype_list as $key => $val) {
  644. if (!in_array($val['id'], $allow_release_channel)) {
  645. unset($node_channeltype_list[$val['id']]);
  646. }
  647. }
  648. unset($node_channeltype_list[7]);
  649. $assign_data['node_channeltype_list'] = $node_channeltype_list;
  650. /*--end*/
  651. /*节点数据*/
  652. $specialNodeList = model('SpecialNode')->getList($id);
  653. $aidlists = '';
  654. foreach ($specialNodeList as $key => $value) {
  655. $aidlists .= 0 === intval($key) ? $value['aidlist'] : ',' . $value['aidlist'];
  656. }
  657. if (!empty($aidlists)) {
  658. $where = [
  659. 'aid' => ['IN', $aidlists]
  660. ];
  661. $archives = Db::name('archives')->field('aid, title')->where($where)->getAllWithIndex('aid');
  662. foreach ($specialNodeList as $key => $value) {
  663. $value['archivesList'] = [];
  664. $aidlists = !empty($value['aidlist']) ? explode(',', $value['aidlist']) : [];
  665. foreach ($aidlists as $value_1) {
  666. if (!empty($archives[$value_1])) array_push($value['archivesList'], $archives[$value_1]);
  667. }
  668. $value['htmllist'] = json_encode($value['archivesList']);
  669. $specialNodeList[$key] = $value;
  670. }
  671. }
  672. $assign_data['specialNodeList'] = $specialNodeList;
  673. /*--end*/
  674. /*文档属性*/
  675. $assign_data['archives_flags'] = model('ArchivesFlag')->getList();
  676. $channelRow = Db::name('channeltype')->where('id', $this->channeltype)->find();
  677. $assign_data['channelRow'] = $channelRow;
  678. // 来源列表
  679. $system_originlist = tpSetting('system.system_originlist');
  680. $system_originlist = json_decode($system_originlist, true);
  681. $system_originlist = !empty($system_originlist) ? $system_originlist : [];
  682. $assign_data['system_originlist_str'] = implode(PHP_EOL, $system_originlist);
  683. $this->assign($assign_data);
  684. return $this->fetch();
  685. }
  686. /**
  687. * 删除
  688. */
  689. public function del()
  690. {
  691. if (IS_POST) {
  692. $archivesLogic = new \app\admin\logic\ArchivesLogic;
  693. $archivesLogic->del([], 0, 'special');
  694. }
  695. }
  696. /**
  697. * 选择节点文档
  698. * @return [type] [description]
  699. */
  700. public function ajax_node_archives_list()
  701. {
  702. $assign_data = array();
  703. $condition = array();
  704. // 获取到所有URL参数
  705. $param = input('param.');
  706. $typeid = input('param.typeid/d');
  707. $channels = input('param.channel/s');
  708. // 应用搜索条件
  709. foreach (['keywords','typeid','channel'] as $key) {
  710. if ($key == 'keywords' && !empty($param[$key])) {
  711. $param[$key] = trim($param[$key]);
  712. $condition['a.title'] = array('LIKE', "%{$param[$key]}%");
  713. } else if ($key == 'typeid' && !empty($param[$key])) {
  714. $typeid = $param[$key];
  715. $hasRow = model('Arctype')->getHasChildren($typeid);
  716. $typeids = get_arr_column($hasRow, 'id');
  717. /*权限控制 by 小虎哥*/
  718. $admin_info = session('admin_info');
  719. if (0 < intval($admin_info['role_id'])) {
  720. $auth_role_info = $admin_info['auth_role_info'];
  721. if(! empty($auth_role_info)){
  722. if(isset($auth_role_info['only_oneself']) && 1 == $auth_role_info['only_oneself']){
  723. $condition['a.admin_id'] = $admin_info['admin_id'];
  724. }
  725. if(! empty($auth_role_info['permission']['arctype'])){
  726. if (!empty($typeid)) {
  727. $typeids = array_intersect($typeids, $auth_role_info['permission']['arctype']);
  728. }
  729. }
  730. }
  731. }
  732. /*--end*/
  733. $condition['a.typeid'] = array('IN', $typeids);
  734. } else if ($key == 'channel') {
  735. if (empty($param[$key])) {
  736. $allow_release_channel = config('global.allow_release_channel');
  737. $key_tmp = array_search('7', $allow_release_channel);
  738. if (is_numeric($key_tmp) && 0 <= $key_tmp) {
  739. unset($allow_release_channel[$key_tmp]);
  740. }
  741. $param[$key] = implode(',', $allow_release_channel);
  742. }
  743. $condition['a.'.$key] = array('in', explode(',', $param[$key]));
  744. } else if (!empty($param[$key])) {
  745. $condition['a.'.$key] = array('eq', $param[$key]);
  746. }
  747. }
  748. // 审核通过
  749. $condition['a.arcrank'] = array('gt', -1);
  750. /*多语言*/
  751. $condition['a.lang'] = array('eq', $this->admin_lang);
  752. /*回收站数据不显示*/
  753. $condition['a.is_del'] = array('eq', 0);
  754. /**
  755. * 数据查询,搜索出主键ID的值
  756. */
  757. $count = Db::name('archives')->alias('a')->where($condition)->count('aid');// 查询满足要求的总记录数
  758. $Page = new Page($count, config('paginate.list_rows'));// 实例化分页类 传入总记录数和每页显示的记录数
  759. $list = Db::name('archives')
  760. ->field("a.aid")
  761. ->alias('a')
  762. ->where($condition)
  763. ->order('a.sort_order asc, a.aid desc')
  764. ->limit($Page->firstRow.','.$Page->listRows)
  765. ->getAllWithIndex('aid');
  766. /**
  767. * 完善数据集信息
  768. * 在数据量大的情况下,经过优化的搜索逻辑,先搜索出主键ID,再通过ID将其他信息补充完整;
  769. */
  770. if ($list) {
  771. $aids = array_keys($list);
  772. $fields = "b.*, a.*, a.aid as aid";
  773. $row = Db::name('archives')
  774. ->field($fields)
  775. ->alias('a')
  776. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  777. ->where('a.aid', 'in', $aids)
  778. ->getAllWithIndex('aid');
  779. foreach ($list as $key => $val) {
  780. $list[$key] = $row[$val['aid']];
  781. }
  782. }
  783. $show = $Page->show(); // 分页显示输出
  784. $assign_data['page'] = $show; // 赋值分页输出
  785. $assign_data['list'] = $list; // 赋值数据集
  786. $assign_data['pager'] = $Page; // 赋值分页对象
  787. /*允许发布文档列表的栏目*/
  788. $allow_release_channel = !empty($channels) ? explode(',', $channels) : config('global.allow_release_channel');
  789. $key_tmp = array_search('7', $allow_release_channel);
  790. if (is_numeric($key_tmp) && 0 <= $key_tmp) {
  791. unset($allow_release_channel[$key_tmp]);
  792. }
  793. $assign_data['arctype_html'] = allow_release_arctype($typeid, $allow_release_channel);
  794. /*--end*/
  795. $this->assign($assign_data);
  796. return $this->fetch();
  797. }
  798. //帮助
  799. public function help()
  800. {
  801. $system_originlist = tpSetting('system.system_originlist');
  802. $system_originlist = json_decode($system_originlist, true);
  803. $system_originlist = !empty($system_originlist) ? $system_originlist : [];
  804. $assign_data['system_originlist_str'] = implode(PHP_EOL, $system_originlist);
  805. $this->assign($assign_data);
  806. return $this->fetch();
  807. }
  808. /**
  809. * 用于专题节点文档来源的逻辑
  810. * @return [type] [description]
  811. */
  812. public function ajax_recordfile()
  813. {
  814. \think\Session::pause(); // 暂停session,防止session阻塞机制
  815. if (IS_AJAX) {
  816. $opt = input('param.opt/s');
  817. $value = input('param.value/s');
  818. $filename = ROOT_PATH . 'data/conf/nodeaids_1619141574.txt';
  819. if ('set' == $opt) {
  820. $redata = $this->writeSpecialaidsFile($value);
  821. if (true !== $redata) {
  822. $this->error($redata);
  823. }
  824. $this->success('写入成功!');
  825. }
  826. else if ('get' == $opt) {
  827. $nodeaids = $this->readSpecialaidsFile();
  828. $list = [];
  829. if (!empty($nodeaids)) {
  830. $aids = explode(',', $nodeaids);
  831. $list = Db::name('archives')->field('aid,title')->where(['aid'=>['IN', $aids]])->orderRaw("FIND_IN_SET(aid, '{$nodeaids}')")->select();
  832. }
  833. $this->success('读取成功!', null, ['nodeaids'=>$nodeaids,'list'=>$list]);
  834. }
  835. }
  836. }
  837. /**
  838. * 读取节点文档列表的aid文件 - 应用于专题发布、编辑的文档
  839. * @return [type] [description]
  840. */
  841. private function readSpecialaidsFile()
  842. {
  843. $node_aids = '';
  844. $filename = ROOT_PATH . 'data/conf/nodeaids_1619141574.txt';
  845. if (file_exists($filename)) {
  846. $len = filesize($filename);
  847. if (!empty($len) && $len > 0) {
  848. $fp = fopen($filename, 'r');
  849. $node_aids = fread($fp, $len);
  850. fclose($fp);
  851. $node_aids = $node_aids ? $node_aids : '';
  852. }
  853. }
  854. return $node_aids;
  855. }
  856. /**
  857. * 写入节点文档列表的aid文件 - 应用于专题发布、编辑的文档
  858. * @return [type] [description]
  859. */
  860. private function writeSpecialaidsFile($value = '')
  861. {
  862. $filename = ROOT_PATH . 'data/conf/nodeaids_1619141574.txt';
  863. if (!file_exists($filename)) tp_mkdir(dirname($filename));
  864. $fp = fopen($filename, "w+");
  865. if (empty($fp)) {
  866. return "请设置" . $filename . "的权限为744";
  867. } else {
  868. if (fwrite($fp, $value)) {
  869. fclose($fp);
  870. }
  871. }
  872. return true;
  873. }
  874. }