backEnd.ts 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. import { RouteRecordRaw } from 'vue-router';
  2. import { storeToRefs } from 'pinia';
  3. import pinia from '/@/stores/index';
  4. import { useUserInfo } from '/@/stores/userInfo';
  5. import { useShopInfo } from '/@/stores/shopInfo';
  6. import { usePublicData } from '/@/stores/publicData'
  7. import { useRequestOldRoutes } from '/@/stores/requestOldRoutes';
  8. import { Session } from '/@/utils/storage';
  9. import { NextLoading } from '/@/utils/loading';
  10. import { dynamicRoutes, notFoundAndNoPower } from '/@/router/route';
  11. import { formatTwoStageRoutes, formatFlatteningRoutes, router } from '/@/router/index';
  12. import { useRoutesList } from '/@/stores/routesList';
  13. import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
  14. import { useMenuApi } from '/@/api/menu/index';
  15. import { handleMenu } from '../utils/menu';
  16. import { BtnPermissionStore } from '/@/plugin/permission/store.permission';
  17. const menuApi = useMenuApi();
  18. const layouModules: any = import.meta.glob('../layout/routerView/*.{vue,tsx}');
  19. const viewsModules: any = import.meta.glob('../views/**/*.{vue,tsx}');
  20. // 后端控制路由
  21. /**
  22. * 获取目录下的 .vue、.tsx 全部文件
  23. * @method import.meta.glob
  24. * @link 参考:https://cn.vitejs.dev/guide/features.html#json
  25. */
  26. const dynamicViewsModules: Record<string, Function> = Object.assign({}, { ...layouModules }, { ...viewsModules });
  27. /**
  28. * 后端控制路由:初始化方法,防止刷新时路由丢失
  29. * @method NextLoading 界面 loading 动画开始执行
  30. * @method useUserInfo().setUserInfos() 触发初始化用户信息 pinia
  31. * @method useRequestOldRoutes().setRequestOldRoutes() 存储接口原始路由(未处理component),根据需求选择使用
  32. * @method setAddRoute 添加动态路由
  33. * @method setFilterMenuAndCacheTagsViewRoutes 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  34. */
  35. export async function initBackEndControlRoutes() {
  36. console.log("--------initBackEndControlRoutes------------")
  37. // 界面 loading 动画开始执行
  38. if (window.nextLoading === undefined) NextLoading.start();
  39. // 无 token 停止执行下一步
  40. if (!Session.get('token')) return false;
  41. // 触发初始化用户信息 pinia
  42. // https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP
  43. await useUserInfo().setUserInfos();
  44. await useShopInfo().initShopInfo();
  45. const publicData = usePublicData()
  46. publicData.initData();
  47. publicData.$subscribe((mutation, state) => {
  48. // console.log(mutation)
  49. // console.log(state.dateRange)
  50. publicData.setDateRange(state.dateRange)
  51. }, { detached: true })
  52. // 获取路由菜单数据
  53. const res = await getBackEndControlRoutes();
  54. // 处理路由(component),替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
  55. dynamicRoutes[0].children = await backEndComponent(handleMenu(res.data));
  56. // 添加动态路由
  57. await setAddRoute();
  58. // 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  59. await setFilterMenuAndCacheTagsViewRoutes();
  60. }
  61. /**
  62. * 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  63. * @description 用于左侧菜单、横向菜单的显示
  64. * @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
  65. */
  66. export function setFilterMenuAndCacheTagsViewRoutes() {
  67. const storesRoutesList = useRoutesList(pinia);
  68. storesRoutesList.setRoutesList(dynamicRoutes[0].children as any);
  69. setCacheTagsViewRoutes();
  70. }
  71. /**
  72. * 缓存多级嵌套数组处理后的一维数组
  73. * @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
  74. */
  75. export function setCacheTagsViewRoutes() {
  76. const storesTagsView = useTagsViewRoutes(pinia);
  77. storesTagsView.setTagsViewRoutes(formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes))[0].children);
  78. }
  79. /**
  80. * 处理路由格式及添加捕获所有路由或 404 Not found 路由
  81. * @description 替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
  82. * @returns 返回替换后的路由数组
  83. */
  84. export function setFilterRouteEnd() {
  85. let filterRouteEnd: any = formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes));
  86. filterRouteEnd[0].children = [...filterRouteEnd[0].children, ...notFoundAndNoPower];
  87. return filterRouteEnd;
  88. }
  89. /**
  90. * 添加动态路由
  91. * @method router.addRoute
  92. * @description 此处循环为 dynamicRoutes(/@/router/route)第一个顶级 children 的路由一维数组,非多级嵌套
  93. * @link 参考:https://next.router.vuejs.org/zh/api/#addroute
  94. */
  95. export async function setAddRoute() {
  96. await setFilterRouteEnd().forEach((route: RouteRecordRaw) => {
  97. router.addRoute(route);
  98. });
  99. }
  100. /**
  101. * 请求后端路由菜单接口
  102. * @description isRequestRoutes 为 true,则开启后端控制路由
  103. * @returns 返回后端路由菜单数据
  104. */
  105. export function getBackEndControlRoutes() {
  106. //获取所有的按钮权限
  107. BtnPermissionStore().getBtnPermissionStore();
  108. return menuApi.getSystemMenu();
  109. }
  110. /**
  111. * 重新请求后端路由菜单接口
  112. * @description 用于菜单管理界面刷新菜单(未进行测试)
  113. * @description 路径:/src/views/system/menu/component/addMenu.vue
  114. */
  115. export function setBackEndControlRefreshRoutes() {
  116. getBackEndControlRoutes();
  117. }
  118. /**
  119. * 后端路由 component 转换
  120. * @param routes 后端返回的路由表数组
  121. * @returns 返回处理成函数后的 component
  122. */
  123. export function backEndComponent(routes: any) {
  124. if (!routes) return;
  125. return routes.map((item: any) => {
  126. if (item.component) item.component = dynamicImport(dynamicViewsModules, item.component as string);
  127. item.children && backEndComponent(item.children);
  128. return item;
  129. });
  130. }
  131. /**
  132. * 后端路由 component 转换函数
  133. * @param dynamicViewsModules 获取目录下的 .vue、.tsx 全部文件
  134. * @param component 当前要处理项 component
  135. * @returns 返回处理成函数后的 component
  136. */
  137. export function dynamicImport(dynamicViewsModules: Record<string, Function>, component: string) {
  138. const keys = Object.keys(dynamicViewsModules);
  139. const matchKeys = keys.filter((key) => {
  140. const k = key.replace(/..\/views|../, '');
  141. return k.startsWith(`${component}`) || k.startsWith(`/${component}`);
  142. });
  143. if (matchKeys?.length === 1) {
  144. const matchKey = matchKeys[0];
  145. return dynamicViewsModules[matchKey];
  146. }
  147. if (matchKeys?.length > 1) {
  148. return false;
  149. }
  150. }