permission.ts 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. import type { AppRouteRecordRaw, Menu } from '/@/router/types';
  2. import { defineStore } from 'pinia';
  3. import { store } from '/@/store';
  4. import { useI18n } from '/@/hooks/web/useI18n';
  5. import { useUserStore } from './user';
  6. import { useAppStoreWidthOut } from './app';
  7. import { toRaw } from 'vue';
  8. import { transformObjToRoute, flatMultiLevelRoutes } from '/@/router/helper/routeHelper';
  9. import { transformRouteToMenu } from '/@/router/helper/menuHelper';
  10. import projectSetting from '/@/settings/projectSetting';
  11. import { PermissionModeEnum } from '/@/enums/appEnum';
  12. import { asyncRoutes } from '/@/router/routes';
  13. import { ERROR_LOG_ROUTE, PAGE_NOT_FOUND_ROUTE } from '/@/router/routes/basic';
  14. import { filter } from '/@/utils/helper/treeHelper';
  15. import { getMenuList } from '/@/api/sys/menu';
  16. import { getPermCode } from '/@/api/sys/user';
  17. import { useMessage } from '/@/hooks/web/useMessage';
  18. interface PermissionState {
  19. // Permission code list
  20. permCodeList: string[];
  21. // Whether the route has been dynamically added
  22. isDynamicAddedRoute: boolean;
  23. // To trigger a menu update
  24. lastBuildMenuTime: number;
  25. // Backstage menu list
  26. backMenuList: Menu[];
  27. }
  28. export const usePermissionStore = defineStore({
  29. id: 'app-permission',
  30. state: (): PermissionState => ({
  31. permCodeList: [],
  32. // Whether the route has been dynamically added
  33. isDynamicAddedRoute: false,
  34. // To trigger a menu update
  35. lastBuildMenuTime: 0,
  36. // Backstage menu list
  37. backMenuList: [],
  38. }),
  39. getters: {
  40. getPermCodeList() {
  41. return this.permCodeList;
  42. },
  43. getBackMenuList() {
  44. return this.backMenuList;
  45. },
  46. getLastBuildMenuTime() {
  47. return this.lastBuildMenuTime;
  48. },
  49. getIsDynamicAddedRoute() {
  50. return this.isDynamicAddedRoute;
  51. },
  52. },
  53. actions: {
  54. setPermCodeList(codeList: string[]) {
  55. this.permCodeList = codeList;
  56. },
  57. setBackMenuList(list: Menu[]) {
  58. this.backMenuList = list;
  59. list?.length > 0 && this.setLastBuildMenuTime();
  60. },
  61. setLastBuildMenuTime() {
  62. this.lastBuildMenuTime = new Date().getTime();
  63. },
  64. setDynamicAddedRoute(added: boolean) {
  65. this.isDynamicAddedRoute = added;
  66. },
  67. resetState(): void {
  68. this.isDynamicAddedRoute = false;
  69. this.permCodeList = [];
  70. this.backMenuList = [];
  71. this.lastBuildMenuTime = 0;
  72. },
  73. async changePermissionCode() {
  74. const codeList = await getPermCode();
  75. this.setPermCodeList(codeList);
  76. },
  77. async buildRoutesAction(): Promise<AppRouteRecordRaw[]> {
  78. const { t } = useI18n();
  79. const userStore = useUserStore();
  80. const appStore = useAppStoreWidthOut();
  81. let routes: AppRouteRecordRaw[] = [];
  82. const roleList = toRaw(userStore.getRoleList) || [];
  83. const { permissionMode = projectSetting.permissionMode } = appStore.getProjectConfig;
  84. // role permissions
  85. if (permissionMode === PermissionModeEnum.ROLE) {
  86. const routeFilter = (route: AppRouteRecordRaw) => {
  87. const { meta } = route;
  88. const { roles } = meta || {};
  89. if (!roles) return true;
  90. return roleList.some((role) => roles.includes(role));
  91. };
  92. routes = filter(asyncRoutes, routeFilter);
  93. routes = routes.filter(routeFilter);
  94. // Convert multi-level routing to level 2 routing
  95. routes = flatMultiLevelRoutes(routes);
  96. // If you are sure that you do not need to do background dynamic permissions, please comment the entire judgment below
  97. } else if (permissionMode === PermissionModeEnum.BACK) {
  98. const { createMessage } = useMessage();
  99. createMessage.loading({
  100. content: t('sys.app.menuLoading'),
  101. duration: 1,
  102. });
  103. // !Simulate to obtain permission codes from the background,
  104. // this function may only need to be executed once, and the actual project can be put at the right time by itself
  105. let routeList: AppRouteRecordRaw[] = [];
  106. try {
  107. this.changePermissionCode();
  108. routeList = (await getMenuList()) as AppRouteRecordRaw[];
  109. } catch (error) {
  110. console.error(error);
  111. }
  112. // Dynamically introduce components
  113. routeList = transformObjToRoute(routeList);
  114. // Background routing to menu structure
  115. const backMenuList = transformRouteToMenu(routeList);
  116. this.setBackMenuList(backMenuList);
  117. routeList = flatMultiLevelRoutes(routeList);
  118. routes = [PAGE_NOT_FOUND_ROUTE, ...routeList];
  119. }
  120. routes.push(ERROR_LOG_ROUTE);
  121. return routes;
  122. },
  123. },
  124. });
  125. // Need to be used outside the setup
  126. export function usePermissionStoreWidthOut() {
  127. return usePermissionStore(store);
  128. }