Ei kuvausta
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.

Guestbook.php 41KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937
  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 app\common\logic\ArctypeLogic;
  17. class Guestbook extends Base
  18. {
  19. // 模型标识
  20. public $nid = 'guestbook';
  21. // 模型ID
  22. public $channeltype = '';
  23. // 表单类型
  24. public $attrInputTypeArr = array();
  25. public function _initialize()
  26. {
  27. parent::_initialize();
  28. $channeltype_list = config('global.channeltype_list');
  29. $this->channeltype = $channeltype_list[$this->nid];
  30. $this->attrInputTypeArr = config('global.guestbook_attr_input_type');
  31. }
  32. /**
  33. * 留言列表 - 仅是栏目关联的留言
  34. */
  35. public function index()
  36. {
  37. $assign_data = array();
  38. $condition = array();
  39. // 获取到所有GET参数
  40. $get = input('get.');
  41. $typeid = input('typeid/d');
  42. $begin = strtotime(input('param.add_time_begin/s'));
  43. $end = input('param.add_time_end/s');
  44. !empty($end) && $end .= ' 23:59:59';
  45. $end = strtotime($end);
  46. // 应用搜索条件
  47. foreach (['keywords', 'typeid'] as $key) {
  48. if (isset($get[$key]) && $get[$key] !== '') {
  49. if ($key == 'keywords') {
  50. $attr_row = Db::name('guestbook_attr')->field('aid')->where(array('attr_value' => array('LIKE', "%{$get[$key]}%"),'form_type'=>0))->group('aid')->getAllWithIndex('aid');
  51. $aids = array_keys($attr_row);
  52. $condition['a.aid'] = array('IN', $aids);
  53. } else if ($key == 'typeid') {
  54. $condition['a.typeid'] = array('eq', $get[$key]);
  55. } else {
  56. $condition['a.' . $key] = array('eq', $get[$key]);
  57. }
  58. }
  59. }
  60. // 时间检索
  61. if ($begin > 0 && $end > 0) {
  62. $condition['a.add_time'] = array('between',"$begin,$end");
  63. } else if ($begin > 0) {
  64. $condition['a.add_time'] = array('egt', $begin);
  65. } else if ($end > 0) {
  66. $condition['a.add_time'] = array('elt', $end);
  67. }
  68. if (empty($typeid)) {
  69. /*权限控制 by 小虎哥*/
  70. $admin_info = session('admin_info');
  71. if (0 < intval($admin_info['role_id'])) {
  72. $auth_role_info = $admin_info['auth_role_info'];
  73. if(! empty($auth_role_info)){
  74. $is_notaccess = false;
  75. $permission_arctype = !empty($auth_role_info['permission']['arctype']) ? $auth_role_info['permission']['arctype'] : [];
  76. if(! empty($permission_arctype)){
  77. $typeids_tmp = Db::name('arctype')->where(['current_channel'=>8,'lang'=>$this->admin_lang])->cache(true, EYOUCMS_CACHE_TIME, 'arctype')->column('id');
  78. $typeids_tmp = !empty($typeids_tmp) ? $typeids_tmp : [];
  79. $typeids_tmp2 = array_intersect($typeids_tmp, $auth_role_info['permission']['arctype']);
  80. if (!empty($typeids_tmp2)) {
  81. $condition['a.typeid'] = ['IN', $typeids_tmp2];
  82. $is_notaccess = true;
  83. }
  84. }
  85. if (false === $is_notaccess) {
  86. $this->error('您没有操作权限,请联系超级管理员分配权限');
  87. }
  88. }
  89. }
  90. /*--end*/
  91. }
  92. $condition['a.form_type'] = 0;
  93. // 多语言
  94. $condition['a.lang'] = array('eq', $this->admin_lang);
  95. /**
  96. * 数据查询,搜索出主键ID的值
  97. */
  98. $count = Db::name('guestbook')->alias('a')->where($condition)->count('aid');// 查询满足要求的总记录数
  99. $Page = new Page($count, config('paginate.list_rows'));// 实例化分页类 传入总记录数和每页显示的记录数
  100. $list = Db::name('guestbook')
  101. ->field("b.*, a.*")
  102. ->alias('a')
  103. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  104. ->where($condition)
  105. ->order('a.is_read asc, a.add_time desc')
  106. ->limit($Page->firstRow . ',' . $Page->listRows)
  107. ->getAllWithIndex('aid');
  108. /**
  109. * 完善数据集信息
  110. * 在数据量大的情况下,经过优化的搜索逻辑,先搜索出主键ID,再通过ID将其他信息补充完整;
  111. */
  112. $typeids = [];
  113. if ($list) {
  114. $where = [
  115. 'b.aid' => ['IN', array_keys($list)],
  116. 'a.is_showlist' => 1,
  117. 'a.lang' => $this->admin_lang,
  118. 'a.is_del' => 0,
  119. ];
  120. $row = Db::name('guestbook_attribute')
  121. ->field('a.attr_name, a.typeid, b.attr_value, b.aid, b.attr_id,a.attr_input_type')
  122. ->alias('a')
  123. ->join('__GUESTBOOK_ATTR__ b', 'b.attr_id = a.attr_id', 'LEFT')
  124. ->where($where)
  125. ->order('b.aid desc, a.sort_order asc, a.attr_id asc')
  126. ->getAllWithIndex();
  127. $attr_list = array();
  128. foreach ($row as $key => $val) {
  129. $typeids[] = $val['typeid'];
  130. if (9 == $val['attr_input_type']){
  131. //如果是区域类型,转换名称
  132. $val['attr_value'] = Db::name('region')->where('id','in',$val['attr_value'])->column('name');
  133. $val['attr_value'] = implode('',$val['attr_value']);
  134. }else if(10 == $val['attr_input_type']){
  135. $val['attr_value'] = date('Y-m-d H:i:s',$val['attr_value']);
  136. } else if(11 == $val['attr_input_type']) {
  137. $attrValueArr = !empty($val['attr_value']) ? explode(',', $val['attr_value']) : [];
  138. $val['attr_value'] = '';
  139. foreach ($attrValueArr as $_k => $_v) {
  140. if ($_k >= 2) {
  141. break;
  142. }
  143. if (preg_match('/(\.('.tpCache('global.image_type').'))$/i', $_v)) {
  144. if (!stristr($_v, '|')) {
  145. $_v = handle_subdir_pic($_v);
  146. $val['attr_value'] .= "<img src='{$_v}' width='60' height='60' style='float: unset; cursor: pointer;margin: 0px 3px;' onclick=\"Images('{$_v}', 650, 350);\" />";
  147. }
  148. } elseif (preg_match('/(\.('.tpCache('global.file_type').'))$/i', $_v)) {
  149. if (!stristr($_v, '|')) {
  150. $_v = handle_subdir_pic($_v);
  151. $val['attr_value'] .= "<a href='{$_v}' download='".time()."'><img src=\"".ROOT_DIR."/public/static/common/images/file.png\" alt=\"\" style=\"width: 16px; height: 16px;\">文件下载</a>";
  152. }
  153. }
  154. }
  155. }
  156. if (preg_match('/(\.('.tpCache('global.image_type').'))$/i', $val['attr_value'])) {
  157. if (!stristr($val['attr_value'], '|')) {
  158. $val['attr_value'] = handle_subdir_pic($val['attr_value']);
  159. $val['attr_value'] = "<img src='{$val['attr_value']}' width='60' height='60' style='float: unset;cursor: pointer;' onclick=\"Images('{$val['attr_value']}', 650, 350);\" />";
  160. }
  161. }elseif (preg_match('/(\.('.tpCache('global.file_type').'))$/i', $val['attr_value'])){
  162. if (!stristr($val['attr_value'], '|')) {
  163. $val['attr_value'] = handle_subdir_pic($val['attr_value']);
  164. $download_name = preg_replace('/^(.*)\/([^\/]+)\.(\w+)$/i', '${2}', $val['attr_value']);
  165. $val['attr_value'] = "<a href='{$val['attr_value']}' download='".$download_name."'><img src=\"".ROOT_DIR."/public/static/common/images/file.png\" alt=\"\" style=\"width:16px;height:16px;float:unset;\">&nbsp;点击下载</a>";
  166. }
  167. } else {
  168. $val['attr_value'] = str_replace(PHP_EOL, ' | ', $val['attr_value']);
  169. }
  170. $attr_list[$val['aid']][] = $val;
  171. }
  172. foreach ($list as $key => $val) {
  173. $list[$key]['attr_list'] = isset($attr_list[$val['aid']]) ? $attr_list[$val['aid']] : array();
  174. }
  175. }
  176. $tab_list = Db::name('guestbook_attribute')->where([
  177. 'typeid' => ['IN', $typeids],
  178. 'form_type' => 0,
  179. 'is_showlist' => 1,
  180. 'lang' => $this->admin_lang,
  181. 'is_del' => 0,
  182. ])->order('typeid asc, sort_order asc, attr_id asc')->select();
  183. $tab_list = group_same_key($tab_list, 'typeid');
  184. if (!empty($typeid)) {
  185. $tab_list = empty($tab_list[$typeid]) ? [] : $tab_list[$typeid];
  186. }
  187. $assign_data['tab_list'] = $tab_list;
  188. $show = $Page->show(); // 分页显示输出
  189. $assign_data['page'] = $show; // 赋值分页输出
  190. $assign_data['list'] = $list; // 赋值数据集
  191. $assign_data['pager'] = $Page; // 赋值分页对象
  192. // 栏目ID
  193. $assign_data['typeid'] = $typeid; // 栏目ID
  194. /*当前栏目信息*/
  195. $arctype_info = array();
  196. if ($typeid > 0) {
  197. $arctype_info = Db::name('arctype')->field('typename')->find($typeid);
  198. }
  199. $assign_data['arctype_info'] = $arctype_info;
  200. /*--end*/
  201. /*选项卡*/
  202. $tab = input('param.tab/d', 3);
  203. $assign_data['tab'] = $tab;
  204. /*--end*/
  205. $this->assign($assign_data);
  206. return $this->fetch();
  207. }
  208. /**
  209. * 删除
  210. */
  211. public function del()
  212. {
  213. $id_arr = input('del_id/a');
  214. $id_arr = eyIntval($id_arr);
  215. $form_type = input('param.form_type/d', 0);
  216. if (!empty($id_arr)) {
  217. $r = Db::name('guestbook')->where([
  218. 'aid' => ['IN', $id_arr],
  219. 'form_type' => $form_type,
  220. 'lang' => $this->admin_lang,
  221. ])->delete();
  222. if ($r !== false) {
  223. // ---------后置操作
  224. model('Guestbook')->afterDel($id_arr);
  225. // ---------end
  226. adminLog('删除留言-id:' . implode(',', $id_arr));
  227. $this->success('删除成功');
  228. }
  229. }
  230. $this->error('删除失败');
  231. }
  232. //留言表单表单列表
  233. public function attribute_index()
  234. {
  235. $assign_data = array();
  236. $condition = array();
  237. $get = input('get.');
  238. $typeid = input('typeid/d');
  239. foreach (['keywords','typeid'] as $key) {
  240. if (isset($get[$key]) && $get[$key] !== '') {
  241. if ($key == 'keywords') {
  242. $condition['a.attr_name'] = array('LIKE', "%{$get[$key]}%");
  243. } else if ($key == 'typeid') {
  244. $typeids = model('Arctype')->getHasChildren($get[$key]);
  245. $condition['a.typeid'] = array('IN', array_keys($typeids));
  246. } else {
  247. $condition['a.'.$key] = array('eq', $get[$key]);
  248. }
  249. }
  250. }
  251. $condition['a.form_type'] = 0;
  252. $condition['b.id'] = ['gt', 0];
  253. $condition['a.is_del'] = 0;
  254. $condition['a.lang'] = $this->admin_lang;
  255. $count = Db::name('guestbook_attribute')->alias('a')
  256. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  257. ->where($condition)
  258. ->count();
  259. $Page = new Page($count, config('paginate.list_rows'));
  260. $list = Db::name('guestbook_attribute')
  261. ->field("a.attr_id")
  262. ->alias('a')
  263. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  264. ->where($condition)
  265. ->order('a.typeid desc, a.sort_order asc, a.attr_id asc')
  266. ->limit($Page->firstRow.','.$Page->listRows)
  267. ->getAllWithIndex('attr_id');
  268. if ($list) {
  269. $attr_ida = array_keys($list);
  270. $fields = "b.*, a.*,a.attr_id as orgin_attr_id";
  271. $row = Db::name('guestbook_attribute')
  272. ->field($fields)
  273. ->alias('a')
  274. ->join('__ARCTYPE__ b', 'a.typeid = b.id', 'LEFT')
  275. ->where('a.attr_id', 'in', $attr_ida)
  276. ->getAllWithIndex('attr_id');
  277. $row = model('LanguageAttr')->getBindValue($row, 'guestbook_attribute', $this->main_lang); // 获取多语言关联绑定的值
  278. foreach ($row as $key => $val) {
  279. $val['fieldname'] = 'attr_'.$val['attr_id'];
  280. $row[$key] = $val;
  281. }
  282. foreach ($list as $key => $val) {
  283. $list[$key] = $row[$val['attr_id']];
  284. }
  285. }
  286. $show = $Page->show();
  287. $assign_data['page'] = $show;
  288. $assign_data['list'] = $list;
  289. $assign_data['pager'] = $Page;
  290. //获取当前模型栏目
  291. $select_html = allow_release_arctype($typeid, array($this->channeltype));
  292. $typeidNum = substr_count($select_html, '</option>');
  293. $this->assign('select_html',$select_html);
  294. $this->assign('typeidNum',$typeidNum);
  295. $assign_data['typeid'] = $typeid;
  296. //当前栏目信息
  297. $arctype_info = array();
  298. if ($typeid > 0) {
  299. $arctype_info = Db::name('arctype')->field('typename')->find($typeid);
  300. }
  301. $assign_data['arctype_info'] = $arctype_info;
  302. $tab = input('param.tab/d', 3);//选项卡
  303. $assign_data['tab'] = $tab;
  304. $assign_data['attrInputTypeArr'] = $this->attrInputTypeArr; // 表单类型
  305. $this->assign($assign_data);
  306. return $this->fetch();
  307. }
  308. /**
  309. * 新增留言表单
  310. */
  311. public function attribute_add()
  312. {
  313. //防止php超时
  314. function_exists('set_time_limit') && set_time_limit(0);
  315. if (is_language() && empty($this->globalConfig['language_split'])) {
  316. $this->language_access(); // 多语言功能操作权限
  317. }
  318. if(IS_AJAX && IS_POST)//ajax提交验证
  319. {
  320. $model = model('GuestbookAttribute');
  321. $attr_values = str_replace('_', '', input('attr_values')); // 替换特殊字符
  322. $attr_values = str_replace('@', '', $attr_values); // 替换特殊字符
  323. $attr_values = trim($attr_values);
  324. /*过滤重复值*/
  325. $attr_values_arr = explode(PHP_EOL, $attr_values);
  326. foreach ($attr_values_arr as $key => $val) {
  327. $tmp_val = trim($val);
  328. if (empty($tmp_val)) {
  329. unset($attr_values_arr[$key]);
  330. continue;
  331. }
  332. $attr_values_arr[$key] = $tmp_val;
  333. }
  334. $attr_values_arr = array_unique($attr_values_arr);
  335. $attr_values = implode(PHP_EOL, $attr_values_arr);
  336. /*end*/
  337. $post_data = input('post.');
  338. $post_data['attr_values'] = $attr_values;
  339. $attr_input_type = isset($post_data['attr_input_type']) ? $post_data['attr_input_type'] : 0;
  340. /*前台输入是否JS验证*/
  341. $validate_type = 0;
  342. $validate_type_list = config("global.validate_type_list"); // 前台输入验证类型
  343. if (!empty($validate_type_list[$attr_input_type])) {
  344. $validate_type = $attr_input_type;
  345. }
  346. /*end*/
  347. if (9 == $post_data['attr_input_type']) {
  348. if (!empty($post_data['region_data'])) {
  349. $post_data['attr_values'] = serialize($post_data['region_data']);
  350. } else {
  351. $this->error("请选择区域范围!");
  352. }
  353. }
  354. $savedata = array(
  355. 'attr_name' => $post_data['attr_name'],
  356. 'typeid' => $post_data['typeid'],
  357. 'form_type' => 0,
  358. 'attr_input_type' => $attr_input_type,
  359. 'attr_values' => isset($post_data['attr_values']) ? $post_data['attr_values'] : '',
  360. 'is_showlist' => $post_data['is_showlist'],
  361. 'required' => $post_data['required'],
  362. 'real_validate' => $post_data['real_validate'],
  363. 'validate_type' => $validate_type,
  364. 'sort_order' => 100,
  365. 'lang' => $this->admin_lang,
  366. 'add_time' => getTime(),
  367. 'update_time' => getTime(),
  368. );
  369. // 如果是添加手机号码类型则执行
  370. if (!empty($savedata['typeid']) && 6 === intval($savedata['attr_input_type']) && 1 === intval($savedata['real_validate'])) {
  371. // 查询是否已添加需要真实验证的手机号码类型
  372. $where = [
  373. 'typeid' => $savedata['typeid'],
  374. 'form_type' => 0,
  375. 'real_validate' => $savedata['real_validate'],
  376. 'attr_input_type' => $savedata['attr_input_type']
  377. ];
  378. $realValidate = $model->get($where);
  379. if (!empty($realValidate)) $this->error('只能设置一个需要真实验证的手机号码类型');
  380. }
  381. // 数据验证
  382. $validate = \think\Loader::validate('GuestbookAttribute');
  383. if(!$validate->batch()->check($savedata))
  384. {
  385. $error = $validate->getError();
  386. $error_msg = array_values($error);
  387. $return_arr = array(
  388. 'status' => -1,
  389. 'msg' => $error_msg[0],
  390. 'data' => $error,
  391. );
  392. respose($return_arr);
  393. } else {
  394. $model->data($savedata,true); // 收集数据
  395. $model->save(); // 写入数据到数据库
  396. $insertId = $model->getLastInsID();
  397. /*同步留言属性ID到多语言的模板变量里*/
  398. model('GuestbookAttribute')->syn_add_language_attribute($insertId);
  399. /*--end*/
  400. $return_arr = array(
  401. 'status' => 1,
  402. 'msg' => '操作成功',
  403. 'data' => array('url'=>url('Guestbook/attribute_index', array('typeid'=>$post_data['typeid']))),
  404. );
  405. adminLog('新增留言表单:'.$savedata['attr_name']);
  406. respose($return_arr);
  407. }
  408. }
  409. $typeid = input('param.typeid/d', 0);
  410. if ($typeid > 0) {
  411. $select_html = Db::name('arctype')->where('id', $typeid)->getField('typename');
  412. $select_html = !empty($select_html) ? $select_html : '该栏目不存在';
  413. } else {
  414. $arctypeLogic = new ArctypeLogic();
  415. $map = array(
  416. 'channeltype' => $this->channeltype,
  417. );
  418. $arctype_max_level = intval(config('global.arctype_max_level'));
  419. $select_html = $arctypeLogic->arctype_list(0, $typeid, true, $arctype_max_level, $map);
  420. }
  421. $assign_data['select_html'] = $select_html; //
  422. $assign_data['typeid'] = $typeid; // 栏目ID
  423. $assign_data['attrInputTypeArr'] = $this->attrInputTypeArr; // 表单类型
  424. //区域
  425. $China[] = [
  426. 'id' => 0,
  427. 'name' => '全国',
  428. ];
  429. $Province = get_province_list();
  430. $assign_data['Province'] = array_merge($China, $Province);
  431. $this->assign($assign_data);
  432. return $this->fetch();
  433. }
  434. /**
  435. * 编辑留言表单
  436. */
  437. public function attribute_edit()
  438. {
  439. if(IS_AJAX && IS_POST)//ajax提交验证
  440. {
  441. $model = model('GuestbookAttribute');
  442. $attr_values = str_replace('_', '', input('attr_values')); // 替换特殊字符
  443. $attr_values = str_replace('@', '', $attr_values); // 替换特殊字符
  444. $attr_values = trim($attr_values);
  445. /*过滤重复值*/
  446. $attr_values_arr = explode(PHP_EOL, $attr_values);
  447. foreach ($attr_values_arr as $key => $val) {
  448. $tmp_val = trim($val);
  449. if (empty($tmp_val)) {
  450. unset($attr_values_arr[$key]);
  451. continue;
  452. }
  453. $attr_values_arr[$key] = $tmp_val;
  454. }
  455. $attr_values_arr = array_unique($attr_values_arr);
  456. $attr_values = implode(PHP_EOL, $attr_values_arr);
  457. /*end*/
  458. $post_data = input('post.');
  459. $post_data['attr_id'] = intval($post_data['attr_id']);
  460. $post_data['attr_values'] = $attr_values;
  461. $attr_input_type = isset($post_data['attr_input_type']) ? $post_data['attr_input_type'] : 0;
  462. /*前台输入是否JS验证*/
  463. $validate_type = 0;
  464. $validate_type_list = config("global.validate_type_list"); // 前台输入验证类型
  465. if (!empty($validate_type_list[$attr_input_type])) {
  466. $validate_type = $attr_input_type;
  467. }
  468. /*end*/
  469. if (9 == $post_data['attr_input_type']) {
  470. if (!empty($post_data['region_data'])) {
  471. $post_data['attr_values'] = serialize($post_data['region_data']);
  472. } else {
  473. $this->error("请选择区域范围!");
  474. }
  475. }
  476. $savedata = array(
  477. 'attr_id' => $post_data['attr_id'],
  478. 'attr_name' => $post_data['attr_name'],
  479. 'typeid' => $post_data['typeid'],
  480. 'form_type' => 0,
  481. 'attr_input_type' => $attr_input_type,
  482. 'attr_values' => isset($post_data['attr_values']) ? $post_data['attr_values'] : '',
  483. 'is_showlist' => $post_data['is_showlist'],
  484. 'required' => $post_data['required'],
  485. 'real_validate' => $post_data['real_validate'],
  486. 'validate_type' => $validate_type,
  487. 'sort_order' => 100,
  488. 'update_time' => getTime(),
  489. );
  490. // 如果是添加手机号码类型则执行
  491. if (!empty($savedata['typeid']) && 6 === intval($savedata['attr_input_type']) && 1 === intval($savedata['real_validate'])) {
  492. // 查询是否已添加需要真实验证的手机号码类型
  493. $where = [
  494. 'typeid' => $savedata['typeid'],
  495. 'form_type' => 0,
  496. 'attr_id' => ['NEQ', $savedata['attr_id']],
  497. 'real_validate' => $savedata['real_validate'],
  498. 'attr_input_type' => $savedata['attr_input_type']
  499. ];
  500. $realValidate = $model->get($where);
  501. if (!empty($realValidate)) $this->error('只能设置一个需要真实验证的手机号码类型');
  502. }
  503. // 数据验证
  504. $validate = \think\Loader::validate('GuestbookAttribute');
  505. if(!$validate->batch()->check($savedata))
  506. {
  507. $error = $validate->getError();
  508. $error_msg = array_values($error);
  509. $return_arr = array(
  510. 'status' => -1,
  511. 'msg' => $error_msg[0],
  512. 'data' => $error,
  513. );
  514. respose($return_arr);
  515. } else {
  516. $model->data($savedata, true); // 收集数据
  517. $model->isUpdate(true, [
  518. 'attr_id' => $post_data['attr_id'],
  519. ])->save(); // 写入数据到数据库
  520. $return_arr = array(
  521. 'status' => 1,
  522. 'msg' => '操作成功',
  523. 'data' => array('url' => url('Guestbook/attribute_index', array('typeid' => intval($post_data['typeid'])))),
  524. );
  525. adminLog('编辑留言表单:' . $savedata['attr_name']);
  526. respose($return_arr);
  527. }
  528. }
  529. $assign_data = array();
  530. $id = input('id/d');
  531. /*获取多语言关联绑定的值*/
  532. $new_id = model('LanguageAttr')->getBindValue($id, 'guestbook_attribute'); // 多语言
  533. !empty($new_id) && $id = $new_id;
  534. /*--end*/
  535. $info = Db::name('GuestbookAttribute')->where([
  536. 'attr_id' => $id,
  537. 'form_type' => 0,
  538. ])->find();
  539. if (empty($info)) {
  540. $this->error('数据不存在,请联系管理员!');
  541. exit;
  542. }
  543. $assign_data['field'] = $info;
  544. // 所在栏目
  545. $select_html = Db::name('arctype')->where('id', $info['typeid'])->getField('typename');
  546. $select_html = !empty($select_html) ? $select_html : '该栏目不存在';
  547. $assign_data['select_html'] = $select_html;
  548. $assign_data['attrInputTypeArr'] = $this->attrInputTypeArr; // 表单类型
  549. /*区域字段处理*/
  550. // 初始化参数
  551. $assign_data['region'] = [
  552. 'parent_id' => '-1',
  553. 'region_id' => '-1',
  554. 'region_names' => '',
  555. 'region_ids' => '',
  556. ];
  557. // 定义全国参数
  558. $China[] = [
  559. 'id' => 0,
  560. 'name' => '全国',
  561. ];
  562. // 查询省份信息并且拼装上$China数组
  563. $Province = get_province_list();
  564. $assign_data['Province'] = array_merge($China, $Province);
  565. // 区域选择时,指定不出现下级地区列表
  566. $assign_data['parent_array'] = "[]";
  567. // 如果是区域类型则执行
  568. if (9 == $info['attr_input_type']) {
  569. // 反序列化默认值参数
  570. $dfvalue = unserialize($info['attr_values']);
  571. if (0 == $dfvalue['region_id']) {
  572. $parent_id = $dfvalue['region_id'];
  573. } else {
  574. // 查询当前选中的区域父级ID
  575. $parent_id = Db::name('region')->where("id", $dfvalue['region_id'])->getField('parent_id');
  576. if (0 == $parent_id) {
  577. $parent_id = $dfvalue['region_id'];
  578. }
  579. }
  580. // 查询市\区\县信息
  581. $assign_data['City'] = Db::name('region')->where("parent_id", $parent_id)->select();
  582. // 加载数据到模板
  583. $assign_data['region'] = [
  584. 'parent_id' => $parent_id,
  585. 'region_id' => $dfvalue['region_id'],
  586. 'region_names' => $dfvalue['region_names'],
  587. 'region_ids' => $dfvalue['region_ids'],
  588. ];
  589. // 删除默认值,防止切换其他类型时使用到
  590. unset($info['attr_values']);
  591. // 区域选择时,指定不出现下级地区列表
  592. $assign_data['parent_array'] = convert_js_array(config('global.field_region_all_type'));
  593. }
  594. /*区域字段处理结束*/
  595. $this->assign($assign_data);
  596. return $this->fetch();
  597. }
  598. /**
  599. * 删除留言表单
  600. */
  601. public function attribute_del()
  602. {
  603. if (is_language() && empty($this->globalConfig['language_split'])) {
  604. $this->language_access(); // 多语言功能操作权限
  605. }
  606. $thorough = input('thorough/d');
  607. $id_arr = input('del_id/a');
  608. $id_arr = eyIntval($id_arr);
  609. if (!empty($id_arr)) {
  610. //多语言
  611. $attr_name_arr = [];
  612. foreach ($id_arr as $key => $val) {
  613. $attr_name_arr[] = 'attr_' . $val;
  614. }
  615. if (is_language() && empty($this->globalConfig['language_split'])) {
  616. $new_id_arr = Db::name('language_attr')->where([
  617. 'attr_name' => ['IN', $attr_name_arr],
  618. 'attr_group' => 'guestbook_attribute',
  619. ])->column('attr_value');
  620. !empty($new_id_arr) && $id_arr = $new_id_arr;
  621. }
  622. if (1 == $thorough){//彻底删除
  623. $r = Db::name('GuestbookAttribute')->where([
  624. 'attr_id' => ['IN', $id_arr],
  625. 'form_type' => 0,
  626. ])->delete();
  627. }else{
  628. $r = Db::name('GuestbookAttribute')->where([
  629. 'attr_id' => ['IN', $id_arr],
  630. 'form_type' => 0,
  631. ])->update([
  632. 'is_del' => 1,
  633. 'update_time' => getTime(),
  634. ]);
  635. }
  636. if($r !== false){
  637. // 删除多语言留言属性关联绑定
  638. if (1 == $thorough){//彻底删除
  639. if (!empty($attr_name_arr)) {
  640. if (get_admin_lang() == get_main_lang()) {
  641. Db::name('language_attribute')->where([
  642. 'attr_name' => ['IN', $attr_name_arr],
  643. 'attr_group' => 'guestbook_attribute',
  644. ])->delete();
  645. }
  646. if (empty($this->globalConfig['language_split'])) {
  647. Db::name('language_attr')->where([
  648. 'attr_name' => ['IN', $attr_name_arr],
  649. 'attr_group' => 'guestbook_attribute',
  650. ])->delete();
  651. } else {
  652. Db::name('language_attr')->where([
  653. 'attr_value' => ['IN', $id_arr],
  654. 'attr_group' => 'guestbook_attribute',
  655. ])->delete();
  656. }
  657. }
  658. }
  659. /*--end*/
  660. adminLog('删除留言表单-id:'.implode(',', $id_arr));
  661. $this->success('删除成功');
  662. }
  663. }
  664. $this->error('删除失败');
  665. }
  666. /**
  667. * 查看详情
  668. */
  669. public function details()
  670. {
  671. $aid = input('param.aid/d');
  672. $form_type = input('param.form_type/d', 0);
  673. // 标记为已读和IP地区
  674. if (1 == $form_type) {
  675. $row = Db::name('guestbook')->field('a.*, b.form_name')
  676. ->alias('a')
  677. ->join('form b','a.typeid = b.form_id','left')
  678. ->where(['a.aid'=>$aid, 'a.form_type'=>$form_type])
  679. ->find();
  680. } else {
  681. $row = Db::name('guestbook')->field('a.*, b.typename as form_name')
  682. ->alias('a')
  683. ->join('arctype b','a.typeid = b.id','left')
  684. ->where(['a.aid'=>$aid, 'a.form_type'=>$form_type])
  685. ->find();
  686. }
  687. $city = "";
  688. $city_arr = getCityLocation($row['ip']);
  689. if (!empty($city_arr)) {
  690. !empty($city_arr['location']) && $city .= $city_arr['location'];
  691. }
  692. $row['city'] = $city;
  693. // 标记为已读
  694. if (empty($row['is_read'])) {
  695. $row['is_read'] = 1;
  696. $row['update_time'] = getTime();
  697. Db::name('guestbook')->where(['aid'=>$aid])->update([
  698. 'is_read' => $row['is_read'],
  699. 'update_time' => $row['update_time'],
  700. ]);
  701. }
  702. $this->assign('row', $row);
  703. // 留言属性
  704. $attr_list = Db::name('guestbook_attribute')->where(['typeid'=>$row['typeid'],'form_type'=>$form_type])->order('attr_id asc')->select();
  705. $attr_values = Db::name('guestbook_attr')->field('attr_id,attr_value')->where(['aid'=>$aid,'form_type'=>$form_type])->getAllWithIndex('attr_id');
  706. foreach ($attr_list as $key => $val) {
  707. $val['attr_value'] = empty($attr_values[$val['attr_id']]) ? '' : $attr_values[$val['attr_id']]['attr_value'];
  708. $attr_list[$key] = $val;
  709. }
  710. foreach ($attr_list as $key => &$val) {
  711. if ($val['attr_input_type'] == 9) {
  712. $val['attr_value'] = Db::name('region')->where('id','in',$val['attr_value'])->column('name');
  713. $val['attr_value'] = implode('',$val['attr_value']);
  714. } else if ($val['attr_input_type'] == 4) {
  715. $val['attr_value'] = filter_line_return($val['attr_value'], '、');
  716. } else if(10 == $val['attr_input_type']){
  717. $val['attr_value'] = date('Y-m-d H:i:s',$val['attr_value']);
  718. } else if(11 == $val['attr_input_type']) {
  719. $attrValueArr = !empty($val['attr_value']) ? explode(',', $val['attr_value']) : [];
  720. $val['attr_value'] = '';
  721. foreach ($attrValueArr as $value) {
  722. if (preg_match('/(\.('.tpCache('global.image_type').'))$/i', $value)) {
  723. if (!stristr($value, '|')) {
  724. $value = handle_subdir_pic($value);
  725. $val['attr_value'] .= "<a class='guest-pic' href='{$value}' target='_blank'><img src='{$value}' width='60' height='60' style='float: unset; cursor: pointer;'/></a>";
  726. }
  727. } elseif (preg_match('/(\.('.tpCache('global.file_type').'))$/i', $value)) {
  728. if (!stristr($value, '|')) {
  729. $value = handle_subdir_pic($value);
  730. $val['attr_value'] .= "<a href='{$value}' download='".time()."'><img src=\"".ROOT_DIR."/public/static/common/images/file.png\" alt=\"\" style=\"width: 16px; height: 16px;\">文件下载</a>";
  731. }
  732. }
  733. }
  734. } else {
  735. if (preg_match('/(\.(jpg|gif|png|bmp|jpeg|ico|webp))$/i', $val['attr_value'])) {
  736. if (!stristr($val['attr_value'], '|')) {
  737. $val['attr_value'] = handle_subdir_pic($val['attr_value']);
  738. $val['attr_value'] = "<a href='{$val['attr_value']}' target='_blank'><img src='{$val['attr_value']}' width='60' height='60' style='float: unset;cursor: pointer;' /></a>";
  739. }
  740. }elseif (preg_match('/(\.('.tpCache('global.file_type').'))$/i', $val['attr_value'])){
  741. if (!stristr($val['attr_value'], '|')) {
  742. $val['attr_value'] = handle_subdir_pic($val['attr_value']);
  743. $val['attr_value'] = "<a href='{$val['attr_value']}' download='".time()."'><img src=\"".ROOT_DIR."/public/static/common/images/file.png\" alt=\"\" style=\"width: 16px;height: 16px;\">点击下载</a>";
  744. }
  745. }
  746. }
  747. }
  748. $this->assign('attr_list', $attr_list);
  749. // 如果安装手机端后台管理插件并且在手机端访问时执行
  750. $isMobile = input('param.isMobile/d', 0);
  751. if (is_dir('./weapp/Mbackend/') && !empty($isMobile)) {
  752. return $this->display('form/details');
  753. } else {
  754. return $this->fetch();
  755. }
  756. }
  757. /**
  758. * excel导出
  759. */
  760. public function ajax_excel_export()
  761. {
  762. $id_arr = input('aid/s');
  763. if (!empty($id_arr)) {
  764. $id_arr = explode(',', $id_arr);
  765. $id_arr = eyIntval($id_arr);
  766. }
  767. $typeid = input('typeid/d');
  768. $start_time = input('start_time/s');
  769. $end_time = input('end_time/s');
  770. $form_type = input('param.form_type/s', 0);
  771. $strTable = '<table width="500" border="1">';
  772. $where = [];
  773. if (!empty($typeid)) $where['typeid'] = $typeid;
  774. if ('all' != $form_type) $where['form_type'] = intval($form_type);
  775. $where['lang'] = $this->admin_lang;
  776. $order = 'add_time asc';
  777. //没有指定ID就导出全部
  778. if (!empty($id_arr)) {
  779. $where['aid'] = ['IN', $id_arr];
  780. }
  781. //根据日期导出
  782. if (!empty($start_time) && !empty($end_time)) {
  783. $start_time = strtotime($start_time);
  784. $end_time = strtotime("+1 day", strtotime($end_time)) - 1;
  785. $where['add_time'] = ['between', [$start_time, $end_time]];
  786. } elseif (!empty($start_time) && empty($end_time)) {
  787. $start_time = strtotime($start_time);
  788. $where['add_time'] = ['>=', $start_time];
  789. } elseif (empty($start_time) && !empty($end_time)) {
  790. $end_time = strtotime("+1 day", strtotime($end_time)) - 1;
  791. $where['add_time'] = ['<=', $end_time];
  792. }
  793. $row = Db::name('guestbook')->where($where)->order($order)->select();
  794. $map = [];
  795. if (!empty($typeid)) $map['typeid'] = $typeid;
  796. if ('all' != $form_type) $map['form_type'] = intval($form_type);
  797. $map['lang'] = $this->admin_lang;
  798. $map['is_del'] = 0;
  799. $title = Db::name('guestbook_attribute')->where($map)->order('sort_order asc, attr_id asc')->select();
  800. if ($row && $title) {
  801. $strTable .= '<tr>';
  802. $strTable .= '<td style="text-align:center;font-size:12px;" width="*">ID</td>';
  803. foreach ($title as &$key) {
  804. $strTable .= '<td style="text-align:center;font-size:12px;" width="*">' . $key['attr_name'] . '</td>';
  805. }
  806. $strTable .= '<td style="text-align:center;font-size:12px;" width="*">新增时间</td>';
  807. $strTable .= '<td style="text-align:center;font-size:12px;" width="*">更新时间</td>';
  808. $strTable .= '</tr>';
  809. foreach ($row as &$val) {
  810. $attr_value = Db::name('guestbook_attr')
  811. ->alias('a')
  812. ->field('a.*,b.attr_input_type')
  813. ->where(['a.aid' => $val['aid'], 'a.lang' => $this->admin_lang])
  814. ->join('guestbook_attribute b','a.attr_id = b.attr_id')
  815. ->getAllWithIndex('attr_id');
  816. foreach ($attr_value as $k => $v){
  817. if ($v['attr_input_type'] == 9){
  818. $v['attr_value'] = Db::name('region')->where('id','in',$v['attr_value'])->column('name');
  819. $attr_value[$k]['attr_value'] = implode('',$v['attr_value']);
  820. }else if(10 == $v['attr_input_type']){
  821. $attr_value[$k]['attr_value'] = date('Y-m-d H:i:s',$v['attr_value']);
  822. }else if(in_array($v['attr_input_type'],[5,8])){ //单张图、附件
  823. if (!stristr($val['attr_value'], '|')){
  824. $attr_value[$k]['attr_value'] = handle_subdir_pic($v['attr_value'],'img',true);
  825. }
  826. }else if(11 == $v['attr_input_type']){ //多张图
  827. $attr_value_arr = explode(",",$v['attr_value']);
  828. foreach ($attr_value_arr as $attr_value_k => $attr_value_v){
  829. $attr_value_arr[$attr_value_k] = handle_subdir_pic($attr_value_v,'img',true);
  830. }
  831. $attr_value[$k]['attr_value'] = implode(PHP_EOL,$attr_value_arr);
  832. }
  833. }
  834. $strTable .= '<tr>';
  835. $strTable .= '<td style="text-align:center;font-size:12px;">' . $val['aid'] . '</td>';
  836. foreach ($title as &$key) {
  837. $strTable .= '<td style="text-align:center;font-size:12px;" style=\'vnd.ms-excel.numberformat:@\' width="*">' . $attr_value[$key['attr_id']]['attr_value'] . '</td>';
  838. }
  839. $strTable .= '<td style="text-align:left;font-size:12px;">' . date('Y-m-d H:i:s', $val['add_time']) . '</td>';
  840. $strTable .= '<td style="text-align:left;font-size:12px;">' . date('Y-m-d H:i:s', $val['update_time']) . '</td>';
  841. $strTable .= '</tr>';
  842. }
  843. }
  844. $strTable .= '</table>';
  845. if ('all' === $form_type) {
  846. downloadExcel($strTable, 'allwords');
  847. } else if (1 == $form_type) {
  848. downloadExcel($strTable, 'form');
  849. } else {
  850. downloadExcel($strTable, 'guestbook');
  851. }
  852. exit();
  853. }
  854. /**
  855. * 设置/取消星标
  856. * @return [type] [description]
  857. */
  858. public function ajax_set_star()
  859. {
  860. $aid = input('param.aid/d');
  861. $is_star = input('param.is_star/d');
  862. if (IS_AJAX && !empty($aid)) {
  863. $r = Db::name('guestbook')->where(['aid'=>$aid])->update([
  864. 'is_star' => $is_star,
  865. ]);
  866. if ($r !== false) {
  867. if ($is_star == 1) {
  868. $this->success('星标成功');
  869. } else {
  870. $this->success('取消成功');
  871. }
  872. }
  873. }
  874. $this->error('操作失败');
  875. }
  876. }