Role.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 海豚PHP框架 [ DolphinPHP ]
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2016~2019 广东卓锐软件有限公司 [ http://www.zrthink.com ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://dolphinphp.com
  8. // +----------------------------------------------------------------------
  9. namespace app\user\admin;
  10. use app\admin\controller\Admin;
  11. use app\common\builder\ZBuilder;
  12. use app\user\model\Role as RoleModel;
  13. use app\admin\model\Menu as MenuModel;
  14. use util\Tree;
  15. use think\Db;
  16. /**
  17. * 角色控制器
  18. * @package app\admin\controller
  19. */
  20. class Role extends Admin
  21. {
  22. /**
  23. * 角色列表页
  24. * @author 蔡伟明 <314013107@qq.com>
  25. * @return mixed
  26. * @throws \think\Exception
  27. * @throws \think\exception\DbException
  28. */
  29. public function index()
  30. {
  31. // 获取查询条件
  32. $map = $this->getMap();
  33. // 非超级管理员检查可管理角色
  34. if (session('user_auth.role') != 1) {
  35. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  36. $map[] = ['id', 'in', $role_list];
  37. }
  38. // 数据列表
  39. $data_list = RoleModel::where($map)->order('pid,id')->paginate();
  40. // 角色列表
  41. $list_role = RoleModel::column('id,name');
  42. $list_role[0] = '顶级角色';
  43. // 使用ZBuilder快速创建数据表格
  44. return ZBuilder::make('table')
  45. ->setPageTitle('角色管理') // 页面标题
  46. ->setTableName('admin_role') // 设置表名
  47. ->setSearch(['name' => '角色名称', 'id' => 'ID']) // 设置搜索参数
  48. ->addColumns([ // 批量添加列
  49. ['id', 'ID'],
  50. ['name', '角色名称'],
  51. ['pid', '上级角色', $list_role],
  52. ['description', '描述'],
  53. ['default_module', '默认模块', 'callback', function($value, $list_module){
  54. if ($value == '') {
  55. return '未设置';
  56. } else {
  57. return isset($list_module[$value]) ? $list_module[$value] : '模块不存在';
  58. }
  59. }, MenuModel::where('pid', 0)->column('id,title')],
  60. ['create_time', '创建时间', 'datetime'],
  61. ['access', '是否可登录后台', 'switch'],
  62. ['status', '状态', 'switch'],
  63. ['right_button', '操作', 'btn']
  64. ])
  65. ->addTopButtons('add,enable,disable,delete') // 批量添加顶部按钮
  66. ->addRightButtons('edit,delete') // 批量添加右侧按钮
  67. ->replaceRightButton(['id' => 1], '<button class="btn btn-danger btn-xs" type="button" disabled>不可操作</button>') // 修改id为1的按钮
  68. ->setRowList($data_list) // 设置表格数据
  69. ->fetch(); // 渲染模板
  70. }
  71. /**
  72. * 新增
  73. * @author 蔡伟明 <314013107@qq.com>
  74. * @return mixed
  75. */
  76. public function add()
  77. {
  78. // 保存数据
  79. if ($this->request->isPost()) {
  80. $data = $this->request->post();
  81. if (!isset($data['menu_auth'])) {
  82. $data['menu_auth'] = [];
  83. } else {
  84. $data['menu_auth'] = explode(',', $data['menu_auth']);
  85. }
  86. // 验证
  87. $result = $this->validate($data, 'Role');
  88. // 验证失败 输出错误信息
  89. if(true !== $result) $this->error($result);
  90. // 非超级管理员检查可添加角色
  91. if (session('user_auth.role') != 1) {
  92. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  93. if (!in_array($data['pid'], $role_list)) {
  94. $this->error('所属角色设置错误,没有权限添加该角色');
  95. }
  96. // 非超级管理员检查可添加的节点权限
  97. $menu_auth = RoleModel::where('id', session('user_auth.role'))->value('menu_auth');
  98. $menu_auth = json_decode($menu_auth, true);
  99. $menu_auth = array_intersect($menu_auth, $data['menu_auth']);
  100. $data['menu_auth'] = $menu_auth;
  101. }
  102. // 添加数据
  103. if ($role = RoleModel::create($data)) {
  104. // 记录行为
  105. action_log('role_add', 'admin_role', $role['id'], UID, $data['name']);
  106. $this->success('新增成功', url('index'));
  107. } else {
  108. $this->error('新增失败');
  109. }
  110. }
  111. // 菜单列表
  112. $menus = cache('access_menus');
  113. if (!$menus) {
  114. $modules = Db::name('admin_module')->where('status', 1)->column('name,title');
  115. $map = [];
  116. // 非超级管理员角色,只能分配当前角色所拥有的权限
  117. if (session('user_auth.role') != 1) {
  118. $menu_auth = RoleModel::where('id', session('user_auth.role'))->value('menu_auth');
  119. $menu_auth = json_decode($menu_auth, true);
  120. $map[] = ['id', 'in', $menu_auth];
  121. }
  122. // 当前用户能分配的所有菜单
  123. $menus = MenuModel::where('module', 'in', array_keys($modules))
  124. ->where($map)
  125. ->order('module,sort,id')
  126. ->column('id,pid,sort,url_value,title,icon,module');
  127. // 按模块分组菜单
  128. $moduleMenus = [];
  129. foreach ($menus as $key => $menu) {
  130. if (!isset($moduleMenus[$menu['module']])) {
  131. $moduleMenus[$menu['module']] = [
  132. 'title' => isset($modules[$menu['module']]) ? $modules[$menu['module']] : '未知',
  133. 'menus' => [$menu]
  134. ];
  135. } else {
  136. $moduleMenus[$menu['module']]['menus'][] = $menu;
  137. }
  138. }
  139. // 层级化每个模块的菜单
  140. foreach ($moduleMenus as $key => $module) {
  141. $menu = Tree::toLayer($module['menus']);
  142. $moduleMenus[$key]['menus'] = $this->buildJsTree($menu);
  143. }
  144. $menus = $moduleMenus;
  145. // 非开发模式,缓存菜单
  146. if (config('develop_mode') == 0) {
  147. cache('access_menus', $menus);
  148. }
  149. }
  150. if (session('user_auth.role') != 1) {
  151. $role_list = RoleModel::getTree(null, false, session('user_auth.role'));
  152. } else {
  153. $role_list = RoleModel::getTree();
  154. }
  155. $this->assign('page_title', '新增');
  156. $this->assign('role_list', $role_list);
  157. $this->assign('module_list', MenuModel::where('pid', 0)->column('id,title'));
  158. $this->assign('menus', $menus);
  159. $this->assign('curr_tab', current(array_keys($menus)));
  160. return $this->fetch();
  161. }
  162. /**
  163. * 编辑
  164. * @param null $id 角色id
  165. * @author 蔡伟明 <314013107@qq.com>
  166. * @return mixed
  167. */
  168. public function edit($id = null)
  169. {
  170. if ($id === null) $this->error('缺少参数');
  171. if ($id == 1) $this->error('超级管理员不可修改');
  172. // 非超级管理员检查可编辑角色
  173. if (session('user_auth.role') != 1) {
  174. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  175. if (!in_array($id, $role_list)) {
  176. $this->error('权限不足,当前没有编辑该角色的权限!');
  177. }
  178. }
  179. // 获取数据
  180. $info = RoleModel::get($id);
  181. // 保存数据
  182. if ($this->request->isPost()) {
  183. $data = $this->request->post();
  184. if (!isset($data['menu_auth'])) {
  185. $data['menu_auth'] = [];
  186. } else {
  187. $data['menu_auth'] = explode(',', $data['menu_auth']);
  188. }
  189. if ($data['pid'] == '') {
  190. $data['pid'] = $info['pid'];
  191. }
  192. // 验证
  193. $result = $this->validate($data, 'Role');
  194. // 验证失败 输出错误信息
  195. if(true !== $result) $this->error($result);
  196. // 非超级管理员检查可添加角色
  197. if (session('user_auth.role') != 1) {
  198. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  199. if ($data['pid'] != $info['pid'] && !in_array($data['pid'], $role_list)) {
  200. $this->error('所属角色设置错误,没有权限添加该角色');
  201. }
  202. }
  203. // 检查所属角色不能是自己当前角色及其子角色
  204. $role_list = RoleModel::getChildsId($data['id']);
  205. if ($data['id'] == $data['pid'] || in_array($data['pid'], $role_list)) {
  206. $this->error('所属角色设置错误,禁止设置为当前角色及其子角色。');
  207. }
  208. // 非超级管理员检查可添加的节点权限
  209. if (session('user_auth.role') != 1) {
  210. $menu_auth = RoleModel::where('id', session('user_auth.role'))->value('menu_auth');
  211. $menu_auth = json_decode($menu_auth, true);
  212. $menu_auth = array_intersect($menu_auth, $data['menu_auth']);
  213. $data['menu_auth'] = $menu_auth;
  214. }
  215. if (RoleModel::update($data)) {
  216. // 更新成功,循环处理子角色权限
  217. RoleModel::resetAuth($id, $data['menu_auth']);
  218. role_auth();
  219. // 记录行为
  220. action_log('role_edit', 'admin_role', $id, UID, $data['name']);
  221. $this->success('编辑成功', url('index'));
  222. } else {
  223. $this->error('编辑失败');
  224. }
  225. }
  226. if (session('user_auth.role') != 1) {
  227. $role_list = RoleModel::getTree($id, false, session('user_auth.role'));
  228. } else {
  229. $role_list = RoleModel::getTree($id, '顶级角色');
  230. }
  231. $modules = Db::name('admin_module')->where('status', 1)->column('name,title');
  232. $map = [];
  233. // 非超级管理员角色,只能分配当前角色所拥有的权限
  234. if (session('user_auth.role') != 1) {
  235. $menu_auth = RoleModel::where('id', session('user_auth.role'))->value('menu_auth');
  236. $menu_auth = json_decode($menu_auth, true);
  237. $map[] = ['id', 'in', $menu_auth];
  238. }
  239. // 当前用户能分配的所有菜单
  240. $menus = MenuModel::where('module', 'in', array_keys($modules))
  241. ->where($map)
  242. ->order('module,sort,id')
  243. ->column('id,pid,sort,url_value,title,icon,module');
  244. // 按模块分组菜单
  245. $moduleMenus = [];
  246. foreach ($menus as $key => $menu) {
  247. if (!isset($moduleMenus[$menu['module']])) {
  248. $moduleMenus[$menu['module']] = [
  249. 'title' => isset($modules[$menu['module']]) ? $modules[$menu['module']] : '未知',
  250. 'menus' => [$menu]
  251. ];
  252. } else {
  253. $moduleMenus[$menu['module']]['menus'][] = $menu;
  254. }
  255. }
  256. // 层级化每个模块的菜单
  257. foreach ($moduleMenus as $key => $module) {
  258. $menu = Tree::toLayer($module['menus']);
  259. $moduleMenus[$key]['menus'] = $this->buildJsTree($menu, $info);
  260. }
  261. $this->assign('page_title', '编辑');
  262. $this->assign('role_list', $role_list);
  263. $this->assign('module_list', MenuModel::where('pid', 0)->column('id,title'));
  264. $this->assign('menus', $moduleMenus);
  265. $this->assign('curr_tab', current(array_keys($moduleMenus)));
  266. $this->assign('info', $info);
  267. return $this->fetch('edit');
  268. }
  269. /**
  270. * 构建jstree代码
  271. * @param array $menus 菜单节点
  272. * @param array $user 用户信息
  273. * @author 蔡伟明 <314013107@qq.com>
  274. * @return string
  275. */
  276. private function buildJsTree($menus = [], $user = [])
  277. {
  278. $result = '';
  279. if (!empty($menus)) {
  280. $option = [
  281. 'opened' => true,
  282. 'selected' => false,
  283. 'icon' => '',
  284. ];
  285. foreach ($menus as $menu) {
  286. $option['icon'] = $menu['icon'];
  287. if (isset($user['menu_auth'])) {
  288. $option['selected'] = in_array($menu['id'], $user['menu_auth']) ? true : false;
  289. }
  290. if (isset($menu['child'])) {
  291. $result .= '<li id="'.$menu['id'].'" data-jstree=\''.json_encode($option).'\'>'.$menu['title'].($menu['url_value'] == '' ? '' : ' ('.$menu['url_value'].')').$this->buildJsTree($menu['child'], $user).'</li>';
  292. } else {
  293. $result .= '<li id="'.$menu['id'].'" data-jstree=\''.json_encode($option).'\'>'.$menu['title'].($menu['url_value'] == '' ? '' : ' ('.$menu['url_value'].')').'</li>';
  294. }
  295. }
  296. }
  297. return '<ul>'.$result.'</ul>';
  298. }
  299. /**
  300. * 删除角色
  301. * @param array $record 行为日志
  302. * @author 蔡伟明 <314013107@qq.com>
  303. * @throws \think\Exception
  304. * @throws \think\exception\PDOException
  305. */
  306. public function delete($record = [])
  307. {
  308. return $this->setStatus('delete');
  309. }
  310. /**
  311. * 启用角色
  312. * @param array $record 行为日志
  313. * @author 蔡伟明 <314013107@qq.com>
  314. * @throws \think\Exception
  315. * @throws \think\exception\PDOException
  316. */
  317. public function enable($record = [])
  318. {
  319. return $this->setStatus('enable');
  320. }
  321. /**
  322. * 禁用角色
  323. * @param array $record 行为日志
  324. * @author 蔡伟明 <314013107@qq.com>
  325. * @throws \think\Exception
  326. * @throws \think\exception\PDOException
  327. */
  328. public function disable($record = [])
  329. {
  330. return $this->setStatus('disable');
  331. }
  332. /**
  333. * 设置角色状态:删除、禁用、启用
  334. * @param string $type 类型:delete/enable/disable
  335. * @param array $record
  336. * @author 蔡伟明 <314013107@qq.com>
  337. * @throws \think\Exception
  338. * @throws \think\exception\PDOException
  339. */
  340. public function setStatus($type = '', $record = [])
  341. {
  342. $ids = $this->request->isPost() ? input('post.ids/a') : input('param.ids');
  343. $ids = (array)$ids;
  344. // 当前角色所能操作的子角色
  345. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  346. if (session('user_auth.role') != 1 && !$role_list) {
  347. $this->error('权限不足,没有可操作的角色');
  348. }
  349. foreach ($ids as $id) {
  350. if ($id == 1) {
  351. // 跳过默认角色
  352. continue;
  353. }
  354. // 非超级管理员检查可管理角色
  355. if (session('user_auth.role') != 1) {
  356. if (!in_array($id, $role_list)) {
  357. $this->error('权限不足,禁止操作角色ID:'.$id);
  358. }
  359. }
  360. switch ($type) {
  361. case 'enable':
  362. if (false === RoleModel::where('id', $id)->setField('status', 1)) {
  363. $this->error('启用失败,角色ID:'.$id);
  364. }
  365. break;
  366. case 'disable':
  367. if (false === RoleModel::where('id', $id)->setField('status', 0)) {
  368. $this->error('禁用失败,角色ID:'.$id);
  369. }
  370. break;
  371. case 'delete':
  372. $all_id = array_merge([$id], RoleModel::getChildsId($id));
  373. if (false === RoleModel::where('id', 'in', $all_id)->delete()) {
  374. $this->error('删除失败,角色ID:'.$id);
  375. }
  376. break;
  377. default:
  378. $this->error('非法操作');
  379. }
  380. action_log('role_'.$type, 'admin_role', $id, UID);
  381. }
  382. $this->success('操作成功');
  383. }
  384. /**
  385. * 快速编辑
  386. * @param array $record 行为日志
  387. * @author 蔡伟明 <314013107@qq.com>
  388. * @return mixed
  389. */
  390. public function quickEdit($record = [])
  391. {
  392. $id = input('post.pk', '');
  393. $field = input('post.name', '');
  394. $value = input('post.value', '');
  395. // 非超级管理员检查可操作的角色
  396. if (session('user_auth.role') != 1) {
  397. $role_list = RoleModel::getChildsId(session('user_auth.role'));
  398. if (!in_array($id, $role_list)) {
  399. $this->error('权限不足,没有可操作的角色');
  400. }
  401. }
  402. $config = RoleModel::where('id', $id)->value($field);
  403. $details = '字段(' . $field . '),原值(' . $config . '),新值:(' . $value . ')';
  404. return parent::quickEdit(['role_edit', 'admin_role', $id, UID, $details]);
  405. }
  406. }