RbacController.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. <?php
  2. // +—————————————————————————————————————————————————————————————————————
  3. // | Created by Yunbao
  4. // +—————————————————————————————————————————————————————————————————————
  5. // | Copyright (c) 2013~2022 http://www.yunbaokj.com All rights reserved.
  6. // +—————————————————————————————————————————————————————————————————————
  7. // | Author: https://gitee.com/yunbaokeji
  8. // +—————————————————————————————————————————————————————————————————————
  9. // | Date: 2022-04-30
  10. // +—————————————————————————————————————————————————————————————————————
  11. namespace app\admin\controller;
  12. use cmf\controller\AdminBaseController;
  13. use think\Db;
  14. use think\facade\Cache;
  15. use tree\Tree;
  16. use app\admin\model\AdminMenuModel;
  17. class RbacController extends AdminBaseController
  18. {
  19. /**
  20. * 角色管理列表
  21. * @adminMenu(
  22. * 'name' => '角色管理',
  23. * 'parent' => 'admin/User/default',
  24. * 'display'=> true,
  25. * 'hasView'=> true,
  26. * 'order' => 10000,
  27. * 'icon' => '',
  28. * 'remark' => '角色管理',
  29. * 'param' => ''
  30. * )
  31. * @return mixed
  32. * @throws \think\db\exception\DataNotFoundException
  33. * @throws \think\db\exception\ModelNotFoundException
  34. * @throws \think\exception\DbException
  35. */
  36. public function index()
  37. {
  38. $content = hook_one('admin_rbac_index_view');
  39. if (!empty($content)) {
  40. return $content;
  41. }
  42. $data = Db::name('role')->order(["list_order" => "ASC", "id" => "DESC"])->select();
  43. $this->assign("roles", $data);
  44. return $this->fetch();
  45. }
  46. /**
  47. * 添加角色
  48. * @adminMenu(
  49. * 'name' => '添加角色',
  50. * 'parent' => 'index',
  51. * 'display'=> false,
  52. * 'hasView'=> true,
  53. * 'order' => 10000,
  54. * 'icon' => '',
  55. * 'remark' => '添加角色',
  56. * 'param' => ''
  57. * )
  58. * @return mixed
  59. */
  60. public function roleAdd()
  61. {
  62. $content = hook_one('admin_rbac_role_add_view');
  63. if (!empty($content)) {
  64. return $content;
  65. }
  66. return $this->fetch();
  67. }
  68. /**
  69. * 添加角色提交
  70. * @adminMenu(
  71. * 'name' => '添加角色提交',
  72. * 'parent' => 'index',
  73. * 'display'=> false,
  74. * 'hasView'=> false,
  75. * 'order' => 10000,
  76. * 'icon' => '',
  77. * 'remark' => '添加角色提交',
  78. * 'param' => ''
  79. * )
  80. */
  81. public function roleAddPost()
  82. {
  83. if ($this->request->isPost()) {
  84. $data = $this->request->param();
  85. $result = $this->validate($data, 'role');
  86. if ($result !== true) {
  87. // 验证失败 输出错误信息
  88. $this->error($result);
  89. } else {
  90. $result = Db::name('role')->insert($data);
  91. if ($result) {
  92. $this->success("添加角色成功", url("rbac/index"));
  93. } else {
  94. $this->error("添加角色失败");
  95. }
  96. }
  97. }
  98. }
  99. /**
  100. * 编辑角色
  101. * @adminMenu(
  102. * 'name' => '编辑角色',
  103. * 'parent' => 'index',
  104. * 'display'=> false,
  105. * 'hasView'=> true,
  106. * 'order' => 10000,
  107. * 'icon' => '',
  108. * 'remark' => '编辑角色',
  109. * 'param' => ''
  110. * )
  111. * @return mixed
  112. * @throws \think\db\exception\DataNotFoundException
  113. * @throws \think\db\exception\ModelNotFoundException
  114. * @throws \think\exception\DbException
  115. */
  116. public function roleEdit()
  117. {
  118. $content = hook_one('admin_rbac_role_edit_view');
  119. if (!empty($content)) {
  120. return $content;
  121. }
  122. $id = $this->request->param("id", 0, 'intval');
  123. if ($id == 1) {
  124. $this->error("超级管理员角色不能被修改!");
  125. }
  126. $data = Db::name('role')->where("id", $id)->find();
  127. if (!$data) {
  128. $this->error("该角色不存在!");
  129. }
  130. $this->assign("data", $data);
  131. return $this->fetch();
  132. }
  133. /**
  134. * 编辑角色提交
  135. * @adminMenu(
  136. * 'name' => '编辑角色提交',
  137. * 'parent' => 'index',
  138. * 'display'=> false,
  139. * 'hasView'=> false,
  140. * 'order' => 10000,
  141. * 'icon' => '',
  142. * 'remark' => '编辑角色提交',
  143. * 'param' => ''
  144. * )
  145. * @throws \think\Exception
  146. * @throws \think\exception\PDOException
  147. */
  148. public function roleEditPost()
  149. {
  150. $id = $this->request->param("id", 0, 'intval');
  151. if ($id == 1) {
  152. $this->error("超级管理员角色不能被修改!");
  153. }
  154. if ($this->request->isPost()) {
  155. $data = $this->request->param();
  156. $result = $this->validate($data, 'role');
  157. if ($result !== true) {
  158. // 验证失败 输出错误信息
  159. $this->error($result);
  160. } else {
  161. if (Db::name('role')->update($data) !== false) {
  162. $this->success("保存成功!", url('rbac/index'));
  163. } else {
  164. $this->error("保存失败!");
  165. }
  166. }
  167. }
  168. }
  169. /**
  170. * 删除角色
  171. * @adminMenu(
  172. * 'name' => '删除角色',
  173. * 'parent' => 'index',
  174. * 'display'=> false,
  175. * 'hasView'=> false,
  176. * 'order' => 10000,
  177. * 'icon' => '',
  178. * 'remark' => '删除角色',
  179. * 'param' => ''
  180. * )
  181. * @throws \think\Exception
  182. * @throws \think\exception\PDOException
  183. */
  184. public function roleDelete()
  185. {
  186. $id = $this->request->param("id", 0, 'intval');
  187. if ($id == 1) {
  188. $this->error("超级管理员角色不能被删除!");
  189. }
  190. $count = Db::name('RoleUser')->where('role_id', $id)->count();
  191. if ($count > 0) {
  192. $this->error("该角色已经有用户!");
  193. } else {
  194. $status = Db::name('role')->delete($id);
  195. if (!empty($status)) {
  196. $this->success("删除成功!", url('rbac/index'));
  197. } else {
  198. $this->error("删除失败!");
  199. }
  200. }
  201. }
  202. /**
  203. * 设置角色权限
  204. * @adminMenu(
  205. * 'name' => '设置角色权限',
  206. * 'parent' => 'index',
  207. * 'display'=> false,
  208. * 'hasView'=> true,
  209. * 'order' => 10000,
  210. * 'icon' => '',
  211. * 'remark' => '设置角色权限',
  212. * 'param' => ''
  213. * )
  214. * @return mixed
  215. */
  216. public function authorize()
  217. {
  218. $content = hook_one('admin_rbac_authorize_view');
  219. if (!empty($content)) {
  220. return $content;
  221. }
  222. $AuthAccess = Db::name("AuthAccess");
  223. $adminMenuModel = new AdminMenuModel();
  224. //角色ID
  225. $roleId = $this->request->param("id", 0, 'intval');
  226. if (empty($roleId)) {
  227. $this->error("参数错误!");
  228. }
  229. $tree = new Tree();
  230. $tree->icon = ['│ ', '├─ ', '└─ '];
  231. $tree->nbsp = '&nbsp;&nbsp;&nbsp;';
  232. $result = $adminMenuModel->menuCache();
  233. $newMenus = [];
  234. $privilegeData = $AuthAccess->where("role_id", $roleId)->column("rule_name");//获取权限表数据
  235. foreach ($result as $m) {
  236. $newMenus[$m['id']] = $m;
  237. }
  238. foreach ($result as $n => $t) {
  239. $result[$n]['checked'] = ($this->_isChecked($t, $privilegeData)) ? ' checked' : '';
  240. $result[$n]['level'] = $this->_getLevel($t['id'], $newMenus);
  241. $result[$n]['style'] = empty($t['parent_id']) ? '' : 'display:none;';
  242. $result[$n]['parentIdNode'] = ($t['parent_id']) ? ' class="child-of-node-' . $t['parent_id'] . '"' : '';
  243. }
  244. $str = "<tr id='node-\$id'\$parentIdNode style='\$style'>
  245. <td style='padding-left:30px;'>\$spacer<input type='checkbox' name='menuId[]' value='\$id' level='\$level' \$checked onclick='javascript:checknode(this);'> \$name</td>
  246. </tr>";
  247. $tree->init($result);
  248. $category = $tree->getTree(0, $str);
  249. $this->assign("category", $category);
  250. $this->assign("roleId", $roleId);
  251. return $this->fetch();
  252. }
  253. /**
  254. * 角色授权提交
  255. * @adminMenu(
  256. * 'name' => '角色授权提交',
  257. * 'parent' => 'index',
  258. * 'display'=> false,
  259. * 'hasView'=> false,
  260. * 'order' => 10000,
  261. * 'icon' => '',
  262. * 'remark' => '角色授权提交',
  263. * 'param' => ''
  264. * )
  265. * @throws \think\Exception
  266. * @throws \think\db\exception\DataNotFoundException
  267. * @throws \think\db\exception\ModelNotFoundException
  268. * @throws \think\exception\DbException
  269. * @throws \think\exception\PDOException
  270. */
  271. public function authorizePost()
  272. {
  273. if ($this->request->isPost()) {
  274. $roleId = $this->request->param("roleId", 0, 'intval');
  275. if (!$roleId) {
  276. $this->error("需要授权的角色不存在!");
  277. }
  278. if (is_array($this->request->param('menuId/a')) && count($this->request->param('menuId/a')) > 0) {
  279. Db::name("authAccess")->where(["role_id" => $roleId, 'type' => 'admin_url'])->delete();
  280. foreach ($_POST['menuId'] as $menuId) {
  281. $menu = Db::name("adminMenu")->where("id", $menuId)->field("app,controller,action")->find();
  282. if ($menu) {
  283. $app = $menu['app'];
  284. $model = $menu['controller'];
  285. $action = $menu['action'];
  286. $name = strtolower("$app/$model/$action");
  287. Db::name("authAccess")->insert(["role_id" => $roleId, "rule_name" => $name, 'type' => 'admin_url']);
  288. }
  289. }
  290. Cache::clear('admin_menus');// 删除后台菜单缓存
  291. $this->success("授权成功!");
  292. } else {
  293. //当没有数据时,清除当前角色授权
  294. Db::name("authAccess")->where("role_id", $roleId)->delete();
  295. $this->error("没有接收到数据,执行清除授权成功!");
  296. }
  297. }
  298. }
  299. /**
  300. * 检查指定菜单是否有权限
  301. * @param array $menu menu表中数组
  302. * @param $privData
  303. * @return bool
  304. */
  305. private function _isChecked($menu, $privData)
  306. {
  307. $app = $menu['app'];
  308. $model = $menu['controller'];
  309. $action = $menu['action'];
  310. $name = strtolower("$app/$model/$action");
  311. if ($privData) {
  312. if (in_array($name, $privData)) {
  313. return true;
  314. } else {
  315. return false;
  316. }
  317. } else {
  318. return false;
  319. }
  320. }
  321. /**
  322. * 获取菜单深度
  323. * @param $id
  324. * @param array $array
  325. * @param int $i
  326. * @return int
  327. */
  328. protected function _getLevel($id, $array = [], $i = 0)
  329. {
  330. if ($array[$id]['parent_id'] == 0 || empty($array[$array[$id]['parent_id']]) || $array[$id]['parent_id'] == $id) {
  331. return $i;
  332. } else {
  333. $i++;
  334. return $this->_getLevel($array[$id]['parent_id'], $array, $i);
  335. }
  336. }
  337. //角色成员管理
  338. public function member()
  339. {
  340. //TODO 添加角色成员管理
  341. }
  342. }