permission.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  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 { useAppStoreWithOut } from './app';
  7. import { toRaw } from 'vue';
  8. import { transformObjToRoute, flatMultiLevelRoutes, addSlashToRouteComponent } 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, QIANKUN_ROUTE } from '/@/router/routes/basic';
  14. import { filter } from '/@/utils/helper/treeHelper';
  15. import { getMenuList, switchVue3Menu } from '/@/api/sys/menu';
  16. import { getPermCode } from '/@/api/sys/user';
  17. import { useMessage } from '/@/hooks/web/useMessage';
  18. import { PageEnum } from '/@/enums/pageEnum';
  19. import { cloneDeep } from 'lodash-es';
  20. import { useGlobSetting } from '/@/hooks/setting';
  21. // 系统权限
  22. interface AuthItem {
  23. // 菜单权限编码,例如:“sys:schedule:list,sys:schedule:info”,多个逗号隔开
  24. action: string;
  25. // 权限策略1显示2禁用
  26. type: string | number;
  27. // 权限状态(0无效1有效)
  28. status: string | number;
  29. // 权限名称
  30. describe?: string;
  31. isAuth?: boolean;
  32. }
  33. interface PermissionState {
  34. // Permission code list
  35. permCodeList: string[] | number[];
  36. // Whether the route has been dynamically added
  37. isDynamicAddedRoute: boolean;
  38. // To trigger a menu update
  39. lastBuildMenuTime: number;
  40. // Backstage menu list
  41. backMenuList: Menu[];
  42. frontMenuList: Menu[];
  43. // 用户所拥有的权限
  44. authList: AuthItem[];
  45. // 全部权限配置
  46. allAuthList: AuthItem[];
  47. // 系统安全模式
  48. sysSafeMode: boolean;
  49. // online子表按钮权限
  50. onlineSubTableAuthMap: object;
  51. }
  52. export const usePermissionStore = defineStore({
  53. id: 'app-permission',
  54. state: (): PermissionState => ({
  55. permCodeList: [],
  56. // Whether the route has been dynamically added
  57. isDynamicAddedRoute: false,
  58. // To trigger a menu update
  59. lastBuildMenuTime: 0,
  60. // Backstage menu list
  61. backMenuList: [],
  62. // menu List
  63. frontMenuList: [],
  64. authList: [],
  65. allAuthList: [],
  66. sysSafeMode: false,
  67. onlineSubTableAuthMap: {},
  68. }),
  69. getters: {
  70. getPermCodeList(): string[] | number[] {
  71. return this.permCodeList;
  72. },
  73. getBackMenuList(): Menu[] {
  74. return this.backMenuList;
  75. },
  76. getFrontMenuList(): Menu[] {
  77. return this.frontMenuList;
  78. },
  79. getLastBuildMenuTime(): number {
  80. return this.lastBuildMenuTime;
  81. },
  82. getIsDynamicAddedRoute(): boolean {
  83. return this.isDynamicAddedRoute;
  84. },
  85. //update-begin-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  86. getOnlineSubTableAuth: (state) => {
  87. return (code) => state.onlineSubTableAuthMap[code];
  88. },
  89. //update-end-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  90. },
  91. actions: {
  92. setPermCodeList(codeList: string[]) {
  93. this.permCodeList = codeList;
  94. },
  95. setBackMenuList(list: Menu[]) {
  96. this.backMenuList = list;
  97. list?.length > 0 && this.setLastBuildMenuTime();
  98. },
  99. setFrontMenuList(list: Menu[]) {
  100. this.frontMenuList = list;
  101. },
  102. setLastBuildMenuTime() {
  103. this.lastBuildMenuTime = new Date().getTime();
  104. },
  105. setDynamicAddedRoute(added: boolean) {
  106. this.isDynamicAddedRoute = added;
  107. },
  108. resetState(): void {
  109. this.isDynamicAddedRoute = false;
  110. this.permCodeList = [];
  111. this.backMenuList = [];
  112. this.lastBuildMenuTime = 0;
  113. },
  114. async changePermissionCode() {
  115. const systemPermission = await getPermCode();
  116. const codeList = systemPermission.codeList;
  117. this.setPermCodeList(codeList);
  118. this.setAuthData(systemPermission);
  119. },
  120. async buildRoutesAction(): Promise<AppRouteRecordRaw[]> {
  121. const { t } = useI18n();
  122. const userStore = useUserStore();
  123. const appStore = useAppStoreWithOut();
  124. let routes: AppRouteRecordRaw[] = [];
  125. const roleList = toRaw(userStore.getRoleList) || [];
  126. const { permissionMode = projectSetting.permissionMode } = appStore.getProjectConfig;
  127. const routeFilter = (route: AppRouteRecordRaw) => {
  128. const { meta } = route;
  129. const { roles } = meta || {};
  130. if (!roles) return true;
  131. return roleList.some((role) => roles.includes(role));
  132. };
  133. const routeRemoveIgnoreFilter = (route: AppRouteRecordRaw) => {
  134. const { meta } = route;
  135. const { ignoreRoute } = meta || {};
  136. return !ignoreRoute;
  137. };
  138. /**
  139. * @description 根据设置的首页path,修正routes中的affix标记(固定首页)
  140. * */
  141. const patchHomeAffix = (routes: AppRouteRecordRaw[]) => {
  142. const glob = useGlobSetting();
  143. if (!routes || routes.length === 0) return;
  144. let homePath: string = userStore.getUserInfo.homePath || glob.homePath || PageEnum.BASE_HOME;
  145. function patcher(routes: AppRouteRecordRaw[], parentPath = '') {
  146. if (parentPath) parentPath = parentPath + '/';
  147. routes.forEach((route: AppRouteRecordRaw) => {
  148. const { path, children, redirect } = route;
  149. const currentPath = path.startsWith('/') ? path : parentPath + path;
  150. if (currentPath === homePath) {
  151. if (redirect) {
  152. homePath = route.redirect! as string;
  153. } else {
  154. route.meta = Object.assign({}, route.meta, { affix: true });
  155. throw new Error('end');
  156. }
  157. }
  158. children && children.length > 0 && patcher(children, currentPath);
  159. });
  160. }
  161. try {
  162. patcher(routes);
  163. } catch (e) {
  164. // 已处理完毕跳出循环
  165. }
  166. return;
  167. };
  168. switch (permissionMode) {
  169. case PermissionModeEnum.ROLE:
  170. routes = filter(asyncRoutes, routeFilter);
  171. routes = routes.filter(routeFilter);
  172. // 将多级路由转换为二级
  173. routes = flatMultiLevelRoutes(routes);
  174. break;
  175. case PermissionModeEnum.ROUTE_MAPPING:
  176. routes = filter(asyncRoutes, routeFilter);
  177. routes = routes.filter(routeFilter);
  178. const menuList = transformRouteToMenu(routes, true);
  179. routes = filter(routes, routeRemoveIgnoreFilter);
  180. routes = routes.filter(routeRemoveIgnoreFilter);
  181. menuList.sort((a, b) => {
  182. return (a.meta?.orderNo || 0) - (b.meta?.orderNo || 0);
  183. });
  184. this.setFrontMenuList(menuList);
  185. // 将多级路由转换为二级
  186. routes = flatMultiLevelRoutes(routes);
  187. break;
  188. // 后台菜单构建
  189. case PermissionModeEnum.BACK:
  190. const { createMessage, createWarningModal } = useMessage();
  191. // 菜单加载提示
  192. // createMessage.loading({
  193. // content: t('sys.app.menuLoading'),
  194. // duration: 1,
  195. // });
  196. // 从后台获取权限码,
  197. // 这个函数可能只需要执行一次,并且实际的项目可以在正确的时间被放置
  198. let routeList: AppRouteRecordRaw[] = [];
  199. try {
  200. this.changePermissionCode();
  201. routeList = (await getMenuList()) as AppRouteRecordRaw[];
  202. // update-begin----author:sunjianlei---date:20220315------for: 判断是否是 vue3 版本的菜单 ---
  203. let hasIndex = false;
  204. let hasIcon = false;
  205. for (const menuItem of routeList) {
  206. // 条件1:判断组件是否是 layouts/default/index
  207. if (!hasIndex) {
  208. hasIndex =
  209. menuItem.component === 'layouts/default/index' ||
  210. menuItem.component === 'layouts/RouteView' ||
  211. menuItem.path === '/dashboard/analysis';
  212. }
  213. // 条件2:判断图标是否带有 冒号
  214. if (!hasIcon) {
  215. hasIcon = !!menuItem.meta?.icon?.includes(':');
  216. }
  217. // 满足任何一个条件都直接跳出循环
  218. if (hasIcon || hasIndex) {
  219. break;
  220. }
  221. }
  222. // 两个条件都不满足,就弹出提示框
  223. if (!hasIcon && !hasIndex) {
  224. // 延迟1.5秒之后再出现提示,否则提示框出不来
  225. setTimeout(
  226. () =>
  227. createWarningModal({
  228. title: '检测提示',
  229. content: '当前菜单表是 <b>Vue2版本</b>,导致菜单加载异常!<br>点击确认,切换到Vue3版菜单!',
  230. onOk: function () {
  231. switchVue3Menu();
  232. location.reload();
  233. },
  234. }),
  235. 1000
  236. );
  237. }
  238. // update-end----author:sunjianlei---date:20220315------for: 判断是否是 vue3 版本的菜单 ---
  239. } catch (error) {
  240. console.error(error);
  241. }
  242. // 组件地址前加斜杠处理 author: lsq date:2021-09-08
  243. routeList = addSlashToRouteComponent(routeList);
  244. // 动态引入组件
  245. routeList = transformObjToRoute(routeList);
  246. // 构建后台路由菜单
  247. const backMenuList = transformRouteToMenu(routeList);
  248. this.setBackMenuList(backMenuList);
  249. // 删除meta.ignoreRoute项
  250. routeList = filter(routeList, routeRemoveIgnoreFilter);
  251. routeList = routeList.filter(routeRemoveIgnoreFilter);
  252. routeList = flatMultiLevelRoutes(routeList);
  253. routes = [PAGE_NOT_FOUND_ROUTE, QIANKUN_ROUTE, , ...routeList];
  254. break;
  255. }
  256. routes.push(ERROR_LOG_ROUTE);
  257. patchHomeAffix(routes);
  258. return routes;
  259. },
  260. setAuthData(systemPermission) {
  261. this.authList = systemPermission.auth;
  262. this.allAuthList = systemPermission.allAuth;
  263. this.sysSafeMode = systemPermission.sysSafeMode;
  264. },
  265. setAuthList(authList: AuthItem[]) {
  266. this.authList = authList;
  267. },
  268. setAllAuthList(authList: AuthItem[]) {
  269. this.allAuthList = authList;
  270. },
  271. //update-begin-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  272. setOnlineSubTableAuth(code, hideBtnList) {
  273. this.onlineSubTableAuthMap[code] = hideBtnList;
  274. },
  275. //update-end-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  276. },
  277. });
  278. // 需要在设置之外使用
  279. export function usePermissionStoreWithOut() {
  280. return usePermissionStore(store);
  281. }