Menu.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 海豚PHP框架 [ DolphinPHP ]
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2016~2017 河源市卓锐科技有限公司 [ http://www.zrthink.com ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://dolphinphp.com
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( http://www.apache.org/licenses/LICENSE-2.0 )
  10. // +----------------------------------------------------------------------
  11. namespace app\admin\controller;
  12. use app\common\builder\ZBuilder;
  13. use app\admin\model\Module as ModuleModel;
  14. use app\admin\model\Menu as MenuModel;
  15. use app\user\model\Role as RoleModel;
  16. use think\Cache;
  17. /**
  18. * 节点管理
  19. * @package app\admin\controller
  20. */
  21. class Menu extends Admin
  22. {
  23. /**
  24. * 节点首页
  25. * @param string $group 分组
  26. * @author 蔡伟明 <314013107@qq.com>
  27. * @return mixed
  28. */
  29. public function index($group = 'admin')
  30. {
  31. // 保存模块排序
  32. if ($this->request->isPost()) {
  33. $modules = $this->request->post('sort/a');
  34. if ($modules) {
  35. $data = [];
  36. foreach ($modules as $key => $module) {
  37. $data[] = [
  38. 'id' => $module,
  39. 'sort' => $key + 1
  40. ];
  41. }
  42. $MenuModel = new MenuModel();
  43. if (false !== $MenuModel->saveAll($data)) {
  44. $this->success('保存成功');
  45. } else {
  46. $this->error('保存失败');
  47. }
  48. }
  49. }
  50. cookie('__forward__', $_SERVER['REQUEST_URI']);
  51. // 配置分组信息
  52. $list_group = MenuModel::getGroup();
  53. foreach ($list_group as $key => $value) {
  54. $tab_list[$key]['title'] = $value;
  55. $tab_list[$key]['url'] = url('index', ['group' => $key]);
  56. }
  57. // 模块排序
  58. if ($group == 'module-sort') {
  59. $map['status'] = 1;
  60. $map['pid'] = 0;
  61. $modules = MenuModel::where($map)->order('sort,id')->column('icon,title', 'id');
  62. $this->assign('modules', $modules);
  63. } else {
  64. // 获取节点数据
  65. $data_list = MenuModel::getMenusByGroup($group);
  66. $max_level = $this->request->get('max', 0);
  67. $this->assign('menus', $this->getNestMenu($data_list, $max_level));
  68. }
  69. $this->assign('tab_nav', ['tab_list' => $tab_list, 'curr_tab' => $group]);
  70. $this->assign('page_title', '节点管理');
  71. return $this->fetch();
  72. }
  73. /**
  74. * 新增节点
  75. * @param string $module 所属模块
  76. * @param string $pid 所属节点id
  77. * @author 蔡伟明 <314013107@qq.com>
  78. * @return mixed
  79. */
  80. public function add($module = 'admin', $pid = '')
  81. {
  82. // 保存数据
  83. if ($this->request->isPost()) {
  84. $data = $this->request->post('', null, 'trim');
  85. // 验证
  86. $result = $this->validate($data, 'Menu');
  87. // 验证失败 输出错误信息
  88. if(true !== $result) $this->error($result);
  89. // 顶部节点url检查
  90. if ($data['pid'] == 0 && $data['url_value'] == '' && ($data['url_type'] == 'module_admin' || $data['url_type'] == 'module_home')) {
  91. $this->error('顶级节点的节点链接不能为空');
  92. }
  93. if ($menu = MenuModel::create($data)) {
  94. // 自动创建子节点
  95. if ($data['auto_create'] == 1 && !empty($data['child_node'])) {
  96. unset($data['icon']);
  97. unset($data['params']);
  98. $this->createChildNode($data, $menu['id']);
  99. }
  100. // 添加角色权限
  101. if (isset($data['role'])) {
  102. $this->setRoleMenu($menu['id'], $data['role']);
  103. }
  104. Cache::clear();
  105. // 记录行为
  106. $details = '所属模块('.$data['module'].'),所属节点ID('.$data['pid'].'),节点标题('.$data['title'].'),节点链接('.$data['url_value'].')';
  107. action_log('menu_add', 'admin_menu', $menu['id'], UID, $details);
  108. $this->success('新增成功', cookie('__forward__'));
  109. } else {
  110. $this->error('新增失败');
  111. }
  112. }
  113. // 使用ZBuilder快速创建表单
  114. return ZBuilder::make('form')
  115. ->setPageTitle('新增节点')
  116. ->addLinkage('module', '所属模块', '', ModuleModel::getModule(), $module, url('ajax/getModuleMenus'), 'pid')
  117. ->addFormItems([
  118. ['select', 'pid', '所属节点', '所属上级节点', MenuModel::getMenuTree(0, '', $module), $pid],
  119. ['text', 'title', '节点标题'],
  120. ['radio', 'url_type', '链接类型', '', ['module_admin' => '模块链接(后台)', 'module_home' => '模块链接(前台)', 'link' => '普通链接'], 'module_admin']
  121. ])
  122. ->addFormItem(
  123. 'text',
  124. 'url_value',
  125. '节点链接',
  126. "可留空,如果是模块链接,请填写<code>模块/控制器/操作</code>,如:<code>admin/menu/add</code>。如果是普通链接,则直接填写url地址,如:<code>http://www.dolphinphp.com</code>"
  127. )
  128. ->addText('params', '参数', '如:a=1&b=2')
  129. ->addSelect('role', '角色', '除超级管理员外,拥有该节点权限的角色', RoleModel::where('id', 'neq', 1)->column('id,name'), '', 'multiple')
  130. ->addRadio('auto_create', '自动添加子节点', '选择【是】则自动添加指定的子节点', ['否', '是'], 0)
  131. ->addCheckbox('child_node', '子节点', '仅上面选项为【是】时起作用', ['add' => '新增', 'edit' => '编辑', 'delete' => '删除', 'enable' => '启用', 'disable' => '禁用', 'quickedit' => '快速编辑'], 'add,edit,delete,enable,disable,quickedit')
  132. ->addRadio('url_target', '打开方式', '', ['_self' => '当前窗口', '_blank' => '新窗口'], '_self')
  133. ->addIcon('icon', '图标', '导航图标')
  134. ->addRadio('online_hide', '网站上线后隐藏', '关闭开发模式后,则隐藏该菜单节点', ['否', '是'], 0)
  135. ->addText('sort', '排序', '', 100)
  136. ->setTrigger('auto_create', '1', 'child_node', false)
  137. ->fetch();
  138. }
  139. /**
  140. * 编辑节点
  141. * @param int $id 节点ID
  142. * @author 蔡伟明 <314013107@qq.com>
  143. * @return mixed
  144. */
  145. public function edit($id = 0)
  146. {
  147. if ($id === 0) $this->error('缺少参数');
  148. // 保存数据
  149. if ($this->request->isPost()) {
  150. $data = $this->request->post('', null, 'trim');
  151. // 验证
  152. $result = $this->validate($data, 'Menu');
  153. // 验证失败 输出错误信息
  154. if(true !== $result) $this->error($result);
  155. // 顶部节点url检查
  156. if ($data['pid'] == 0 && $data['url_value'] == '' && ($data['url_type'] == 'module_admin' || $data['url_type'] == 'module_home')) {
  157. $this->error('顶级节点的节点链接不能为空');
  158. }
  159. // 设置角色权限
  160. $this->setRoleMenu($data['id'], isset($data['role']) ? $data['role'] : []);
  161. // 验证是否更改所属模块,如果是,则该节点的所有子孙节点的模块都要修改
  162. $map['id'] = $data['id'];
  163. $map['module'] = $data['module'];
  164. if (!MenuModel::where($map)->find()) {
  165. MenuModel::changeModule($data['id'], $data['module']);
  166. }
  167. if (MenuModel::update($data)) {
  168. Cache::clear();
  169. // 记录行为
  170. $details = '节点ID('.$id.')';
  171. action_log('menu_edit', 'admin_menu', $id, UID, $details);
  172. $this->success('编辑成功', cookie('__forward__'));
  173. } else {
  174. $this->error('编辑失败');
  175. }
  176. }
  177. // 获取数据
  178. $info = MenuModel::get($id);
  179. // 拥有该节点权限的角色
  180. $info['role'] = RoleModel::getRoleWithMenu($id);
  181. // 使用ZBuilder快速创建表单
  182. return ZBuilder::make('form')
  183. ->setPageTitle('编辑节点')
  184. ->addFormItem('hidden', 'id')
  185. ->addLinkage('module', '所属模块', '', ModuleModel::getModule(), '', url('ajax/getModuleMenus'), 'pid')
  186. ->addFormItem('select', 'pid', '所属节点', '所属上级节点', MenuModel::getMenuTree(0, '', $info['module']))
  187. ->addFormItem('text', 'title', '节点标题')
  188. ->addFormItem('radio', 'url_type', '链接类型', '', ['module_admin' => '模块链接(后台)', 'module_home' => '模块链接(前台)', 'link' => '普通链接'], 'module_admin')
  189. ->addFormItem(
  190. 'text',
  191. 'url_value',
  192. '节点链接',
  193. "可留空,如果是模块链接,请填写<code>模块/控制器/操作</code>,如:<code>admin/menu/add</code>。如果是普通链接,则直接填写url地址,如:<code>http://www.dolphinphp.com</code>"
  194. )
  195. ->addText('params', '参数', '如:a=1&b=2')
  196. ->addSelect('role', '角色', '除超级管理员外,拥有该节点权限的角色', RoleModel::where('id', 'neq', 1)->column('id,name'), '', 'multiple')
  197. ->addRadio('url_target', '打开方式', '', ['_self' => '当前窗口', '_blank' => '新窗口'], '_self')
  198. ->addIcon('icon', '图标', '导航图标')
  199. ->addRadio('online_hide', '网站上线后隐藏', '关闭开发模式后,则隐藏该菜单节点', ['否', '是'])
  200. ->addText('sort', '排序', '', 100)
  201. ->setFormData($info)
  202. ->fetch();
  203. }
  204. /**
  205. * 设置角色权限
  206. * @param string $role_id 角色id
  207. * @param array $roles 角色id
  208. * @author 蔡伟明 <314013107@qq.com>
  209. */
  210. private function setRoleMenu($role_id = '', $roles = [])
  211. {
  212. $RoleModel = new RoleModel();
  213. // 该节点的所有子节点,包括本身节点
  214. $menu_child = MenuModel::getChildsId($role_id);
  215. $menu_child[] = (int)$role_id;
  216. // 该节点的所有上下级节点
  217. $menu_all = MenuModel::getLinkIds($role_id);
  218. $menu_all = array_map('strval', $menu_all);
  219. if (!empty($roles)) {
  220. // 拥有该节点的所有角色id及节点权限
  221. $role_menu_auth = RoleModel::getRoleWithMenu($role_id, true);
  222. // 已有该节点权限的角色id
  223. $role_exists = array_keys($role_menu_auth);
  224. // 新节点权限的角色
  225. $role_new = $roles;
  226. // 原有权限角色差集
  227. $role_diff = array_diff($role_exists, $role_new);
  228. // 新权限角色差集
  229. $role_diff_new = array_diff($role_new, $role_exists);
  230. // 新节点角色权限
  231. $role_new_auth = RoleModel::getAuthWithRole($roles);
  232. // 删除原先角色的该节点权限
  233. if ($role_diff) {
  234. $role_del_auth = [];
  235. foreach ($role_diff as $role) {
  236. $auth = json_decode($role_menu_auth[$role], true);
  237. $auth_new = array_diff($auth, $menu_child);
  238. $role_del_auth[] = [
  239. 'id' => $role,
  240. 'menu_auth' => array_values($auth_new)
  241. ];
  242. }
  243. if ($role_del_auth) {
  244. $RoleModel->saveAll($role_del_auth);
  245. }
  246. }
  247. // 新增权限角色
  248. if ($role_diff_new) {
  249. $role_update_auth = [];
  250. foreach ($role_new_auth as $role => $auth) {
  251. $auth = json_decode($auth, true);
  252. if (in_array($role, $role_diff_new)) {
  253. $auth = array_unique(array_merge($auth, $menu_all));
  254. }
  255. $role_update_auth[] = [
  256. 'id' => $role,
  257. 'menu_auth' => array_values($auth)
  258. ];
  259. }
  260. if ($role_update_auth) {
  261. $RoleModel->saveAll($role_update_auth);
  262. }
  263. }
  264. } else {
  265. $role_menu_auth = RoleModel::getRoleWithMenu($role_id, true);
  266. $role_del_auth = [];
  267. foreach ($role_menu_auth as $role => $auth) {
  268. $auth = json_decode($auth, true);
  269. $auth_new = array_diff($auth, $menu_child);
  270. $role_del_auth[] = [
  271. 'id' => $role,
  272. 'menu_auth' => array_values($auth_new)
  273. ];
  274. }
  275. if ($role_del_auth) {
  276. $RoleModel->saveAll($role_del_auth);
  277. }
  278. }
  279. }
  280. /**
  281. * 删除节点
  282. * @param array $record 行为日志内容
  283. * @author 蔡伟明 <314013107@qq.com>
  284. * @return mixed
  285. */
  286. public function delete($record = [])
  287. {
  288. $id = $this->request->param('id');
  289. $menu = MenuModel::where('id', $id)->find();
  290. if ($menu['system_menu'] == '1') $this->error('系统节点,禁止删除');
  291. // 获取该节点的所有后辈节点id
  292. $menu_childs = MenuModel::getChildsId($id);
  293. // 要删除的所有节点id
  294. $all_ids = array_merge([(int)$id], $menu_childs);
  295. // 删除节点
  296. if (MenuModel::destroy($all_ids)) {
  297. Cache::clear();
  298. // 记录行为
  299. $details = '节点ID('.$id.'),节点标题('.$menu['title'].'),节点链接('.$menu['url_value'].')';
  300. action_log('menu_delete', 'admin_menu', $id, UID, $details);
  301. $this->success('删除成功');
  302. } else {
  303. $this->error('删除失败');
  304. }
  305. }
  306. /**
  307. * 保存节点排序
  308. * @author 蔡伟明 <314013107@qq.com>
  309. * @return mixed
  310. */
  311. public function save()
  312. {
  313. if ($this->request->isPost()) {
  314. $data = $this->request->post();
  315. if (!empty($data)) {
  316. $menus = $this->parseMenu($data['menus']);
  317. foreach ($menus as $menu) {
  318. if ($menu['pid'] == 0) {
  319. continue;
  320. }
  321. MenuModel::update($menu);
  322. }
  323. Cache::clear();
  324. $this->success('保存成功');
  325. } else {
  326. $this->error('没有需要保存的节点');
  327. }
  328. }
  329. $this->error('非法请求');
  330. }
  331. /**
  332. * 添加子节点
  333. * @param array $data 节点数据
  334. * @param string $pid 父节点id
  335. * @author 蔡伟明 <314013107@qq.com>
  336. */
  337. private function createChildNode($data = [], $pid = '')
  338. {
  339. $url_value = substr($data['url_value'], 0, strrpos($data['url_value'], '/')).'/';
  340. $child_node = [];
  341. $data['pid'] = $pid;
  342. foreach ($data['child_node'] as $item) {
  343. switch ($item) {
  344. case 'add':
  345. $data['title'] = '新增';
  346. break;
  347. case 'edit':
  348. $data['title'] = '编辑';
  349. break;
  350. case 'delete':
  351. $data['title'] = '删除';
  352. break;
  353. case 'enable':
  354. $data['title'] = '启用';
  355. break;
  356. case 'disable':
  357. $data['title'] = '禁用';
  358. break;
  359. case 'quickedit':
  360. $data['title'] = '快速编辑';
  361. break;
  362. }
  363. $data['url_value'] = $url_value.$item;
  364. $data['create_time'] = $this->request->time();
  365. $data['update_time'] = $this->request->time();
  366. $child_node[] = $data;
  367. }
  368. if ($child_node) {
  369. $MenuModel = new MenuModel();
  370. $MenuModel->insertAll($child_node);
  371. }
  372. }
  373. /**
  374. * 递归解析节点
  375. * @param array $menus 节点数据
  376. * @param int $pid 上级节点id
  377. * @author 蔡伟明 <314013107@qq.com>
  378. * @return array 解析成可以写入数据库的格式
  379. */
  380. private function parseMenu($menus = [], $pid = 0)
  381. {
  382. $sort = 1;
  383. $result = [];
  384. foreach ($menus as $menu) {
  385. $result[] = [
  386. 'id' => (int)$menu['id'],
  387. 'pid' => (int)$pid,
  388. 'sort' => $sort,
  389. ];
  390. if (isset($menu['children'])) {
  391. $result = array_merge($result, $this->parseMenu($menu['children'], $menu['id']));
  392. }
  393. $sort ++;
  394. }
  395. return $result;
  396. }
  397. /**
  398. * 获取嵌套式节点
  399. * @param array $lists 原始节点数组
  400. * @param int $pid 父级id
  401. * @param int $max_level 最多返回多少层,0为不限制
  402. * @param int $curr_level 当前层数
  403. * @author 蔡伟明 <314013107@qq.com>
  404. * @return string
  405. */
  406. private function getNestMenu($lists = [], $max_level = 0, $pid = 0, $curr_level = 1)
  407. {
  408. $result = '';
  409. foreach ($lists as $key => $value) {
  410. if ($value['pid'] == $pid) {
  411. $disable = $value['status'] == 0 ? 'dd-disable' : '';
  412. // 组合节点
  413. $result .= '<li class="dd-item dd3-item '.$disable.'" data-id="'.$value['id'].'">';
  414. $result .= '<div class="dd-handle dd3-handle">拖拽</div><div class="dd3-content"><i class="'.$value['icon'].'"></i> '.$value['title'];
  415. if ($value['url_value'] != '') {
  416. $result .= '<span class="link"><i class="fa fa-link"></i> '.$value['url_value'].'</span>';
  417. }
  418. $result .= '<div class="action">';
  419. $result .= '<a href="'.url('add', ['module' => $value['module'], 'pid' => $value['id']]).'" data-toggle="tooltip" data-original-title="新增子节点"><i class="list-icon fa fa-plus fa-fw"></i></a><a href="'.url('edit', ['id' => $value['id']]).'" data-toggle="tooltip" data-original-title="编辑"><i class="list-icon fa fa-pencil fa-fw"></i></a>';
  420. if ($value['status'] == 0) {
  421. // 启用
  422. $result .= '<a href="javascript:void(0);" data-ids="'.$value['id'].'" class="enable" data-toggle="tooltip" data-original-title="启用"><i class="list-icon fa fa-check-circle-o fa-fw"></i></a>';
  423. } else {
  424. // 禁用
  425. $result .= '<a href="javascript:void(0);" data-ids="'.$value['id'].'" class="disable" data-toggle="tooltip" data-original-title="禁用"><i class="list-icon fa fa-ban fa-fw"></i></a>';
  426. }
  427. $result .= '<a href="'.url('delete', ['id' => $value['id'], 'table' => 'admin_menu']).'" data-toggle="tooltip" data-original-title="删除" class="ajax-get confirm"><i class="list-icon fa fa-times fa-fw"></i></a></div>';
  428. $result .= '</div>';
  429. if ($max_level == 0 || $curr_level != $max_level) {
  430. unset($lists[$key]);
  431. // 下级节点
  432. $children = $this->getNestMenu($lists, $max_level, $value['id'], $curr_level + 1);
  433. if ($children != '') {
  434. $result .= '<ol class="dd-list">'.$children.'</ol>';
  435. }
  436. }
  437. $result .= '</li>';
  438. }
  439. }
  440. return $result;
  441. }
  442. /**
  443. * 启用节点
  444. * @param array $record 行为日志
  445. * @author 蔡伟明 <314013107@qq.com>
  446. * @return void
  447. */
  448. public function enable($record = [])
  449. {
  450. $id = input('param.ids');
  451. $menu = MenuModel::where('id', $id)->find();
  452. $details = '节点ID('.$id.'),节点标题('.$menu['title'].'),节点链接('.$menu['url_value'].')';
  453. $this->setStatus('enable', ['menu_enable', 'admin_menu', $id, UID, $details]);
  454. }
  455. /**
  456. * 禁用节点
  457. * @param array $record 行为日志
  458. * @author 蔡伟明 <314013107@qq.com>
  459. * @return void
  460. */
  461. public function disable($record = [])
  462. {
  463. $id = input('param.ids');
  464. $menu = MenuModel::where('id', $id)->find();
  465. $details = '节点ID('.$id.'),节点标题('.$menu['title'].'),节点链接('.$menu['url_value'].')';
  466. $this->setStatus('disable', ['menu_disable', 'admin_menu', $id, UID, $details]);
  467. }
  468. /**
  469. * 设置状态
  470. * @param string $type 类型
  471. * @param array $record 行为日志
  472. * @author 小乌 <82950492@qq.com>
  473. * @return void
  474. */
  475. public function setStatus($type = '', $record = [])
  476. {
  477. $id = input('param.ids');
  478. $status = $type == 'enable' ? 1 : 0;
  479. if (false !== MenuModel::where('id', $id)->setField('status', $status)) {
  480. Cache::clear();
  481. // 记录行为日志
  482. if (!empty($record)) {
  483. call_user_func_array('action_log', $record);
  484. }
  485. $this->success('操作成功');
  486. } else {
  487. $this->error('操作失败');
  488. }
  489. }
  490. }