permission.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  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, getUserPermissionByToken } 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(); //getUserPermissionByToken
  116. // const codeList = systemPermission.codeList;
  117. const systemPermission = await getUserPermissionByToken(); //getUserPermissionByToken
  118. const codeList = systemPermission.auth;
  119. this.setPermCodeList(codeList);
  120. this.setAuthData(systemPermission);
  121. },
  122. async buildRoutesAction(): Promise<AppRouteRecordRaw[]> {
  123. const { t } = useI18n();
  124. const userStore = useUserStore();
  125. const appStore = useAppStoreWithOut();
  126. let routes: AppRouteRecordRaw[] = [];
  127. const roleList = toRaw(userStore.getRoleList) || [];
  128. const { permissionMode = projectSetting.permissionMode } = appStore.getProjectConfig;
  129. const routeFilter = (route: AppRouteRecordRaw) => {
  130. const { meta } = route;
  131. const { roles } = meta || {};
  132. if (!roles) return true;
  133. return roleList.some((role) => roles.includes(role));
  134. };
  135. const routeRemoveIgnoreFilter = (route: AppRouteRecordRaw) => {
  136. const { meta } = route;
  137. const { ignoreRoute } = meta || {};
  138. return !ignoreRoute;
  139. };
  140. /**
  141. * @description 根据设置的首页path,修正routes中的affix标记(固定首页)
  142. * */
  143. const patchHomeAffix = (routes: AppRouteRecordRaw[]) => {
  144. const glob = useGlobSetting();
  145. if (!routes || routes.length === 0) return;
  146. let homePath: string = userStore.getUserInfo.homePath || glob.homePath || PageEnum.BASE_HOME;
  147. function patcher(routes: AppRouteRecordRaw[], parentPath = '') {
  148. if (parentPath) parentPath = parentPath + '/';
  149. routes.forEach((route: AppRouteRecordRaw) => {
  150. const { path, children, redirect } = route;
  151. const currentPath = path.startsWith('/') ? path : parentPath + path;
  152. if (currentPath === homePath) {
  153. if (redirect) {
  154. homePath = route.redirect! as string;
  155. } else {
  156. route.meta = Object.assign({}, route.meta, { affix: true });
  157. throw new Error('end');
  158. }
  159. }
  160. children && children.length > 0 && patcher(children, currentPath);
  161. });
  162. }
  163. try {
  164. patcher(routes);
  165. } catch (e) {
  166. // 已处理完毕跳出循环
  167. }
  168. return;
  169. };
  170. switch (permissionMode) {
  171. case PermissionModeEnum.ROLE:
  172. routes = filter(asyncRoutes, routeFilter);
  173. routes = routes.filter(routeFilter);
  174. // 将多级路由转换为二级
  175. routes = flatMultiLevelRoutes(routes);
  176. break;
  177. case PermissionModeEnum.ROUTE_MAPPING:
  178. routes = filter(asyncRoutes, routeFilter);
  179. routes = routes.filter(routeFilter);
  180. const menuList = transformRouteToMenu(routes, true);
  181. routes = filter(routes, routeRemoveIgnoreFilter);
  182. routes = routes.filter(routeRemoveIgnoreFilter);
  183. menuList.sort((a, b) => {
  184. return (a.meta?.orderNo || 0) - (b.meta?.orderNo || 0);
  185. });
  186. this.setFrontMenuList(menuList);
  187. // 将多级路由转换为二级
  188. routes = flatMultiLevelRoutes(routes);
  189. break;
  190. // 后台菜单构建
  191. case PermissionModeEnum.BACK:
  192. const { createMessage, createWarningModal } = useMessage();
  193. // 菜单加载提示
  194. // createMessage.loading({
  195. // content: t('sys.app.menuLoading'),
  196. // duration: 1,
  197. // });
  198. // 从后台获取权限码,
  199. // 这个函数可能只需要执行一次,并且实际的项目可以在正确的时间被放置
  200. let routeList: AppRouteRecordRaw[] = [];
  201. try {
  202. this.changePermissionCode();
  203. routeList = (await getMenuList()) as AppRouteRecordRaw[];
  204. // update-begin----author:sunjianlei---date:20220315------for: 判断是否是 vue3 版本的菜单 ---
  205. let hasIndex = false;
  206. let hasIcon = false;
  207. for (const menuItem of routeList) {
  208. // 条件1:判断组件是否是 layouts/default/index
  209. if (!hasIndex) {
  210. hasIndex =
  211. menuItem.component === 'layouts/default/index' ||
  212. menuItem.component === 'layouts/RouteView' ||
  213. menuItem.path === '/dashboard/analysis';
  214. }
  215. // 条件2:判断图标是否带有 冒号
  216. if (!hasIcon) {
  217. hasIcon = !!menuItem.meta?.icon?.includes(':');
  218. }
  219. // 满足任何一个条件都直接跳出循环
  220. if (hasIcon || hasIndex) {
  221. break;
  222. }
  223. }
  224. // 两个条件都不满足,就弹出提示框
  225. if (!hasIcon && !hasIndex) {
  226. // 延迟1.5秒之后再出现提示,否则提示框出不来
  227. setTimeout(
  228. () =>
  229. createWarningModal({
  230. title: '检测提示',
  231. content: '当前菜单表是 <b>Vue2版本</b>,导致菜单加载异常!<br>点击确认,切换到Vue3版菜单!',
  232. onOk: function () {
  233. switchVue3Menu();
  234. location.reload();
  235. },
  236. }),
  237. 1000
  238. );
  239. }
  240. // update-end----author:sunjianlei---date:20220315------for: 判断是否是 vue3 版本的菜单 ---
  241. } catch (error) {
  242. console.error(error);
  243. }
  244. // 组件地址前加斜杠处理 author: lsq date:2021-09-08
  245. routeList = addSlashToRouteComponent(routeList);
  246. // 动态引入组件
  247. routeList = transformObjToRoute(routeList);
  248. // 构建后台路由菜单
  249. const backMenuList = transformRouteToMenu(routeList);
  250. this.setBackMenuList(backMenuList);
  251. // 删除meta.ignoreRoute项
  252. routeList = filter(routeList, routeRemoveIgnoreFilter);
  253. routeList = routeList.filter(routeRemoveIgnoreFilter);
  254. routeList = flatMultiLevelRoutes(routeList);
  255. routes = [PAGE_NOT_FOUND_ROUTE, QIANKUN_ROUTE, , ...routeList];
  256. break;
  257. }
  258. routes.push(ERROR_LOG_ROUTE);
  259. patchHomeAffix(routes);
  260. return routes;
  261. },
  262. setAuthData(systemPermission) {
  263. this.authList = systemPermission.auth;
  264. this.allAuthList = systemPermission.allAuth;
  265. this.sysSafeMode = systemPermission.sysSafeMode;
  266. },
  267. setAuthList(authList: AuthItem[]) {
  268. this.authList = authList;
  269. },
  270. setAllAuthList(authList: AuthItem[]) {
  271. this.allAuthList = authList;
  272. },
  273. //update-begin-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  274. setOnlineSubTableAuth(code, hideBtnList) {
  275. this.onlineSubTableAuthMap[code] = hideBtnList;
  276. },
  277. //update-end-author:taoyan date:2022-6-1 for: VUEN-1162 子表按钮没控制
  278. },
  279. });
  280. // 需要在设置之外使用
  281. export function usePermissionStoreWithOut() {
  282. return usePermissionStore(store);
  283. }