permission.vue 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. <template>
  2. <div class="system-edit-role-container">
  3. <el-dialog :title="`【${state.rowName}】权限配置`" v-model="state.isShowDialog" draggable @opened="opened">
  4. <div class="custom-tree-node-container" v-loading="state.loading">
  5. <el-row :gutter="10">
  6. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  7. 展开/折叠:<el-switch
  8. v-model="state.menuExpand"
  9. @change="handleCheckedTreeExpand"
  10. inline-prompt
  11. active-text="折叠"
  12. inactive-text="展开"
  13. ></el-switch>
  14. </el-col>
  15. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  16. 全选/全不选:<el-switch
  17. v-model="state.menuNodeAll"
  18. @change="handleCheckedTreeNodeAll"
  19. inline-prompt
  20. active-text="全不选"
  21. inactive-text="全选"
  22. ></el-switch>
  23. </el-col>
  24. <!-- <el-col :xs="24" :sm="8" :md="8" :lg="8" :xl="8">
  25. 父子联动:<el-switch
  26. v-model="state.menuCheckStrictly"
  27. @change="handleCheckedTreeConnect"
  28. inline-prompt
  29. active-text="联动"
  30. inactive-text="不联动"
  31. disabled
  32. ></el-switch>
  33. </el-col> -->
  34. </el-row>
  35. <el-tree
  36. class="tree-border mt20"
  37. :data="state.menuTableData"
  38. show-checkbox
  39. ref="menuRef"
  40. node-key="permissionCode"
  41. check-on-click-node
  42. :expand-on-click-node="false"
  43. :check-strictly="!state.menuCheckStrictly"
  44. :props="{ label: 'pageName', children: 'children', class: customNodeClass }"
  45. >
  46. <template #default="{ data }">
  47. <span>{{ data.pageName }}</span>
  48. </template>
  49. </el-tree>
  50. </div>
  51. <template #footer>
  52. <span class="dialog-footer">
  53. <el-button @click="onCancel" class="default-button">取 消</el-button>
  54. <el-button type="primary" @click="onSubmit">确 定</el-button>
  55. </span>
  56. </template>
  57. </el-dialog>
  58. </div>
  59. </template>
  60. <script setup lang="ts" name="systemEditRole">
  61. import { reactive, onMounted, ref } from 'vue';
  62. import { ElMessage } from 'element-plus';
  63. import { auth } from '/@/utils/authFunction';
  64. import { getMenuList } from '/@/api/system/menu';
  65. import { getRolePower, setRolePower } from '/@/api/system/roles';
  66. // 定义子组件向父组件传值/事件
  67. const emit = defineEmits(['updateList']);
  68. interface Tree {
  69. id: number;
  70. label: string;
  71. isPenultimate?: boolean;
  72. expanded: boolean;
  73. menuType?: number;
  74. children?: Tree[];
  75. }
  76. // 定义变量内容
  77. const state = reactive<any>({
  78. isShowDialog: false,
  79. menuTableData: <any>[], //所有菜单按钮
  80. systemMenuArr: <any>[], // 选中的菜单
  81. currentRow: {},
  82. expandRowKeys: [],
  83. rowName: '',
  84. menuExpand: false, // 展开
  85. menuNodeAll: false, // 全选
  86. menuCheckStrictly: false, // 是否联动
  87. loading: false,
  88. });
  89. const menuRef = ref();
  90. const customNodeClass = (data: Tree) => {
  91. if (data.menuType === 2) {
  92. return 'is-penultimate';
  93. }
  94. return null;
  95. };
  96. // 打开弹窗
  97. const openDialog = (row: any) => {
  98. state.currentRow = row;
  99. state.systemMenuArr = <any>[];
  100. state.menuExpand = false;
  101. state.menuNodeAll = false;
  102. state.rowName = row.displayName;
  103. state.loading = true;
  104. state.isShowDialog = true;
  105. };
  106. // 树形扁平化
  107. const treeFlat = (source: any[]) => {
  108. let res: any = [];
  109. source.forEach((el) => {
  110. res.push(el);
  111. el.children && res.push(...treeFlat(el.children));
  112. });
  113. return res;
  114. };
  115. // 打开弹窗后查询已有权限
  116. const opened = () => {
  117. if (!auth('system:role:detail')) ElMessage.error('抱歉,您没有权限获取当前角色权限!');
  118. else {
  119. getRolePower({ roleid: state.currentRow.id })
  120. .then((res: any) => {
  121. let arr: string[] = res.result?.systemMenuArr ?? [];
  122. arr = arr.map((v: any) => v.code);
  123. menuRef.value.setCheckedKeys(arr);
  124. const arr1 = treeFlat(state.menuTableData);
  125. if (arr.length === arr1.length) state.menuNodeAll = true;
  126. state.loading = false;
  127. })
  128. .catch(() => {
  129. state.loading = false;
  130. });
  131. }
  132. };
  133. // 关闭弹窗
  134. const closeDialog = () => {
  135. state.isShowDialog = false;
  136. };
  137. // 取消
  138. const onCancel = () => {
  139. closeDialog();
  140. };
  141. let newArr: string[] = [];
  142. // 获取所有code
  143. const getCode = (arr: any) => {
  144. if (!arr) return [];
  145. arr.forEach((v: any) => {
  146. newArr.push(v.permissionCode);
  147. if (v.children?.length) {
  148. getCode(v.children);
  149. }
  150. });
  151. return newArr;
  152. };
  153. // 获取所有菜单
  154. const getMenuListApi = () => {
  155. getMenuList().then((res: any) => {
  156. state.menuTableData = res?.result ?? [];
  157. });
  158. };
  159. /** 树权限(展开/折叠)*/
  160. const handleCheckedTreeExpand = (value: boolean) => {
  161. for (let i = 0; i < state.menuTableData.length; i++) {
  162. menuRef.value.store.nodesMap[state.menuTableData[i].permissionCode].expanded = value;
  163. }
  164. // expandNodes(menuRef.value.root, value);
  165. };
  166. // 遍历树形数据,设置每一项的expanded属性,实现展开收起
  167. // const expandNodes = (node: any, value: boolean) => {
  168. // node.expanded = value;
  169. // for (let i = 0; i < node.childNodes.length; i++) {
  170. // node.childNodes[i].expanded = value;
  171. // if (node.childNodes[i].childNodes.length > 0) {
  172. // expandNodes(node.childNodes[i], value);
  173. // }
  174. // }
  175. // };
  176. /** 树权限(全选/全不选) */
  177. const handleCheckedTreeNodeAll = (value: boolean) => {
  178. menuRef.value.setCheckedKeys(value ? getCode(state.menuTableData) : []);
  179. };
  180. /** 树权限(父子联动) */
  181. // const handleCheckedTreeConnect = (value: boolean) => {
  182. // state.menuCheckStrictly = value ? true : false;
  183. // };
  184. // 保存
  185. const onSubmit = () => {
  186. let systemMenuArr = <any>[];
  187. systemMenuArr = menuRef.value.getCheckedNodes();
  188. systemMenuArr = systemMenuArr.map((v: any) => {
  189. return {
  190. code: v.permissionCode,
  191. type: v.menuType,
  192. };
  193. });
  194. let req = {
  195. roleId: state.currentRow.id,
  196. roleCode: state.currentRow.name,
  197. systemMenuArr: systemMenuArr,
  198. };
  199. setRolePower(req)
  200. .then(() => {
  201. ElMessage({
  202. message: '操作成功',
  203. type: 'success',
  204. });
  205. closeDialog();
  206. emit('updateList');
  207. })
  208. .catch(() => {
  209. // 新增失败
  210. closeDialog();
  211. });
  212. };
  213. onMounted(() => {
  214. getMenuListApi();
  215. });
  216. // 暴露变量
  217. defineExpose({
  218. openDialog,
  219. closeDialog,
  220. });
  221. </script>
  222. <style scoped lang="scss">
  223. :deep(.el-table td.el-table__cell div.cell) {
  224. display: flex;
  225. box-sizing: border-box;
  226. align-items: center;
  227. }
  228. .tree-border {
  229. border: var(--el-border);
  230. border-radius: var(--el-border-radius-base);
  231. padding: 15px;
  232. }
  233. :deep(.el-tree-node.is-expanded.is-penultimate > .el-tree-node__children) {
  234. display: flex;
  235. flex-direction: row;
  236. flex-wrap: wrap;
  237. }
  238. :deep(.is-penultimate > .el-tree-node__children > div) {
  239. width: 25%;
  240. }
  241. </style>