@utogether/udp-core 1.0.1-beta.15 → 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 (126) hide show
  1. package/build/plugins.ts +39 -39
  2. package/dist/{403-DpJevH-h.js → 403-C3cfjZIT.js} +1 -1
  3. package/dist/{404-BpqJK3YD.js → 404-Bg3j7QIo.js} +1 -1
  4. package/dist/{500-DyvRbMD_.js → 500-CMT7Zyy7.js} +1 -1
  5. package/dist/{AuthorityInfo-Cyu4XHXx.js → AuthorityInfo-B1mvqs4x.js} +1 -1
  6. package/dist/{AuthorityInfo.vue_vue_type_style_index_0_lang-BxmnEEFq.js → AuthorityInfo.vue_vue_type_style_index_0_lang-DwoW7QWO.js} +3 -3
  7. package/dist/{Company-BLygcYrI.js → Company-qTguidK8.js} +3 -3
  8. package/dist/{CompanyPanel-D5Tgw6LF.js → CompanyPanel-BBC-o-k3.js} +17 -17
  9. package/dist/{Department-qI9OninZ.js → Department-C3mKZOnK.js} +3 -3
  10. package/dist/{DepartmentPanel-P-P8fPZQ.js → DepartmentPanel-CNdpOHBT.js} +25 -25
  11. package/dist/{DesignPanel-BxjD0AZi.js → DesignPanel-D9jrRCoG.js} +1 -1
  12. package/dist/{DesignPanel.vue_vue_type_style_index_0_lang-DN_OKxCu.js → DesignPanel.vue_vue_type_style_index_0_lang-A-Aj1Hl0.js} +6 -6
  13. package/dist/{DictView-PWL_Vf0T.js → DictView-CgdVkSiP.js} +15 -16
  14. package/dist/{InvOrganization-JXBXyn3G.js → InvOrganization-RE7s3vBq.js} +1 -1
  15. package/dist/{Org-DcTJn6YQ.js → Org-wXDGvTrS.js} +3 -3
  16. package/dist/{Preview-IKvbHI8a.js → Preview-Pvt1__BV.js} +1 -1
  17. package/dist/{ReportDefine-_G7ts7Ch.js → ReportDefine-xEQsgFyv.js} +1 -1
  18. package/dist/{ReportDesign-7h5oFklh.js → ReportDesign-jOSuEFH9.js} +9 -9
  19. package/dist/{ReportQuery-dj0BQUyi.js → ReportQuery-DxEmmGiN.js} +1 -1
  20. package/dist/{ReportQueryFrom-6WaZPg9i.js → ReportQueryFrom-CFlxfS5M.js} +1 -1
  21. package/dist/{ReportQueryFrom.vue_vue_type_style_index_0_lang-ChhcBPqa.js → ReportQueryFrom.vue_vue_type_style_index_0_lang-iHpiYz8f.js} +1 -1
  22. package/dist/{ReportTemplate-DIvJ3RrB.js → ReportTemplate-Dtz3aet7.js} +11 -11
  23. package/dist/{Role-ia_3qAZU.js → Role-B_NsHIyE.js} +3 -3
  24. package/dist/{RoleAssign-D0zVl4y3.js → RoleAssign-C9wm4zq4.js} +8 -8
  25. package/dist/{RolePanel-CQxOiAzj.js → RolePanel-BlHidBR0.js} +1 -1
  26. package/dist/{RolePanel-BiGGmP2w.js → RolePanel-DWM9mfNs.js} +1 -1
  27. package/dist/{RolePanel.vue_vue_type_script_setup_true_lang-BYDLRS_x.js → RolePanel.vue_vue_type_script_setup_true_lang-Cn4RRzXm.js} +1 -1
  28. package/dist/{RolePanel.vue_vue_type_script_setup_true_lang-Dwv5LFbf.js → RolePanel.vue_vue_type_script_setup_true_lang-DVQ9aqb1.js} +9 -9
  29. package/dist/{ScrollPanel.vue_vue_type_style_index_0_lang-gPP4duO8.js → ScrollPanel.vue_vue_type_style_index_0_lang-CCm-_i8O.js} +8 -8
  30. package/dist/{Staff-Bf_oUqF6.js → Staff-BfDc61QS.js} +3 -3
  31. package/dist/{StaffInfo-D5aisjx2.js → StaffInfo-BaJFo4mg.js} +1 -1
  32. package/dist/{StaffInfo.vue_vue_type_script_setup_true_lang-6PX7Bx6A.js → StaffInfo.vue_vue_type_script_setup_true_lang-Bwj4x1wu.js} +7 -7
  33. package/dist/{StaffPanel--zKzd4rt.js → StaffPanel-B6I2ZS6A.js} +1 -1
  34. package/dist/{StaffPanel.vue_vue_type_script_setup_true_lang-BWZSIFpq.js → StaffPanel.vue_vue_type_script_setup_true_lang-vbyS4w3V.js} +2 -2
  35. package/dist/{SysUser-BelbESLR.js → SysUser-B_r9LYc5.js} +2 -2
  36. package/dist/{SysUserPanel-BixKqQcA.js → SysUserPanel-C4ajcHPV.js} +1 -1
  37. package/dist/{SysUserPanel.vue_vue_type_script_setup_true_lang-BhaWZEfo.js → SysUserPanel.vue_vue_type_script_setup_true_lang-8Q0VvC9B.js} +41 -37
  38. package/dist/{SystemMenu-D40tzcSj.js → SystemMenu-CyPG23uP.js} +10 -10
  39. package/dist/{UserInfo-DTyk071j.js → UserInfo-BkoGt4ib.js} +1 -1
  40. package/dist/{UserInfo.vue_vue_type_style_index_0_lang-p9MByikC.js → UserInfo.vue_vue_type_style_index_0_lang-DrzF1YYG.js} +10 -10
  41. package/dist/{childView-so5qSxr0.js → childView-9vEO8ymO.js} +1 -1
  42. package/dist/{childView-xL3iE1yN.js → childView-Dh0NyiQe.js} +1 -1
  43. package/dist/{childView.vue_vue_type_style_index_0_lang-CKcXpSqa.js → childView.vue_vue_type_style_index_0_lang-BOeoZuAK.js} +7 -7
  44. package/dist/{childView.vue_vue_type_style_index_0_lang-DK1ei5uM.js → childView.vue_vue_type_style_index_0_lang-CMOjfXBX.js} +1 -1
  45. package/dist/{code-rule-DC35y76w.js → code-rule-C7jI_1gJ.js} +6 -6
  46. package/dist/core.es.js +17 -12
  47. package/dist/{cron-task-BhjQ4S68.js → cron-task-Dg9DJXvv.js} +11 -11
  48. package/dist/{frameView-G_zhG9pf.js → frameView-CeR-_hjB.js} +1 -1
  49. package/dist/{index-C6ehxLLO.js → index-Bb7gIEvu.js} +739 -496
  50. package/dist/{layoutView-CO2U8qCQ.js → layoutView-wsLHrEeX.js} +1347 -1346
  51. package/dist/login-CluzzdqH.js +249 -0
  52. package/dist/{lov-view-Cr929NMD.js → lov-view-BvQCvwbW.js} +5 -5
  53. package/dist/{menuInfo-BWuROp7u.js → menuInfo-B8RLelIC.js} +1 -1
  54. package/dist/{menuInfo.vue_vue_type_style_index_0_lang-9nouHUjO.js → menuInfo.vue_vue_type_style_index_0_lang-DvkapEZr.js} +7 -7
  55. package/dist/{pda-app-CQP1sMAI.js → pda-app-D3LNmD9a.js} +19 -19
  56. package/dist/{resource-BDQLrMpK.js → resource-D8JA_03-.js} +7 -7
  57. package/dist/{su-welcome-CZysrZQ1.js → su-welcome-BjxeEKSo.js} +88 -88
  58. package/dist/{sys-config-BiGNMYSA.js → sys-config-sbeLk9bP.js} +6 -6
  59. package/dist/udp-core.css +1 -9
  60. package/dist/utogether-MlnyYtNS.js +4 -0
  61. package/index.ts +48 -40
  62. package/package.json +1 -1
  63. package/src/App.vue +65 -65
  64. package/src/components/SuCharts/src/UserInfo.vue +78 -78
  65. package/src/components/SuScrollTree/ScrollPanel.vue +170 -175
  66. package/src/components/udp/{grid.vue → grid/index.vue} +27 -24
  67. package/src/components/udp/index.ts +2 -9
  68. package/src/components/udp/utils.ts +60 -107
  69. package/src/layout/components/lay-navbar/index.vue +239 -239
  70. package/src/layout/components/lay-panel/index.vue +150 -150
  71. package/src/layout/components/lay-search/index.vue +25 -25
  72. package/src/layout/components/lay-select-org/index.vue +64 -64
  73. package/src/layout/components/lay-sidebar/horizontal.vue +190 -190
  74. package/src/layout/components/lay-sidebar/mixNav.vue +260 -260
  75. package/src/layout/components/lay-sidebar/sidebar-logo.vue +92 -92
  76. package/src/layout/components/lay-tag/index.vue +20 -48
  77. package/src/layout/hooks/useNav.ts +173 -173
  78. package/src/layout/hooks/useTag.ts +223 -233
  79. package/src/layout/types.ts +3 -2
  80. package/src/main.ts +117 -113
  81. package/src/plugins/i18n/en.ts +302 -302
  82. package/src/plugins/i18n/zh.ts +356 -354
  83. package/src/plugins/vxe-table/index.ts +116 -53
  84. package/src/plugins/vxe-table/render.tsx +956 -956
  85. package/src/router/index.ts +186 -177
  86. package/src/router/modules/remaining.ts +58 -58
  87. package/src/router/utils.ts +393 -377
  88. package/src/store/modules/app.ts +1 -3
  89. package/src/store/modules/multiTags.ts +110 -109
  90. package/src/store/modules/permission.ts +113 -100
  91. package/src/style/button.scss +85 -85
  92. package/src/style/login.css +1 -1
  93. package/src/style/vxetable.scss +25 -2
  94. package/src/utils/index.ts +3 -1
  95. package/src/views/organization/company/CompanyPanel.vue +259 -259
  96. package/src/views/organization/department/DepartmentPanel.vue +269 -283
  97. package/src/views/system/menu/SystemMenu.vue +183 -183
  98. package/src/views/system/menu/menuInfo.vue +363 -363
  99. package/src/views/system/role/UserInfo.vue +195 -195
  100. package/src/views/system/role-assign/RoleAssign.vue +57 -57
  101. package/src/views/system/role-assign/RolePanel.vue +139 -139
  102. package/src/views/system/sys/sys-config.vue +336 -336
  103. package/src/views/system/sysUser/SysUserPanel.vue +279 -278
  104. package/src/views/uapp/pda/pda-app.vue +208 -208
  105. package/src/views/udev/dict/DictView.vue +118 -118
  106. package/src/views/udev/dict/childView.vue +1 -1
  107. package/src/views/udev/lov/lov-view.vue +91 -91
  108. package/src/views/ulogin/login.vue +2 -2
  109. package/src/views/upms/interface/log-out.vue +101 -101
  110. package/src/views/urpt/design/DesignPanel.vue +507 -507
  111. package/types/global.d.ts +2 -8
  112. package/dist/login-DzyK2soP.js +0 -278
  113. package/dist/utogether-BRirriOz.js +0 -182
  114. package/src/components/udp/count-down.vue +0 -536
  115. package/src/components/udp/flip-down/FlipCard/flip-card.vue +0 -251
  116. package/src/components/udp/flip-down/FlipCard/interfaces.ts +0 -4
  117. package/src/components/udp/flip-down/FlipClock/flip-clock.vue +0 -113
  118. package/src/components/udp/form-upload.vue +0 -482
  119. package/src/components/udp/form.vue +0 -112
  120. package/src/components/udp/lov.vue +0 -388
  121. package/src/components/udp/modal-form.vue +0 -190
  122. package/src/components/udp/modal-grid.vue +0 -298
  123. package/src/components/udp/upload.vue +0 -423
  124. package/src/utils/udp/http/index.ts +0 -294
  125. package/src/utils/udp/http/types.d.ts +0 -49
  126. 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
+ };