@utogether/udp-core 1.0.1-beta.16 → 1.0.1-beta.17

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (122) hide show
  1. package/build/plugins.ts +39 -39
  2. package/dist/{403-RBzulrcG.js → 403-C3cfjZIT.js} +1 -1
  3. package/dist/{404-DX-dU_6C.js → 404-Bg3j7QIo.js} +1 -1
  4. package/dist/{500-BCTk0__d.js → 500-CMT7Zyy7.js} +1 -1
  5. package/dist/{AuthorityInfo-BADTOenq.js → AuthorityInfo-B1mvqs4x.js} +1 -1
  6. package/dist/{AuthorityInfo.vue_vue_type_style_index_0_lang-Cv46yAx_.js → AuthorityInfo.vue_vue_type_style_index_0_lang-DwoW7QWO.js} +1 -1
  7. package/dist/{Company-B-61KrWd.js → Company-qTguidK8.js} +3 -3
  8. package/dist/{CompanyPanel-CawSrEff.js → CompanyPanel-BBC-o-k3.js} +8 -8
  9. package/dist/{Department-CrzJPtvP.js → Department-C3mKZOnK.js} +3 -3
  10. package/dist/{DepartmentPanel-DYuL8H-j.js → DepartmentPanel-CNdpOHBT.js} +3 -3
  11. package/dist/{DesignPanel-DEAv6V68.js → DesignPanel-D9jrRCoG.js} +1 -1
  12. package/dist/{DesignPanel.vue_vue_type_style_index_0_lang-BsnXZ2td.js → DesignPanel.vue_vue_type_style_index_0_lang-A-Aj1Hl0.js} +2 -2
  13. package/dist/{DictView-BuO-2b-c.js → DictView-CgdVkSiP.js} +1 -1
  14. package/dist/{InvOrganization-CBgKBJAn.js → InvOrganization-RE7s3vBq.js} +1 -1
  15. package/dist/{Org-BaQw2r-o.js → Org-wXDGvTrS.js} +1 -1
  16. package/dist/{Preview-DvCrmKNk.js → Preview-Pvt1__BV.js} +1 -1
  17. package/dist/{ReportDefine-DyhWz0as.js → ReportDefine-xEQsgFyv.js} +1 -1
  18. package/dist/{ReportDesign-DgZ2f2zv.js → ReportDesign-jOSuEFH9.js} +2 -2
  19. package/dist/{ReportQuery-DTZYbCDi.js → ReportQuery-DxEmmGiN.js} +1 -1
  20. package/dist/{ReportQueryFrom-B-vtJegP.js → ReportQueryFrom-CFlxfS5M.js} +1 -1
  21. package/dist/{ReportQueryFrom.vue_vue_type_style_index_0_lang-D6izgxph.js → ReportQueryFrom.vue_vue_type_style_index_0_lang-iHpiYz8f.js} +1 -1
  22. package/dist/{ReportTemplate-DA1ZqKGf.js → ReportTemplate-Dtz3aet7.js} +1 -1
  23. package/dist/{Role-mbqoav5t.js → Role-B_NsHIyE.js} +3 -3
  24. package/dist/{RoleAssign-ltEJIEcr.js → RoleAssign-C9wm4zq4.js} +8 -8
  25. package/dist/{RolePanel-u90v9eS0.js → RolePanel-BlHidBR0.js} +1 -1
  26. package/dist/{RolePanel-1g1a-G8D.js → RolePanel-DWM9mfNs.js} +1 -1
  27. package/dist/{RolePanel.vue_vue_type_script_setup_true_lang-D9YxpGnt.js → RolePanel.vue_vue_type_script_setup_true_lang-Cn4RRzXm.js} +1 -1
  28. package/dist/{RolePanel.vue_vue_type_script_setup_true_lang-LC4Egk-Q.js → RolePanel.vue_vue_type_script_setup_true_lang-DVQ9aqb1.js} +3 -3
  29. package/dist/{ScrollPanel.vue_vue_type_style_index_0_lang-DZ3W1epK.js → ScrollPanel.vue_vue_type_style_index_0_lang-CCm-_i8O.js} +1 -1
  30. package/dist/{Staff-CJ2GgCL3.js → Staff-BfDc61QS.js} +3 -3
  31. package/dist/{StaffInfo-3KkodDCK.js → StaffInfo-BaJFo4mg.js} +1 -1
  32. package/dist/{StaffInfo.vue_vue_type_script_setup_true_lang-CW-isIxJ.js → StaffInfo.vue_vue_type_script_setup_true_lang-Bwj4x1wu.js} +1 -1
  33. package/dist/{StaffPanel-DoXPSoY5.js → StaffPanel-B6I2ZS6A.js} +1 -1
  34. package/dist/{StaffPanel.vue_vue_type_script_setup_true_lang-DvhGnOMZ.js → StaffPanel.vue_vue_type_script_setup_true_lang-vbyS4w3V.js} +2 -2
  35. package/dist/{SysUser-BuIZhMPh.js → SysUser-B_r9LYc5.js} +2 -2
  36. package/dist/{SysUserPanel-t1GJcJGM.js → SysUserPanel-C4ajcHPV.js} +1 -1
  37. package/dist/{SysUserPanel.vue_vue_type_script_setup_true_lang-CDDkFLlN.js → SysUserPanel.vue_vue_type_script_setup_true_lang-8Q0VvC9B.js} +43 -39
  38. package/dist/{SystemMenu-DuLEL6_l.js → SystemMenu-CyPG23uP.js} +2 -2
  39. package/dist/{UserInfo-WWby4lts.js → UserInfo-BkoGt4ib.js} +1 -1
  40. package/dist/{UserInfo.vue_vue_type_style_index_0_lang-B06ZSrJV.js → UserInfo.vue_vue_type_style_index_0_lang-DrzF1YYG.js} +1 -1
  41. package/dist/{childView-B0-EAoAi.js → childView-9vEO8ymO.js} +1 -1
  42. package/dist/{childView-IYY4L8VK.js → childView-Dh0NyiQe.js} +1 -1
  43. package/dist/{childView.vue_vue_type_style_index_0_lang-CjF8nZuj.js → childView.vue_vue_type_style_index_0_lang-BOeoZuAK.js} +2 -2
  44. package/dist/{childView.vue_vue_type_style_index_0_lang-B56v6yK7.js → childView.vue_vue_type_style_index_0_lang-CMOjfXBX.js} +1 -1
  45. package/dist/{code-rule-DDLdi3pP.js → code-rule-C7jI_1gJ.js} +1 -1
  46. package/dist/core.es.js +1 -1
  47. package/dist/{cron-task-p3-AFN1w.js → cron-task-Dg9DJXvv.js} +1 -1
  48. package/dist/{frameView-B_WAZErD.js → frameView-CeR-_hjB.js} +1 -1
  49. package/dist/{index-24fnvUBy.js → index-Bb7gIEvu.js} +611 -569
  50. package/dist/{layoutView-CoEp2afn.js → layoutView-wsLHrEeX.js} +1340 -1337
  51. package/dist/{login-DP0oP70W.js → login-CluzzdqH.js} +2 -2
  52. package/dist/{lov-view-BDoZRK-v.js → lov-view-BvQCvwbW.js} +2 -2
  53. package/dist/{menuInfo-5Xls76qB.js → menuInfo-B8RLelIC.js} +1 -1
  54. package/dist/{menuInfo.vue_vue_type_style_index_0_lang-BfZczW9a.js → menuInfo.vue_vue_type_style_index_0_lang-DvkapEZr.js} +1 -1
  55. package/dist/{pda-app-DxGYn4G7.js → pda-app-D3LNmD9a.js} +2 -2
  56. package/dist/{resource-BnYWe_-Q.js → resource-D8JA_03-.js} +1 -1
  57. package/dist/{su-welcome-B2rNhhDH.js → su-welcome-BjxeEKSo.js} +1 -1
  58. package/dist/{sys-config-Bb4WkxZK.js → sys-config-sbeLk9bP.js} +1 -1
  59. package/dist/udp-core.css +1 -9
  60. package/index.ts +48 -48
  61. package/package.json +1 -1
  62. package/src/App.vue +65 -65
  63. package/src/components/SuCharts/src/UserInfo.vue +78 -78
  64. package/src/components/SuScrollTree/ScrollPanel.vue +170 -175
  65. package/src/components/udp/{grid.vue → grid/index.vue} +27 -24
  66. package/src/components/udp/index.ts +2 -9
  67. package/src/components/udp/utils.ts +60 -107
  68. package/src/layout/components/lay-navbar/index.vue +239 -239
  69. package/src/layout/components/lay-panel/index.vue +150 -150
  70. package/src/layout/components/lay-search/index.vue +25 -25
  71. package/src/layout/components/lay-select-org/index.vue +64 -64
  72. package/src/layout/components/lay-sidebar/horizontal.vue +190 -190
  73. package/src/layout/components/lay-sidebar/mixNav.vue +260 -260
  74. package/src/layout/components/lay-sidebar/sidebar-logo.vue +92 -92
  75. package/src/layout/components/lay-tag/index.vue +20 -48
  76. package/src/layout/hooks/useNav.ts +173 -173
  77. package/src/layout/hooks/useTag.ts +5 -3
  78. package/src/layout/types.ts +3 -2
  79. package/src/main.ts +9 -6
  80. package/src/plugins/i18n/en.ts +302 -302
  81. package/src/plugins/i18n/zh.ts +356 -354
  82. package/src/plugins/vxe-table/index.ts +116 -53
  83. package/src/plugins/vxe-table/render.tsx +956 -956
  84. package/src/router/index.ts +186 -177
  85. package/src/router/modules/remaining.ts +58 -58
  86. package/src/router/utils.ts +393 -377
  87. package/src/store/modules/app.ts +1 -3
  88. package/src/store/modules/multiTags.ts +110 -109
  89. package/src/store/modules/permission.ts +113 -100
  90. package/src/style/button.scss +85 -85
  91. package/src/style/login.css +336 -336
  92. package/src/style/vxetable.scss +13 -3
  93. package/src/views/organization/company/CompanyPanel.vue +259 -259
  94. package/src/views/organization/department/DepartmentPanel.vue +269 -283
  95. package/src/views/system/menu/SystemMenu.vue +183 -183
  96. package/src/views/system/menu/menuInfo.vue +363 -363
  97. package/src/views/system/role/UserInfo.vue +195 -195
  98. package/src/views/system/role-assign/RoleAssign.vue +57 -57
  99. package/src/views/system/role-assign/RolePanel.vue +139 -139
  100. package/src/views/system/sys/sys-config.vue +336 -336
  101. package/src/views/system/sysUser/SysUserPanel.vue +279 -278
  102. package/src/views/uapp/pda/pda-app.vue +208 -208
  103. package/src/views/udev/dict/DictView.vue +118 -118
  104. package/src/views/udev/dict/childView.vue +1 -1
  105. package/src/views/udev/lov/lov-view.vue +91 -91
  106. package/src/views/ulogin/login.vue +323 -323
  107. package/src/views/upms/interface/log-out.vue +101 -101
  108. package/src/views/urpt/design/DesignPanel.vue +507 -507
  109. package/types/global.d.ts +230 -230
  110. package/src/components/udp/count-down.vue +0 -536
  111. package/src/components/udp/flip-down/FlipCard/flip-card.vue +0 -251
  112. package/src/components/udp/flip-down/FlipCard/interfaces.ts +0 -4
  113. package/src/components/udp/flip-down/FlipClock/flip-clock.vue +0 -113
  114. package/src/components/udp/form-upload.vue +0 -482
  115. package/src/components/udp/form.vue +0 -112
  116. package/src/components/udp/lov.vue +0 -388
  117. package/src/components/udp/modal-form.vue +0 -190
  118. package/src/components/udp/modal-grid.vue +0 -298
  119. package/src/components/udp/upload.vue +0 -423
  120. package/src/utils/udp/http/index.ts +0 -294
  121. package/src/utils/udp/http/types.d.ts +0 -49
  122. package/src/utils/udp/useRender.ts +0 -431
@@ -1,377 +1,393 @@
1
- import type { RouteRecordRaw, RouteComponent } from 'vue-router';
2
- import { isProxy, toRaw, defineAsyncComponent } from 'vue';
3
- import { useTimeoutFn } from '@vueuse/core';
4
- import { storageLocal, buildHierarchyTree, dbstorage } from '@utogether/utils';
5
- import { clone } from 'xe-utils';
6
- import { router, getViews } from './index';
7
- import { getConfig } from '../config';
8
- import { geti18n } from '../plugins/i18n';
9
- import { routerArrays } from '../layout/types';
10
- import { usePermissionStoreHook } from '../store/modules/permission';
11
- import { useSystemStoreHook } from '../store/modules/system';
12
- import { useMultiTagsStoreHook } from '../store/modules/multiTags';
13
- import { getServiceApi } from '../api';
14
- import createComponent from './createComponent';
15
-
16
- const Layout = () => import('../layout/layoutView.vue');
17
- const IFrame = () => import('../layout/frameView.vue');
18
-
19
- // 动态路由
20
- import { initDict, initSystemInfo, getAsyncRoutes } from '../api/user';
21
- // https://cn.vitejs.dev/guide/features.html#glob-import
22
- // const modulesRoutes = import.meta.glob('/src/views/**/*.{vue,tsx}');
23
-
24
- // 菜单类型
25
- enum MenuCategoty {
26
- Menu = '5', // 菜单
27
- Lowcode = '10' // 低代码
28
- }
29
-
30
- /** 按照路由中meta下的rank等级升序来排序路由 */
31
- function ascending(arr: any[]) {
32
- arr.forEach(v => {
33
- if (v?.meta?.rank === null) v.meta.rank = undefined;
34
- if (v?.meta?.rank === 0) {
35
- if (v.name !== 'home' && v.path !== '/') {
36
- console.warn('rank only the home page can be 0');
37
- }
38
- }
39
- });
40
- return arr.sort((a: { meta: { rank: number } }, b: { meta: { rank: number } }) => {
41
- return a?.meta?.rank - b?.meta?.rank;
42
- });
43
- }
44
-
45
- /** 过滤meta中showLink为false的菜单 */
46
- function filterTree(data: RouteComponent[]) {
47
- const newTree = clone(data, true).filter(
48
- (v: { meta: { showLink: boolean } }) => v && v.meta?.showLink !== false
49
- );
50
- newTree.forEach((v: { children }) => v.children && (v.children = filterTree(v.children)));
51
- return newTree;
52
- }
53
-
54
- /** 通过指定 `key` 获取父级路径集合,默认 `key` 为 `path` */
55
- function getParentPaths(value: string, routes: RouteRecordRaw[], key = 'path') {
56
- // 深度遍历查找
57
- function dfs(routes: RouteRecordRaw[], value: string, parents: string[]) {
58
- for (let i = 0; i < routes.length; i++) {
59
- const item = routes[i];
60
- // 找到path则返回父级path
61
- // if (item[key] === value) {
62
- // return parents.length ? parents : routes[0];
63
- // }
64
- if (item[key] === value) return parents;
65
- // children不存在或为空则不递归
66
- if (!item.children || !item.children.length) continue;
67
- // 往下查找时将当前path入栈
68
- parents.push(item.path);
69
- // @ts-ignore
70
- if (dfs(item.children, value, parents).length) return parents;
71
- // 深度遍历查找未找到时当前path 出栈
72
- parents.pop();
73
- }
74
- // 未找到时返回空数组
75
- return [];
76
- }
77
-
78
- return dfs(routes, value, []);
79
- }
80
-
81
- /** 查找对应 `path` 的路由信息 */
82
- function findRouteByPath(path: string, routes: RouteRecordRaw[]) {
83
- let res = routes.find((item: { path: string }) => item.path == path);
84
- if (res) {
85
- return isProxy(res) ? toRaw(res) : res;
86
- } else {
87
- for (let i = 0; i < routes.length; i++) {
88
- if (routes[i].children instanceof Array && routes[i].children.length > 0) {
89
- res = findRouteByPath(path, routes[i].children);
90
- if (res) {
91
- return isProxy(res) ? toRaw(res) : res;
92
- }
93
- }
94
- }
95
- return null;
96
- }
97
- }
98
- function addPathMatch() {
99
- if (!router.hasRoute('pathMatch')) {
100
- router.addRoute({
101
- path: '/:pathMatch(.*)',
102
- name: 'pathMatch',
103
- redirect: '/error/404'
104
- });
105
- }
106
- }
107
-
108
- /**
109
- * @description: 添加审批动态路由
110
- * @param {*} syncRoutes 异步获取的路由表数据
111
- * @param {*} routes 格式化后的路由表数据
112
- * @return {*} 格式化后的路由表数据
113
- */
114
- // function addWorkflowRouter(syncRoutes, routes) {
115
- // const flowIdx = syncRoutes.findIndex(f => f.menuCode === 'Workflow');
116
- // const isExit = flowIdx !== -1 && syncRoutes[flowIdx].children?.some(s => s.name === 'workflowApprove');
117
- // if (!isExit) {
118
- // flowIdx !== -1 && routes[flowIdx].children?.push(workflowRouter);
119
- // }
120
- // return routes;
121
- // }
122
- // 菜单国家化
123
- function setMenui18n() {
124
- const i18n = geti18n();
125
- i18n.global?.mergeLocaleMessage('zh', menuI18n.zh);
126
- i18n.global?.mergeLocaleMessage('en', menuI18n.en);
127
- storageLocal.setItem('menu_zh', menuI18n.zh);
128
- storageLocal.setItem('menu_en', menuI18n.en);
129
- }
130
-
131
- function handleAsyncRoutes(routesList) {
132
- if (routesList?.length) {
133
- const idx = routesList.findIndex(f => f.menuCode === 'SysHome');
134
- if (idx !== -1) {
135
- // 仪表盘报表
136
- const dashboardTabs = [];
137
- const homeMenus = [];
138
- routesList[idx].children.forEach(child => {
139
- if (child.menuCategory !== '1') {
140
- dashboardTabs.push(child);
141
- } else {
142
- homeMenus.push(child);
143
- }
144
- });
145
- routesList[idx].children = homeMenus;
146
- useSystemStoreHook().setHomeMenus(dashboardTabs);
147
- }
148
- const routes = clone(addAsyncRoutes(routesList), true);
149
-
150
- // routes = addWorkflowRouter(routesList, routes);
151
- setMenui18n();
152
- formatFlatteningRoutes(routes).map((v: RouteRecordRaw) => {
153
- // 菜单国际化
154
- // 防止重复添加路由
155
- const { options } = router;
156
- if (options.routes[0].children.findIndex(value => value.path === v.path) !== -1) {
157
- return;
158
- } else {
159
- // 切记将路由push到routes后还需要使用addRoute,这样路由才能正常跳转
160
- router.options.routes[0].children.push(v);
161
- // 最终路由进行升序
162
- ascending(router.options.routes[0].children);
163
- if (!router.hasRoute(v?.name)) router.addRoute(v);
164
- }
165
- });
166
- }
167
- usePermissionStoreHook().changeSetting(routesList);
168
- if (!useMultiTagsStoreHook().getMultiTagsCache) {
169
- useMultiTagsStoreHook().handleTags('equal', [
170
- ...routerArrays,
171
- ...usePermissionStoreHook().flatteningRoutes.filter(v => v?.meta?.fixedTag)
172
- ]);
173
- }
174
- addPathMatch();
175
- }
176
-
177
- /** 初始化路由(`new Promise` 写法防止在异步请求中造成无限循环)*/
178
- async function initRouter(userName) {
179
- let asyncRouteList = [];
180
- const key = `U-${userName}-ROUTES`;
181
- // 开启动态路由缓存本地
182
- if (getConfig().CachingAsyncRoutes) {
183
- await dbstorage.getItem(key).then(data => (asyncRouteList = data));
184
- }
185
- initDict().then(res => useSystemStoreHook().setDict(res));
186
- await initSystemInfo().then(async (res: IRecord) => {
187
- useSystemStoreHook().setSystemInfo(res);
188
- await getServiceApi()
189
- .get('/uums/cusOrganization', { pageSize: 100, pageNum: 1, orgId: res.orgId })
190
- .then((res: IResponseData) => useSystemStoreHook().setInvOrgList(res?.list || []));
191
- });
192
- if (asyncRouteList?.length) {
193
- return new Promise(resolve => {
194
- handleAsyncRoutes(asyncRouteList);
195
- // 缓存最新的路由
196
- getAsyncRoutes({ userName }).then((data: any) => {
197
- dbstorage.setItem(key, data);
198
- });
199
- resolve(router);
200
- });
201
- } else {
202
- return new Promise(resolve => {
203
- getAsyncRoutes({ userName }).then((data: any) => {
204
- handleAsyncRoutes(clone(data, true));
205
- dbstorage.setItem(key, data);
206
- resolve(router);
207
- });
208
- });
209
- }
210
- }
211
-
212
- /**
213
- * 将多级嵌套路由处理成一维数组
214
- * @param routesList 传入路由
215
- * @returns 返回处理后的一维路由
216
- */
217
- function formatFlatteningRoutes(routesList: RouteRecordRaw[]) {
218
- if (routesList.length === 0) return routesList;
219
- let hierarchyList = buildHierarchyTree(routesList);
220
- for (let i = 0; i < hierarchyList.length; i++) {
221
- if (hierarchyList[i].children) {
222
- hierarchyList = hierarchyList
223
- .slice(0, i + 1)
224
- .concat(hierarchyList[i].children, hierarchyList.slice(i + 1));
225
- }
226
- }
227
- return hierarchyList;
228
- }
229
-
230
- /**
231
- * 一维数组处理成多级嵌套数组(三级及以上的路由全部拍成二级,keep-alive 只支持到二级缓存)
232
- * @param routesList 处理后的一维路由菜单数组
233
- * @returns 返回将一维数组重新处理成规定路由的格式
234
- */
235
- function formatTwoStageRoutes(routesList: RouteRecordRaw[]) {
236
- if (routesList.length === 0) return routesList;
237
- const newRoutesList: RouteRecordRaw[] = [];
238
- routesList.forEach((v: RouteRecordRaw) => {
239
- if (v.path === '/') {
240
- newRoutesList.push({
241
- component: v.component,
242
- name: v.name,
243
- path: v.path,
244
- redirect: v.redirect,
245
- meta: v.meta,
246
- children: []
247
- });
248
- } else {
249
- newRoutesList[0].children.push({ ...v });
250
- }
251
- });
252
- return newRoutesList;
253
- }
254
-
255
- /** 处理缓存路由(添加、删除、刷新) */
256
- function handleAliveRoute({ name }: ToRouteType, mode?: string) {
257
- switch (mode) {
258
- case 'add':
259
- usePermissionStoreHook().cacheOperate({ mode: 'add', name });
260
- break;
261
- case 'delete':
262
- usePermissionStoreHook().cacheOperate({ mode: 'delete', name });
263
- break;
264
- case 'refresh':
265
- usePermissionStoreHook().cacheOperate({ mode: 'refresh', name });
266
- break;
267
- default:
268
- usePermissionStoreHook().cacheOperate({ mode: 'delete', name });
269
- useTimeoutFn(() => {
270
- usePermissionStoreHook().cacheOperate({ mode: 'add', name });
271
- }, 100);
272
- }
273
- }
274
-
275
- /** 过滤后端传来的动态路由 重新生成规范路由 */
276
- const menuI18n = {
277
- zh: {},
278
- en: {}
279
- };
280
- const addAsyncRoutes = (arrRoutes: Array<any>) => {
281
- const modulesRoutes = getViews();
282
-
283
- if (!arrRoutes || !arrRoutes.length) return;
284
- const modulesRoutesKeys = Object.keys(modulesRoutes);
285
- for (let i = 0; i < arrRoutes.length; i++) {
286
- const v = arrRoutes[i];
287
-
288
- if (v?.children && v.children.length && !v.redirect) {
289
- // 父级的redirect属性取值:如果子级存在且父级的redirect属性不存在,默认取第一个子级的path;如果子级存在且父级的redirect属性存在,取存在的redirect属性,会覆盖默认值
290
- v.redirect = v.children[0].path;
291
- }
292
- const { icon, permissionCode, extraIcon, frameSrc } = v;
293
- v.meta = { keepAlive: true, rank: v.sort, title: `${v.i18nField}` };
294
- Object.assign(v.meta, { icon, permissionCode, extraIcon, frameSrc });
295
- menuI18n.zh[v.i18nField] = v.menuName || v.menuNameEn;
296
- menuI18n.en[v.i18nField] = v.menuNameEn || v.menuName;
297
- v.name = v.menuCode || v.name;
298
- v.path = v.menuPath || v.path;
299
- if (!v.parentId) {
300
- v.component = Layout;
301
- } else if (v.meta?.frameSrc) {
302
- v.component = IFrame;
303
- } else if (v.menuCategory === MenuCategoty.Lowcode) {
304
- // 低开路由处理
305
- const index = modulesRoutesKeys.findIndex(ev => ev.includes('lowcode-contain'));
306
- v.component = createComponent(
307
- v.name,
308
- defineAsyncComponent(() => import(/* @vite-ignore */ modulesRoutesKeys[index]))
309
- );
310
- // 将模块类型赋值到meta中
311
- Object.assign(v.meta, { moduleType: v.moduleType });
312
- } else if (v.menuCategory !== MenuCategoty.Menu) {
313
- const index = modulesRoutesKeys.findIndex(ev => ev.includes(v.path));
314
- v.component = modulesRoutes[modulesRoutesKeys[index]];
315
- }
316
- if (v?.children && v.children.length) {
317
- addAsyncRoutes(v.children);
318
- } else {
319
- delete v.children;
320
- }
321
- }
322
- return arrRoutes;
323
- };
324
-
325
- // 是否有权限
326
- function hasPermissions(value: Array<string>): boolean {
327
- if (value && value instanceof Array && value.length > 0) {
328
- const roles = usePermissionStoreHook().buttonAuth;
329
- const permissionRoles = value;
330
-
331
- const hasPermission = roles.some(role => {
332
- return permissionRoles.includes(role);
333
- });
334
-
335
- if (!hasPermission) {
336
- return false;
337
- }
338
- return true;
339
- } else {
340
- return false;
341
- }
342
- }
343
- // 删除当前路由
344
- function delCurrentRoute(current: any, toNext = true) {
345
- const startIndex: number = useMultiTagsStoreHook().multiTags.findIndex(tag => {
346
- if (current.query) {
347
- if (current.path === tag.path) {
348
- return current.query === tag.query;
349
- }
350
- } else {
351
- return current.path === tag.path;
352
- }
353
- });
354
- useMultiTagsStoreHook().handleTags('splice', '', { startIndex, length: 1 });
355
- handleAliveRoute(current.matched, 'delete');
356
- if (toNext) {
357
- const newRoute = useMultiTagsStoreHook().handleTags('slice');
358
- router.push({
359
- path: newRoute[0].path,
360
- query: newRoute[0].query
361
- });
362
- }
363
- }
364
-
365
- export {
366
- ascending,
367
- filterTree,
368
- initRouter,
369
- hasPermissions,
370
- addAsyncRoutes,
371
- getParentPaths,
372
- findRouteByPath,
373
- handleAliveRoute,
374
- formatTwoStageRoutes,
375
- formatFlatteningRoutes,
376
- delCurrentRoute
377
- };
1
+ import type { RouteRecordRaw, RouteComponent } from 'vue-router';
2
+ import { isProxy, toRaw, defineAsyncComponent } from 'vue';
3
+ import { useTimeoutFn } from '@vueuse/core';
4
+ import { storageLocal, buildHierarchyTree, dbstorage } from '@utogether/utils';
5
+ import { clone } from 'xe-utils';
6
+ import { router, getViews } from './index';
7
+ import { getConfig } from '../config';
8
+ import { geti18n } from '../plugins/i18n';
9
+ import { routerArrays } from '../layout/types';
10
+ import { usePermissionStoreHook } from '../store/modules/permission';
11
+ import { useSystemStoreHook } from '../store/modules/system';
12
+ import { useMultiTagsStoreHook } from '../store/modules/multiTags';
13
+ import { getServiceApi } from '../api';
14
+ import createComponent from './createComponent';
15
+
16
+ const Layout = () => import('../layout/layoutView.vue');
17
+ const IFrame = () => import('../layout/frameView.vue');
18
+
19
+ // 动态路由
20
+ import { initDict, initSystemInfo, getAsyncRoutes } from '../api/user';
21
+ // https://cn.vitejs.dev/guide/features.html#glob-import
22
+ // const modulesRoutes = import.meta.glob('/src/views/**/*.{vue,tsx}');
23
+
24
+ // 菜单类型
25
+ enum MenuCategoty {
26
+ Menu = '5', // 菜单
27
+ Lowcode = '10' // 低代码
28
+ }
29
+
30
+ /** 按照路由中meta下的rank等级升序来排序路由 */
31
+ function ascending(arr: any[]) {
32
+ arr.forEach(v => {
33
+ if (v?.meta?.rank === null) v.meta.rank = undefined;
34
+ if (v?.meta?.rank === 0) {
35
+ if (v.name !== 'home' && v.path !== '/') {
36
+ console.warn('rank only the home page can be 0');
37
+ }
38
+ }
39
+ });
40
+ return arr.sort((a: { meta: { rank: number } }, b: { meta: { rank: number } }) => {
41
+ return a?.meta?.rank - b?.meta?.rank;
42
+ });
43
+ }
44
+
45
+ /** 过滤meta中showLink为false的菜单 */
46
+ function filterTree(data: RouteComponent[]) {
47
+ const newTree = clone(data, true).filter((v: { meta: { showLink: boolean } }) => v && v.meta?.showLink !== false);
48
+ newTree.forEach((v: { children }) => v.children && (v.children = filterTree(v.children)));
49
+ return newTree;
50
+ }
51
+
52
+ /** 通过指定 `key` 获取父级路径集合,默认 `key` 为 `path` */
53
+ function getParentPaths(value: string, routes: RouteRecordRaw[], key = 'path') {
54
+ // 深度遍历查找
55
+ function dfs(routes: RouteRecordRaw[], value: string, parents: string[]) {
56
+ for (let i = 0; i < routes.length; i++) {
57
+ const item = routes[i];
58
+ // 找到path则返回父级path
59
+ // if (item[key] === value) {
60
+ // return parents.length ? parents : routes[0];
61
+ // }
62
+ if (item[key] === value) return parents;
63
+ // children不存在或为空则不递归
64
+ if (!item.children || !item.children.length) continue;
65
+ // 往下查找时将当前path入栈
66
+ parents.push(item.path);
67
+ // @ts-ignore
68
+ if (dfs(item.children, value, parents).length) return parents;
69
+ // 深度遍历查找未找到时当前path 出栈
70
+ parents.pop();
71
+ }
72
+ // 未找到时返回空数组
73
+ return [];
74
+ }
75
+
76
+ return dfs(routes, value, []);
77
+ }
78
+
79
+ /** 查找对应 `path` 的路由信息 */
80
+ function findRouteByPath(path: string, routes: RouteRecordRaw[]) {
81
+ let res = routes.find((item: { path: string }) => item.path == path);
82
+ if (res) {
83
+ return isProxy(res) ? toRaw(res) : res;
84
+ } else {
85
+ for (let i = 0; i < routes.length; i++) {
86
+ if (routes[i].children instanceof Array && routes[i].children.length > 0) {
87
+ res = findRouteByPath(path, routes[i].children);
88
+ if (res) {
89
+ return isProxy(res) ? toRaw(res) : res;
90
+ }
91
+ }
92
+ }
93
+ return null;
94
+ }
95
+ }
96
+ function addPathMatch() {
97
+ if (!router.hasRoute('pathMatch')) {
98
+ router.addRoute({
99
+ path: '/:pathMatch(.*)',
100
+ name: 'pathMatch',
101
+ redirect: '/error/404'
102
+ });
103
+ }
104
+ }
105
+
106
+ /**
107
+ * @description: 添加审批动态路由
108
+ * @param {*} syncRoutes 异步获取的路由表数据
109
+ * @param {*} routes 格式化后的路由表数据
110
+ * @return {*} 格式化后的路由表数据
111
+ */
112
+ // function addWorkflowRouter(syncRoutes, routes) {
113
+ // const flowIdx = syncRoutes.findIndex(f => f.menuCode === 'Workflow');
114
+ // const isExit = flowIdx !== -1 && syncRoutes[flowIdx].children?.some(s => s.name === 'workflowApprove');
115
+ // if (!isExit) {
116
+ // flowIdx !== -1 && routes[flowIdx].children?.push(workflowRouter);
117
+ // }
118
+ // return routes;
119
+ // }
120
+ // 菜单国家化
121
+ function setMenui18n() {
122
+ const i18n = geti18n();
123
+ i18n.global?.mergeLocaleMessage('zh', menuI18n.zh);
124
+ i18n.global?.mergeLocaleMessage('en', menuI18n.en);
125
+ storageLocal.setItem('menu_zh', menuI18n.zh);
126
+ storageLocal.setItem('menu_en', menuI18n.en);
127
+ }
128
+
129
+ function handleAsyncRoutes(routesList) {
130
+ if (routesList?.length) {
131
+ const idx = routesList.findIndex(f => f.menuCode === 'SysHome');
132
+ if (idx !== -1) {
133
+ // 仪表盘报表
134
+ const dashboardTabs = [];
135
+ const homeMenus = [];
136
+ routesList[idx].children.forEach(child => {
137
+ if (child.menuCategory !== '1') {
138
+ dashboardTabs.push(child);
139
+ } else {
140
+ homeMenus.push(child);
141
+ }
142
+ });
143
+ routesList[idx].children = homeMenus;
144
+ useSystemStoreHook().setHomeMenus(dashboardTabs);
145
+ }
146
+ const routes = clone(addAsyncRoutes(routesList), true);
147
+
148
+ // routes = addWorkflowRouter(routesList, routes);
149
+ setMenui18n();
150
+ formatFlatteningRoutes(routes).map((v: RouteRecordRaw) => {
151
+ // 菜单国际化
152
+ // 防止重复添加路由
153
+ const { options } = router;
154
+ if (options.routes[0].children.findIndex(value => value.path === v.path) !== -1) {
155
+ return;
156
+ } else {
157
+ // 切记将路由push到routes后还需要使用addRoute,这样路由才能正常跳转
158
+ router.options.routes[0].children.push(v);
159
+ // 最终路由进行升序
160
+ ascending(router.options.routes[0].children);
161
+ if (!router.hasRoute(v?.name)) router.addRoute(v);
162
+ }
163
+ });
164
+ }
165
+ usePermissionStoreHook().changeSetting(routesList);
166
+ if (!useMultiTagsStoreHook().getMultiTagsCache) {
167
+ useMultiTagsStoreHook().handleTags('equal', [
168
+ ...routerArrays,
169
+ ...usePermissionStoreHook().flatteningRoutes.filter(v => v?.meta?.fixedTag)
170
+ ]);
171
+ }
172
+ addPathMatch();
173
+ }
174
+
175
+ /** 初始化路由(`new Promise` 写法防止在异步请求中造成无限循环)*/
176
+ async function initRouter(userName) {
177
+ let asyncRouteList = [];
178
+ const key = `U-${userName}-ROUTES`;
179
+ // 开启动态路由缓存本地
180
+ if (getConfig().CachingAsyncRoutes) {
181
+ await dbstorage.getItem(key).then(data => (asyncRouteList = data));
182
+ }
183
+ initDict().then(res => useSystemStoreHook().setDict(res));
184
+ await initSystemInfo().then(async (res: IRecord) => {
185
+ useSystemStoreHook().setSystemInfo(res);
186
+ await getServiceApi()
187
+ .get('/uums/cusOrganization', { pageSize: 100, pageNum: 1, orgId: res.orgId })
188
+ .then((res: IResponseData) => useSystemStoreHook().setInvOrgList(res?.list || []));
189
+ });
190
+ if (asyncRouteList?.length) {
191
+ return new Promise(resolve => {
192
+ handleAsyncRoutes(asyncRouteList);
193
+ // 缓存最新的路由
194
+ getAsyncRoutes({ userName }).then((data: any) => {
195
+ dbstorage.setItem(key, data);
196
+ });
197
+ resolve(router);
198
+ });
199
+ } else {
200
+ return new Promise(resolve => {
201
+ getAsyncRoutes({ userName }).then((data: any) => {
202
+ handleAsyncRoutes(clone(data, true));
203
+ dbstorage.setItem(key, data);
204
+ resolve(router);
205
+ });
206
+ });
207
+ }
208
+ }
209
+
210
+ /**
211
+ * 将多级嵌套路由处理成一维数组
212
+ * @param routesList 传入路由
213
+ * @returns 返回处理后的一维路由
214
+ */
215
+ function formatFlatteningRoutes(routesList: RouteRecordRaw[]) {
216
+ if (routesList.length === 0) return routesList;
217
+ let hierarchyList = buildHierarchyTree(routesList);
218
+ for (let i = 0; i < hierarchyList.length; i++) {
219
+ if (hierarchyList[i].children) {
220
+ hierarchyList = hierarchyList.slice(0, i + 1).concat(hierarchyList[i].children, hierarchyList.slice(i + 1));
221
+ }
222
+ }
223
+ return hierarchyList;
224
+ }
225
+
226
+ /**
227
+ * 一维数组处理成多级嵌套数组(三级及以上的路由全部拍成二级,keep-alive 只支持到二级缓存)
228
+ * @param routesList 处理后的一维路由菜单数组
229
+ * @returns 返回将一维数组重新处理成规定路由的格式
230
+ */
231
+ function formatTwoStageRoutes(routesList: RouteRecordRaw[]) {
232
+ if (routesList.length === 0) return routesList;
233
+ const newRoutesList: RouteRecordRaw[] = [];
234
+ routesList.forEach((v: RouteRecordRaw) => {
235
+ if (v.path === '/') {
236
+ newRoutesList.push({
237
+ component: v.component,
238
+ name: v.name,
239
+ path: v.path,
240
+ redirect: v.redirect,
241
+ meta: v.meta,
242
+ children: []
243
+ });
244
+ } else {
245
+ newRoutesList[0].children.push({ ...v });
246
+ }
247
+ });
248
+ return newRoutesList;
249
+ }
250
+
251
+ /** 处理缓存路由(添加、删除、刷新) */
252
+ function handleAliveRoute({ name }: ToRouteType, mode?: string) {
253
+ switch (mode) {
254
+ case 'add':
255
+ usePermissionStoreHook().cacheOperate({ mode: 'add', name });
256
+ break;
257
+ case 'delete':
258
+ usePermissionStoreHook().cacheOperate({ mode: 'delete', name });
259
+ break;
260
+ case 'refresh':
261
+ usePermissionStoreHook().cacheOperate({ mode: 'refresh', name });
262
+ break;
263
+ default:
264
+ usePermissionStoreHook().cacheOperate({ mode: 'delete', name });
265
+ useTimeoutFn(() => {
266
+ usePermissionStoreHook().cacheOperate({ mode: 'add', name });
267
+ }, 100);
268
+ }
269
+ }
270
+
271
+ /** 过滤后端传来的动态路由 重新生成规范路由 */
272
+ const menuI18n = {
273
+ zh: {},
274
+ en: {}
275
+ };
276
+ const addAsyncRoutes = (arrRoutes: Array<any>) => {
277
+ const modulesRoutes = getViews();
278
+
279
+ if (!arrRoutes || !arrRoutes.length) return;
280
+ const modulesRoutesKeys = Object.keys(modulesRoutes);
281
+ for (let i = 0; i < arrRoutes.length; i++) {
282
+ const v = arrRoutes[i];
283
+
284
+ if (v?.children && v.children.length && !v.redirect) {
285
+ // 父级的redirect属性取值:如果子级存在且父级的redirect属性不存在,默认取第一个子级的path;如果子级存在且父级的redirect属性存在,取存在的redirect属性,会覆盖默认值
286
+ v.redirect = v.children[0].path;
287
+ }
288
+ const { icon, permissionCode, extraIcon, frameSrc } = v;
289
+ v.meta = { keepAlive: true, rank: v.sort, title: `${v.i18nField}` };
290
+ Object.assign(v.meta, { icon, permissionCode, extraIcon, frameSrc });
291
+ menuI18n.zh[v.i18nField] = v.menuName || v.menuNameEn;
292
+ menuI18n.en[v.i18nField] = v.menuNameEn || v.menuName;
293
+ v.name = v.menuCode || v.name;
294
+ v.path = v.menuPath || v.path;
295
+ if (!v.parentId) {
296
+ v.component = Layout;
297
+ } else if (v.meta?.frameSrc) {
298
+ v.component = IFrame;
299
+ } else if (v.menuCategory === MenuCategoty.Lowcode) {
300
+ // 低开路由处理
301
+ const index = modulesRoutesKeys.findIndex(ev => ev.includes('lowcode-contain'));
302
+ v.component = createComponent(
303
+ v.name,
304
+ defineAsyncComponent(() => import(/* @vite-ignore */ modulesRoutesKeys[index]))
305
+ );
306
+ // 将模块类型赋值到meta中
307
+ Object.assign(v.meta, { moduleType: v.moduleType });
308
+ } else if (v.menuCategory !== MenuCategoty.Menu) {
309
+ const index = modulesRoutesKeys.findIndex(ev => ev.includes(v.path));
310
+ v.component = modulesRoutes[modulesRoutesKeys[index]];
311
+ }
312
+ if (v?.children && v.children.length) {
313
+ addAsyncRoutes(v.children);
314
+ } else {
315
+ delete v.children;
316
+ }
317
+ }
318
+ return arrRoutes;
319
+ };
320
+
321
+ // 是否有权限
322
+ function hasPermissions(value: Array<string>): boolean {
323
+ if (value && value instanceof Array && value.length > 0) {
324
+ const roles = usePermissionStoreHook().buttonAuth;
325
+ const permissionRoles = value;
326
+
327
+ const hasPermission = roles.some(role => {
328
+ return permissionRoles.includes(role);
329
+ });
330
+
331
+ if (!hasPermission) {
332
+ return false;
333
+ }
334
+ return true;
335
+ } else {
336
+ return false;
337
+ }
338
+ }
339
+ // 删除当前路由
340
+ function delCurrentRoute(current: any, toNext = true) {
341
+ const startIndex: number = useMultiTagsStoreHook().multiTags.findIndex(tag => {
342
+ if (current.query) {
343
+ if (current.path === tag.path) {
344
+ return current.query === tag.query;
345
+ }
346
+ } else {
347
+ return current.path === tag.path;
348
+ }
349
+ });
350
+ useMultiTagsStoreHook().handleTags('splice', '', { startIndex, length: 1 });
351
+ handleAliveRoute(current.matched, 'delete');
352
+ if (toNext) {
353
+ const newRoute = useMultiTagsStoreHook().handleTags('slice');
354
+ router.push({
355
+ path: newRoute[0].path,
356
+ query: newRoute[0].query
357
+ });
358
+ }
359
+ }
360
+
361
+ function handleTopMenu(route) {
362
+ if (route?.children && route.children.length > 1) {
363
+ if (route.redirect) {
364
+ return route.children.filter(cur => cur.path === route.redirect)[0];
365
+ } else {
366
+ return route.children[0];
367
+ }
368
+ } else {
369
+ return route;
370
+ }
371
+ }
372
+
373
+ /** 获取所有菜单中的第一个菜单(顶级菜单)*/
374
+ function getTopMenu(tag = false) {
375
+ const topMenu = handleTopMenu(usePermissionStoreHook().wholeMenus[0]?.children[0]);
376
+ tag && useMultiTagsStoreHook().handleTags('push', topMenu);
377
+ return topMenu;
378
+ }
379
+
380
+ export {
381
+ ascending,
382
+ filterTree,
383
+ initRouter,
384
+ getTopMenu,
385
+ hasPermissions,
386
+ addAsyncRoutes,
387
+ getParentPaths,
388
+ findRouteByPath,
389
+ handleAliveRoute,
390
+ formatTwoStageRoutes,
391
+ formatFlatteningRoutes,
392
+ delCurrentRoute
393
+ };