backEnd.ts 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. import { RouteRecordRaw } from 'vue-router';
  2. import pinia from '@/stores/index';
  3. import { useUserInfo } from '@/stores/userInfo';
  4. import { useAppConfig } from '@/stores/appConfig';
  5. import { useRequestOldRoutes } from '@/stores/requestOldRoutes';
  6. import { Local, Cookie } from '@/utils/storage';
  7. import { NextLoading } from '@/utils/loading';
  8. import { dynamicRoutes, notFoundAndNoPower } from '@/router/route';
  9. import { formatFlatteningRoutes, formatTwoStageRoutes, router } from '@/router/index';
  10. import { useRoutesList } from '@/stores/routesList';
  11. import { useTagsViewRoutes } from '@/stores/tagsViewRoutes';
  12. import { getMenu } from '@/api/login/user';
  13. import { removeNullAndDuplicates } from '@/utils/arrayOperation';
  14. import other from '@/utils/other';
  15. import { pwdCheck } from '@/api/login';
  16. import { appConfigInfo } from '@/api/public/system';
  17. import signalR from '@/utils/signalR';
  18. const layoutModules: 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({}, { ...layoutModules }, { ...viewsModules });
  27. // 菜单格式化
  28. const formatRouter = (arr: Array<any>): Array<any> => {
  29. if (!arr) return [];
  30. return arr.map((v: any) => {
  31. const isDynamic = v.path.indexOf('/:') > -1;
  32. v.name = v.ruleName;
  33. v.meta = {
  34. title: v.pageName,
  35. icon: v.icon,
  36. isHide: v.isHide,
  37. isKeepAlive: v.isKeepAlive,
  38. isAffix: v.isAffix,
  39. isLink: v.link,
  40. isIframe: v.isIframe,
  41. ...(isDynamic && { isDynamic, isDynamicPath: v.path }),
  42. };
  43. if (v.children?.length) {
  44. v.children = formatRouter(v.children);
  45. }
  46. return v;
  47. });
  48. };
  49. // 获取系统配置 存入pinia
  50. const getAppConfigFn = async () => {
  51. try {
  52. const { result } = await appConfigInfo();
  53. useAppConfig().setAppConfigInfo({
  54. isRestApproval: result.isRestApproval ?? false, //是否开启小休审批
  55. isNeedTelNo: result.isNeedTelNo ?? false, //分机签入是否需要选择号码
  56. talkingDealTime: result.talkingDealTime ?? 0, // 自动话后整理时间
  57. isTelNeedVerify: result.isTelNeedVerify ?? false, //分机签入是否需要输入密码
  58. isCustomEvent: result.isCustomEvent ?? false, //是否开启自定义事件
  59. });
  60. console.log(
  61. `是否开启小休审批${result.isRestApproval},自动话后整理时间${result.talkingDealTime}秒,分机签入是否需要选择号码${result.isNeedTelNo},分机签入是否需要输入密码${result.isTelNeedVerify},是否开启自定义事件${result.isCustomEvent}`
  62. );
  63. } catch (e) {
  64. console.log(e);
  65. }
  66. };
  67. // 检查是否修改过密码 如果没有修改 跳转到修改密码页面
  68. const checkPwd = (): void => {
  69. pwdCheck().then((res: any) => {
  70. if (!res.result) {
  71. router.replace('/resetPwd');
  72. }
  73. });
  74. };
  75. /**
  76. * 后端控制路由:初始化方法,防止刷新时路由丢失
  77. * @method NextLoading 界面 loading 动画开始执行
  78. * @method useUserInfo().setUserInfos() 触发初始化用户信息 pinia
  79. * @method useRequestOldRoutes().setRequestOldRoutes() 存储接口原始路由(未处理component),根据需求选择使用
  80. * @method setAddRoute 添加动态路由
  81. * @method setFilterMenuAndCacheTagsViewRoutes 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  82. */
  83. export async function initBackEndControlRoutes() {
  84. // 界面 loading 动画开始执行
  85. if (window.nextLoading === undefined) NextLoading.start();
  86. // 无 token 停止执行下一步
  87. if (!Cookie.get('token')) return false;
  88. let resRouter = null;
  89. if (Local.get('requestOldRoutes')) {
  90. //获取到缓存
  91. resRouter = Local.get('requestOldRoutes');
  92. // 触发初始化用户信息 pinia
  93. const authBtnList = Local.get('authBtn') ?? [];
  94. await useUserInfo().setUserInfos(authBtnList);
  95. // 无登录权限时,添加判断
  96. // https://gitee.com/lyt-top/vue-next-admin/issues/I64HVO
  97. if (resRouter?.length <= 0) return Promise.resolve(true);
  98. } else {
  99. // 获取路由菜单数据
  100. const res: any = await getBackEndControlRoutes();
  101. const buttons = res.result?.buttons ?? []; // 权限按钮列表
  102. // 无登录权限时,添加判断
  103. // https://gitee.com/lyt-top/vue-next-admin/issues/I64HVO
  104. let routerList = res.result?.menus ?? [];
  105. if (routerList <= 0) return Promise.resolve(true);
  106. // 路由内容格式化
  107. resRouter = formatRouter(routerList);
  108. // 存储接口原始路由(未处理component),根据需求选择使用
  109. useRequestOldRoutes().setRequestOldRoutes(other.deepClone(resRouter));
  110. // 存入缓存
  111. Local.set('requestOldRoutes', resRouter);
  112. // 获取系统配置
  113. await getAppConfigFn();
  114. // 触发初始化用户信息 pinia
  115. await useUserInfo().setUserInfos(removeNullAndDuplicates(buttons));
  116. // 存入缓存
  117. Local.set('authBtn', removeNullAndDuplicates(buttons));
  118. }
  119. // 检查是否修改过密码
  120. checkPwd();
  121. // signalR 初始化signalr
  122. signalR.init();
  123. // 处理路由(component),替换 dynamicRoutes(@/router/route)第一个顶级 children 的路由
  124. dynamicRoutes[0].children = await backEndComponent(resRouter); // 首页需要权限控制
  125. // dynamicRoutes[0].children = dynamicRoutes[0].children?.concat(await backEndComponent(resRouter)); //首页如果不需要权限控制
  126. // 添加动态路由
  127. await setAddRoute();
  128. // 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  129. await setFilterMenuAndCacheTagsViewRoutes();
  130. }
  131. /**
  132. * 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
  133. * @description 用于左侧菜单、横向菜单的显示
  134. * @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
  135. */
  136. export function setFilterMenuAndCacheTagsViewRoutes() {
  137. const storesRoutesList = useRoutesList(pinia);
  138. storesRoutesList.setRoutesList(dynamicRoutes[0].children as any);
  139. setCacheTagsViewRoutes();
  140. }
  141. /**
  142. * 缓存多级嵌套数组处理后的一维数组
  143. * @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
  144. */
  145. export function setCacheTagsViewRoutes() {
  146. const storesTagsView = useTagsViewRoutes(pinia);
  147. storesTagsView.setTagsViewRoutes(formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes))[0].children);
  148. }
  149. /**
  150. * 处理路由格式及添加捕获所有路由或 404 Not found 路由
  151. * @description 替换 dynamicRoutes(@/router/route)第一个顶级 children 的路由
  152. * @returns 返回替换后的路由数组
  153. */
  154. export function setFilterRouteEnd() {
  155. let filterRouteEnd: any = formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes));
  156. // notFoundAndNoPower 防止 404、401 不在 layout 布局中,不设置的话,404、401 界面将全屏显示
  157. // 关联问题 No match found for location with path 'xxx'
  158. filterRouteEnd[0].children = [...filterRouteEnd[0].children, ...notFoundAndNoPower];
  159. return filterRouteEnd;
  160. }
  161. /**
  162. * 添加动态路由
  163. * @method router.addRoute
  164. * @description 此处循环为 dynamicRoutes(@/router/route)第一个顶级 children 的路由一维数组,非多级嵌套
  165. * @link 参考:https://next.router.vuejs.org/zh/api/#addroute
  166. */
  167. export async function setAddRoute() {
  168. await setFilterRouteEnd().forEach((route: RouteRecordRaw) => {
  169. router.addRoute(route);
  170. });
  171. }
  172. /**
  173. * 请求后端路由菜单接口
  174. * @description isRequestRoutes 为 true,则开启后端控制路由
  175. * @returns 返回后端路由菜单数据
  176. */
  177. export function getBackEndControlRoutes() {
  178. // 管理员 admin
  179. return getMenu();
  180. }
  181. /**
  182. * 重新请求后端路由菜单接口
  183. * @description 用于菜单管理界面刷新菜单(未进行测试)
  184. * @description 路径:/src/views/system/menu/component/Menu-add.vue
  185. */
  186. export function setBackEndControlRefreshRoutes() {
  187. Local.remove('requestOldRoutes');
  188. getBackEndControlRoutes();
  189. window.location.reload();
  190. }
  191. /**
  192. * 后端路由 component 转换
  193. * @param routes 后端返回的路由表数组
  194. * @returns 返回处理成函数后的 component
  195. */
  196. export function backEndComponent(routes: any) {
  197. if (!routes) return;
  198. return routes.map((item: any) => {
  199. if (item.component) item.component = dynamicImport(dynamicViewsModules, item.component as string);
  200. item.children && backEndComponent(item.children);
  201. return item;
  202. });
  203. }
  204. /**
  205. * 后端路由 component 转换函数
  206. * @param dynamicViewsModules 获取目录下的 .vue、.tsx 全部文件
  207. * @param component 当前要处理项 component
  208. * @returns 返回处理成函数后的 component
  209. */
  210. export function dynamicImport(dynamicViewsModules: Record<string, Function>, component: string) {
  211. const keys = Object.keys(dynamicViewsModules);
  212. const matchKeys = keys.filter((key) => {
  213. const k = key.replace(/..\/views|../, '');
  214. return k.startsWith(`${component}`) || k.startsWith(`/${component}`);
  215. });
  216. if (matchKeys?.length === 1) {
  217. const matchKey = matchKeys[0];
  218. return dynamicViewsModules[matchKey];
  219. }
  220. if (matchKeys?.length > 1) {
  221. return false;
  222. }
  223. }