RoleDrawer.vue 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. <template>
  2. <el-drawer :title="title" v-model="drawerOpened" @close="reset">
  3. <el-form
  4. label-position="left"
  5. label-width="80px"
  6. :model="formData"
  7. :rules="formRules"
  8. ref="formInstance"
  9. >
  10. <el-form-item label="角色名称" prop="roleName">
  11. <el-input placeholder="角色名称" v-model="formData.roleName" />
  12. </el-form-item>
  13. <el-form-item label="角色权限">
  14. <PermissioTreenCard title="相机权限" :tree-data="[]" ref="cameraCardInstance" />
  15. <PermissioTreenCard title="菜单权限" :tree-data="menuPermTreeData" ref="menuCardInstance" />
  16. <PermissioTreenCard title="功能权限" :tree-data="funcPermTreeData" ref="funcCardInstance" />
  17. </el-form-item>
  18. <el-form-item label="备注" prop="remark">
  19. <el-input type="textarea" placeholder="备注" v-model="formData.remark" />
  20. </el-form-item>
  21. </el-form>
  22. <template #footer>
  23. <el-button @click="reset">重置</el-button>
  24. <el-button type="primary" @click="submit">提交</el-button>
  25. </template>
  26. </el-drawer>
  27. </template>
  28. <script setup lang="ts">
  29. import { reactive, ref, computed, toRaw } from 'vue';
  30. import PermissioTreenCard from './PermissionTreeCard.vue';
  31. import { RoleForm, Role, AssignedPermissions } from '@/types/role/type';
  32. import { FormRules, FormInstance, ElMessage } from 'element-plus';
  33. import useMenuPermTree from '../hooks/useMenuPermissionTree';
  34. import useFuncPermTree from '../hooks/useFunctionPermissionTree';
  35. import { createRole, editRole, getAssignedPerms } from '@/api/system/role';
  36. defineProps<{
  37. title: string;
  38. }>();
  39. const emits = defineEmits<{
  40. (e: 'submitted'): void; // 提交之后触发的事件
  41. }>();
  42. const { menuPermTreeData } = useMenuPermTree();
  43. const { funcPermTreeData} = useFuncPermTree();
  44. const drawerOpened = ref(false);
  45. const cameraCardInstance = ref<InstanceType<typeof PermissioTreenCard>>();
  46. const menuCardInstance = ref<InstanceType<typeof PermissioTreenCard>>();
  47. const funcCardInstance = ref<InstanceType<typeof PermissioTreenCard>>();
  48. // 表单相关
  49. const defaultFormData = (): RoleForm => ({
  50. id: null,
  51. roleName: '',
  52. remark: '',
  53. cameraIds: [],
  54. menuIds: [],
  55. permIds: [],
  56. });
  57. const formData = reactive<RoleForm>(defaultFormData());
  58. const formRules: FormRules = {
  59. roleName: { required: true, trigger: 'blur', message: '请填写角色名称' },
  60. remark: {},
  61. };
  62. const formInstance = ref<FormInstance>();
  63. const isEditing = computed(() => formData.id != null);
  64. /**
  65. * 打开 drawer。如果未传递 roleId,表示创建角色;反之,表示编辑角色
  66. * @param roleId 可选
  67. */
  68. const open = (role?: Role) => {
  69. if (role) {
  70. formData.id = role.id;
  71. formData.roleName = role.roleName;
  72. formData.remark = role.remark;
  73. getAssignedPermissions(role.id);
  74. // TODO: 获取已拥有的权限
  75. }
  76. drawerOpened.value = true;
  77. };
  78. /**
  79. * 获取当前角色已分配的权限数据,并更新相应的权限数
  80. */
  81. const getAssignedPermissions = async (roleId: number) => {
  82. try {
  83. const result = await getAssignedPerms(roleId);
  84. displayAssignedPermissions(result);
  85. } catch (e) {
  86. console.error(e);
  87. }
  88. }
  89. const displayAssignedPermissions = (result: AssignedPermissions) => {
  90. cameraCardInstance.value?.setAssignedPermissions(result.cameraIds ?? []);
  91. menuCardInstance.value?.setAssignedPermissions(result.menuIds ?? []);
  92. funcCardInstance.value?.setAssignedPermissions(result.permIds ?? []);
  93. }
  94. /**
  95. * 重置表单
  96. */
  97. const reset = () => {
  98. formInstance.value?.resetFields();
  99. Object.assign(formData, defaultFormData());
  100. cameraCardInstance.value?.reset();
  101. menuCardInstance.value?.reset();
  102. funcCardInstance.value?.reset();
  103. };
  104. /**
  105. * 提交。创建和编辑统一
  106. */
  107. const submit = async () => {
  108. formData.cameraIds = cameraCardInstance.value!.getSelectedPermissions();
  109. formData.menuIds = menuCardInstance.value!.getSelectedPermissions();
  110. formData.permIds = funcCardInstance.value!.getSelectedPermissions();
  111. const api = isEditing.value ? editRole : createRole;
  112. try {
  113. await api(toRaw(formData));
  114. drawerOpened.value = false;
  115. ElMessage.success('提交成功');
  116. // 让父组件更新列表
  117. emits('submitted');
  118. } catch (e) {
  119. console.error(e);
  120. }
  121. }
  122. defineExpose({ open });
  123. </script>