Sin descripción
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Article.php 41KB

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