xiaoxian521
2 years ago
69 changed files with 1355 additions and 1043 deletions
-
2.env
-
4.env.development
-
7.env.production
-
5.env.staging
-
4README.md
-
5build/index.ts
-
26build/optimize.ts
-
9build/plugins.ts
-
41package.json
-
1195pnpm-lock.yaml
-
5public/serverConfig.json
-
BINsrc/assets/avatars.jpg
-
2src/assets/login/avatar.svg
-
2src/assets/login/illustration.svg
-
2src/assets/status/403.svg
-
2src/assets/status/404.svg
-
2src/assets/status/500.svg
-
2src/assets/svg/back_top.svg
-
2src/assets/svg/dark.svg
-
2src/assets/svg/day.svg
-
2src/assets/svg/enter_outlined.svg
-
2src/assets/svg/exit_screen.svg
-
2src/assets/svg/full_screen.svg
-
2src/assets/svg/globalization.svg
-
2src/assets/svg/keyboard_esc.svg
-
8src/components/ReIcon/src/iconifyIconOffline.ts
-
3src/config/index.ts
-
65src/layout/components/notice/index.vue
-
2src/layout/components/panel/index.vue
-
5src/layout/components/search/components/SearchFooter.vue
-
4src/layout/components/search/components/SearchModal.vue
-
2src/layout/components/search/components/SearchResult.vue
-
2src/layout/components/search/index.vue
-
7src/layout/components/setting/index.vue
-
2src/layout/components/sidebar/breadCrumb.vue
-
15src/layout/components/sidebar/sidebarItem.vue
-
1src/layout/components/sidebar/vertical.vue
-
2src/layout/frameView.vue
-
4src/layout/hooks/useNav.ts
-
18src/layout/hooks/useTag.ts
-
3src/main.ts
-
1src/plugins/i18n.ts
-
29src/router/index.ts
-
7src/router/modules/error.ts
-
7src/router/modules/home.ts
-
9src/router/modules/remaining.ts
-
9src/router/types.ts
-
4src/router/utils.ts
-
1src/store/modules/app.ts
-
1src/store/modules/epTheme.ts
-
7src/store/modules/multiTags.ts
-
16src/store/modules/user.ts
-
7src/style/dark.scss
-
3src/style/index.scss
-
10src/style/reset.scss
-
3src/style/transition.scss
-
8src/utils/auth.ts
-
7src/utils/globalPolyfills.ts
-
33src/utils/message.ts
-
59src/utils/sso.ts
-
188src/utils/tree.ts
-
4src/views/login/index.vue
-
2tsconfig.json
-
269types/global.d.ts
-
85types/index.d.ts
-
134types/index.ts
-
6types/shims-tsx.d.ts
-
2types/shims-vue.d.ts
-
15vite.config.ts
@ -1,2 +1,2 @@ |
|||
# 项目本地运行端口号 |
|||
# 平台本地运行端口号 |
|||
VITE_PORT = 8848 |
@ -1,8 +1,8 @@ |
|||
# 项目本地运行端口号 |
|||
# 平台本地运行端口号 |
|||
VITE_PORT = 8848 |
|||
|
|||
# 开发环境读取配置文件路径 |
|||
VITE_PUBLIC_PATH = / |
|||
|
|||
# 开发环境路由历史模式 |
|||
# 开发环境路由历史模式(Hash模式传"hash"、HTML5模式传"h5"、Hash模式带base参数传"hash,base参数"、HTML5模式带base参数传"h5,base参数") |
|||
VITE_ROUTER_HISTORY = "hash" |
@ -0,0 +1,26 @@ |
|||
/** |
|||
* 此文件作用于 `vite.config.ts` 的 `optimizeDeps.include` 依赖预构建配置项 |
|||
* 依赖预构建,`vite` 启动时会将下面 include 里的模块,编译成 esm 格式并缓存到 node_modules/.vite 文件夹,页面加载到对应模块时如果浏览器有缓存就读取浏览器缓存,如果没有会读取本地缓存并按需加载 |
|||
* 尤其当您禁用浏览器缓存时(这种情况只应该发生在调试阶段)必须将对应模块加入到 include里,否则会遇到开发环境切换页面卡顿的问题(vite 会认为它是一个新的依赖包会重新加载并强制刷新页面),因为它既无法使用浏览器缓存,又没有在本地 node_modules/.vite 里缓存 |
|||
* 温馨提示:如果您使用的第三方库是全局引入,也就是引入到 src/main.ts 文件里,就不需要再添加到 include 里了,因为 vite 会自动将它们缓存到 node_modules/.vite |
|||
*/ |
|||
export const include = [ |
|||
"qs", |
|||
"mitt", |
|||
"dayjs", |
|||
"axios", |
|||
"pinia", |
|||
"lodash", |
|||
"echarts", |
|||
"vue-i18n", |
|||
"xe-utils", |
|||
"vxe-table", |
|||
"js-cookie", |
|||
"lodash-es", |
|||
"@vueuse/core", |
|||
"lodash-unified", |
|||
"@ctrl/tinycolor", |
|||
"@pureadmin/utils", |
|||
"responsive-storage", |
|||
"element-resize-detector" |
|||
]; |
1195
pnpm-lock.yaml
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
Before Width: 400 | Height: 400 | Size: 23 KiB |
@ -1 +1 @@ |
|||
<svg t="1636193306629" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1847" width="32" height="32"><path d="M410.558481 0.10861C410.558481 211.083075 109.682285 361.860579 109.682285 633.656511c0 174.943176 134.703259 316.787527 300.876196 316.787527s300.876197-141.817198 300.876197-316.787527C711.407525 361.751969 410.558481 210.974465 410.558481 0.10861z" fill="#386BF3" p-id="1848"></path><path d="M613.468671 73.664572c0 211.055922-300.876197 361.914883-300.876196 633.547901 0 174.943176 134.703259 316.787527 300.876196 316.787527s300.876197-141.817198 300.876197-316.787527c-0.054305-271.633018-300.876197-422.491979-300.876197-633.547901z" fill="#C3D2FB" p-id="1849"></path><path d="M312.592475 707.212473c0-183.713414 137.635722-312.171612 226.72288-441.390078 81.701694 106.111739 172.119322 218.740063 172.119323 367.725506a309.755045 309.755045 0 0 1-291.074166 316.516003 323.114046 323.114046 0 0 1-107.768037-242.851431z" fill="#303F5B" p-id="1850"></path></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32" class="icon" viewBox="0 0 1024 1024"><path fill="#386BF3" d="M410.558.109c0 210.974-300.876 361.752-300.876 633.548 0 174.943 134.704 316.787 300.876 316.787s300.877-141.817 300.877-316.787C711.408 361.752 410.558 210.974 410.558.109z"/><path fill="#C3D2FB" d="M613.469 73.665c0 211.055-300.877 361.914-300.877 633.547C312.592 882.156 447.296 1024 613.47 1024s300.876-141.817 300.876-316.788C914.29 435.58 613.469 284.72 613.469 73.665z"/><path fill="#303F5B" d="M312.592 707.212c0-183.713 137.636-312.171 226.723-441.39 81.702 106.112 172.12 218.74 172.12 367.726A309.755 309.755 0 0 1 420.36 950.064a323.114 323.114 0 0 1-107.769-242.852z"/></svg> |
2
src/assets/login/illustration.svg
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
2
src/assets/status/403.svg
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
2
src/assets/status/404.svg
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
2
src/assets/status/500.svg
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="24" height="24"><path fill="none" d="M0 0h24v24H0z"/><path d="M2.88 18.054a35.897 35.897 0 0 1 8.531-16.32.8.8 0 0 1 1.178 0c.166.18.304.332.413.455a35.897 35.897 0 0 1 8.118 15.865c-2.141.451-4.34.747-6.584.874l-2.089 4.178a.5.5 0 0 1-.894 0l-2.089-4.178a44.019 44.019 0 0 1-6.584-.874zm6.698-1.123l1.157.066L12 19.527l1.265-2.53 1.157-.066a42.137 42.137 0 0 0 4.227-.454A33.913 33.913 0 0 0 12 4.09a33.913 33.913 0 0 0-6.649 12.387c1.395.222 2.805.374 4.227.454zM12 15a3 3 0 1 1 0-6 3 3 0 0 1 0 6zm0-2a1 1 0 1 0 0-2 1 1 0 0 0 0 2z"/></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24"><path fill="none" d="M0 0h24v24H0z"/><path d="M2.88 18.054a35.897 35.897 0 0 1 8.531-16.32.8.8 0 0 1 1.178 0c.166.18.304.332.413.455a35.897 35.897 0 0 1 8.118 15.865c-2.141.451-4.34.747-6.584.874l-2.089 4.178a.5.5 0 0 1-.894 0l-2.089-4.178a44.019 44.019 0 0 1-6.584-.874zm6.698-1.123 1.157.066L12 19.527l1.265-2.53 1.157-.066a42.137 42.137 0 0 0 4.227-.454A33.913 33.913 0 0 0 12 4.09a33.913 33.913 0 0 0-6.649 12.387c1.395.222 2.805.374 4.227.454zM12 15a3 3 0 1 1 0-6 3 3 0 0 1 0 6zm0-2a1 1 0 1 0 0-2 1 1 0 0 0 0 2z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="16" height="16"><path fill="none" d="M0 0h24v24H0z"/><path d="M11.38 2.019a7.5 7.5 0 1 0 10.6 10.6C21.662 17.854 17.316 22 12.001 22 6.477 22 2 17.523 2 12c0-5.315 4.146-9.661 9.38-9.981z"/></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M11.38 2.019a7.5 7.5 0 1 0 10.6 10.6C21.662 17.854 17.316 22 12.001 22 6.477 22 2 17.523 2 12c0-5.315 4.146-9.661 9.38-9.981z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="16" height="16"><path fill="none" d="M0 0h24v24H0z"/><path d="M12 18a6 6 0 1 1 0-12 6 6 0 0 1 0 12zM11 1h2v3h-2V1zm0 19h2v3h-2v-3zM3.515 4.929l1.414-1.414L7.05 5.636 5.636 7.05 3.515 4.93zM16.95 18.364l1.414-1.414 2.121 2.121-1.414 1.414-2.121-2.121zm2.121-14.85l1.414 1.415-2.121 2.121-1.414-1.414 2.121-2.121zM5.636 16.95l1.414 1.414-2.121 2.121-1.414-1.414 2.121-2.121zM23 11v2h-3v-2h3zM4 11v2H1v-2h3z"/></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24"><path fill="none" d="M0 0h24v24H0z"/><path d="M12 18a6 6 0 1 1 0-12 6 6 0 0 1 0 12zM11 1h2v3h-2V1zm0 19h2v3h-2v-3zM3.515 4.929l1.414-1.414L7.05 5.636 5.636 7.05 3.515 4.93zM16.95 18.364l1.414-1.414 2.121 2.121-1.414 1.414-2.121-2.121zm2.121-14.85 1.414 1.415-2.121 2.121-1.414-1.414 2.121-2.121zM5.636 16.95l1.414 1.414-2.121 2.121-1.414-1.414 2.121-2.121zM23 11v2h-3v-2h3zM4 11v2H1v-2h3z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--ant-design" width="20" height="20" preserveAspectRatio="xMidYMid meet" viewBox="0 0 1024 1024"><path fill="currentColor" d="M864 170h-60c-4.4 0-8 3.6-8 8v518H310v-73c0-6.7-7.8-10.5-13-6.3l-141.9 112a8 8 0 0 0 0 12.6l141.9 112c5.3 4.2 13 .4 13-6.3v-75h498c35.3 0 64-28.7 64-64V178c0-4.4-3.6-8-8-8z"></path></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" aria-hidden="true" class="iconify iconify--ant-design" viewBox="0 0 1024 1024"><path fill="currentColor" d="M864 170h-60c-4.4 0-8 3.6-8 8v518H310v-73c0-6.7-7.8-10.5-13-6.3l-141.9 112a8 8 0 0 0 0 12.6l141.9 112c5.3 4.2 13 .4 13-6.3v-75h498c35.3 0 64-28.7 64-64V178c0-4.4-3.6-8-8-8z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" width="1em" height="1em" class="re-screen" color="#00000073" preserveAspectRatio="xMidYMid meet" viewBox="0 0 16 16"><g fill="currentColor"><path d="M3.5 4H1V3h2V1h1v2.5l-.5.5zM13 3V1h-1v2.5l.5.5H15V3h-2zm-1 9.5V15h1v-2h2v-1h-2.5l-.5.5zM1 12v1h2v2h1v-2.5l-.5-.5H1zm11-1.5l-.5.5h-7l-.5-.5v-5l.5-.5h7l.5.5v5zM10 7H6v2h4V7z"></path></g></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" aria-hidden="true" class="re-screen" color="#00000073" viewBox="0 0 16 16"><path fill="currentColor" d="M3.5 4H1V3h2V1h1v2.5l-.5.5zM13 3V1h-1v2.5l.5.5H15V3h-2zm-1 9.5V15h1v-2h2v-1h-2.5l-.5.5zM1 12v1h2v2h1v-2.5l-.5-.5H1zm11-1.5-.5.5h-7l-.5-.5v-5l.5-.5h7l.5.5v5zM10 7H6v2h4V7z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" width="1em" height="1em" class="re-screen" color="#00000073" preserveAspectRatio="xMidYMid meet" viewBox="0 0 16 16"><g fill="currentColor"><path d="M3 12h10V4H3v8zm2-6h6v4H5V6zM2 6H1V2.5l.5-.5H5v1H2v3zm13-3.5V6h-1V3h-3V2h3.5l.5.5zM14 10h1v3.5l-.5.5H11v-1h3v-3zM2 13h3v1H1.5l-.5-.5V10h1v3z"></path></g></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" aria-hidden="true" class="re-screen" color="#00000073" viewBox="0 0 16 16"><path fill="currentColor" d="M3 12h10V4H3v8zm2-6h6v4H5V6zM2 6H1V2.5l.5-.5H5v1H2v3zm13-3.5V6h-1V3h-3V2h3.5l.5.5zM14 10h1v3.5l-.5.5H11v-1h3v-3zM2 13h3v1H1.5l-.5-.5V10h1v3z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="globalization" width="1em" height="1em" preserveAspectRatio="xMidYMid meet" viewBox="0 0 512 512"><path d="M478.33 433.6l-90-218a22 22 0 0 0-40.67 0l-90 218a22 22 0 1 0 40.67 16.79L316.66 406h102.67l18.33 44.39A22 22 0 0 0 458 464a22 22 0 0 0 20.32-30.4zM334.83 362L368 281.65L401.17 362z" fill="currentColor"></path><path d="M267.84 342.92a22 22 0 0 0-4.89-30.7c-.2-.15-15-11.13-36.49-34.73c39.65-53.68 62.11-114.75 71.27-143.49H330a22 22 0 0 0 0-44H214V70a22 22 0 0 0-44 0v20H54a22 22 0 0 0 0 44h197.25c-9.52 26.95-27.05 69.5-53.79 108.36c-31.41-41.68-43.08-68.65-43.17-68.87a22 22 0 0 0-40.58 17c.58 1.38 14.55 34.23 52.86 83.93c.92 1.19 1.83 2.35 2.74 3.51c-39.24 44.35-77.74 71.86-93.85 80.74a22 22 0 1 0 21.07 38.63c2.16-1.18 48.6-26.89 101.63-85.59c22.52 24.08 38 35.44 38.93 36.1a22 22 0 0 0 30.75-4.9z" fill="currentColor"></path></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" aria-hidden="true" class="globalization" viewBox="0 0 512 512"><path fill="currentColor" d="m478.33 433.6-90-218a22 22 0 0 0-40.67 0l-90 218a22 22 0 1 0 40.67 16.79L316.66 406h102.67l18.33 44.39A22 22 0 0 0 458 464a22 22 0 0 0 20.32-30.4zM334.83 362 368 281.65 401.17 362zm-66.99-19.08a22 22 0 0 0-4.89-30.7c-.2-.15-15-11.13-36.49-34.73 39.65-53.68 62.11-114.75 71.27-143.49H330a22 22 0 0 0 0-44H214V70a22 22 0 0 0-44 0v20H54a22 22 0 0 0 0 44h197.25c-9.52 26.95-27.05 69.5-53.79 108.36-31.41-41.68-43.08-68.65-43.17-68.87a22 22 0 0 0-40.58 17c.58 1.38 14.55 34.23 52.86 83.93.92 1.19 1.83 2.35 2.74 3.51-39.24 44.35-77.74 71.86-93.85 80.74a22 22 0 1 0 21.07 38.63c2.16-1.18 48.6-26.89 101.63-85.59 22.52 24.08 38 35.44 38.93 36.1a22 22 0 0 0 30.75-4.9z"/></svg> |
@ -1 +1 @@ |
|||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--mdi" width="20" height="20" preserveAspectRatio="xMidYMid meet" viewBox="0 0 24 24"><path fill="currentColor" d="M1 7h6v2H3v2h4v2H3v2h4v2H1V7m10 0h4v2h-4v2h2a2 2 0 0 1 2 2v2c0 1.11-.89 2-2 2H9v-2h4v-2h-2a2 2 0 0 1-2-2V9c0-1.1.9-2 2-2m8 0h2a2 2 0 0 1 2 2v1h-2V9h-2v6h2v-1h2v1c0 1.11-.89 2-2 2h-2a2 2 0 0 1-2-2V9c0-1.1.9-2 2-2Z"></path></svg> |
|||
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" aria-hidden="true" class="iconify iconify--mdi" viewBox="0 0 24 24"><path fill="currentColor" d="M1 7h6v2H3v2h4v2H3v2h4v2H1V7m10 0h4v2h-4v2h2a2 2 0 0 1 2 2v2c0 1.11-.89 2-2 2H9v-2h4v-2h-2a2 2 0 0 1-2-2V9c0-1.1.9-2 2-2m8 0h2a2 2 0 0 1 2 2v1h-2V9h-2v6h2v-1h2v1c0 1.11-.89 2-2 2h-2a2 2 0 0 1-2-2V9c0-1.1.9-2 2-2Z"/></svg> |
@ -1,9 +0,0 @@ |
|||
import { RouteLocationNormalized } from "vue-router"; |
|||
|
|||
export interface toRouteType extends RouteLocationNormalized { |
|||
meta: { |
|||
roles: Array<string>; |
|||
keepAlive?: boolean; |
|||
dynamicLevel?: string; |
|||
}; |
|||
} |
@ -0,0 +1,7 @@ |
|||
// 如果项目出现 `global is not defined` 报错,可能是您引入某个库的问题,比如 aws-sdk-js https://github.com/aws/aws-sdk-js
|
|||
// 解决办法就是将该文件引入 src/main.ts 即可 import "@/utils/globalPolyfills";
|
|||
if (typeof (window as any).global === "undefined") { |
|||
(window as any).global = window; |
|||
} |
|||
|
|||
export {}; |
@ -0,0 +1,33 @@ |
|||
import { type MessageHandler, ElMessage } from "element-plus"; |
|||
|
|||
// 更多配置请看:https://element-plus.org/zh-CN/component/message.html#message-%E9%85%8D%E7%BD%AE%E9%A1%B9
|
|||
|
|||
type messageTypes = "success" | "info" | "warning" | "error"; |
|||
|
|||
/** |
|||
* `element-plus` 的 `info` 消息类型 |
|||
*/ |
|||
const message = ( |
|||
message: string, |
|||
type = "info" as messageTypes, |
|||
showClose = true, |
|||
duration = 2000, |
|||
center = false, |
|||
grouping = false |
|||
): MessageHandler => { |
|||
return ElMessage({ |
|||
message, |
|||
type, |
|||
showClose, |
|||
duration, |
|||
center, |
|||
grouping |
|||
}); |
|||
}; |
|||
|
|||
/** |
|||
* 关闭 `element-plus` 的所有消息实例 |
|||
*/ |
|||
const closeAllMessage = (): void => ElMessage.closeAll(); |
|||
|
|||
export { message, closeAllMessage }; |
@ -0,0 +1,59 @@ |
|||
import { removeToken, setToken, type DataInfo } from "./auth"; |
|||
import { subBefore, getQueryMap } from "@pureadmin/utils"; |
|||
|
|||
/** |
|||
* 简版前端单点登录,根据实际业务自行编写,平台启动后本地可以跳后面这个链接进行测试 http://localhost:8848/#/permission/page/index?username=sso&roles=admin&accessToken=eyJhbGciOiJIUzUxMiJ9.admin
|
|||
* 划重点: |
|||
* 判断是否为单点登录,不为则直接返回不再进行任何逻辑处理,下面是单点登录后的逻辑处理 |
|||
* 1.清空本地旧信息; |
|||
* 2.获取url中的重要参数信息,然后通过 setToken 保存在本地; |
|||
* 3.删除不需要显示在 url 的参数 |
|||
* 4.使用 window.location.replace 跳转正确页面 |
|||
*/ |
|||
(function () { |
|||
// 获取 url 中的参数
|
|||
const params = getQueryMap(location.href) as DataInfo<Date>; |
|||
const must = ["username", "roles", "accessToken"]; |
|||
const mustLength = must.length; |
|||
if (Object.keys(params).length !== mustLength) return; |
|||
|
|||
// url 参数满足 must 里的全部值,才判定为单点登录,避免非单点登录时刷新页面无限循环
|
|||
let sso = []; |
|||
let start = 0; |
|||
|
|||
while (start < mustLength) { |
|||
if (Object.keys(params).includes(must[start]) && sso.length <= mustLength) { |
|||
sso.push(must[start]); |
|||
} else { |
|||
sso = []; |
|||
} |
|||
start++; |
|||
} |
|||
|
|||
if (sso.length === mustLength) { |
|||
// 判定为单点登录
|
|||
|
|||
// 清空本地旧信息
|
|||
removeToken(); |
|||
|
|||
// 保存新信息到本地
|
|||
setToken(params); |
|||
|
|||
// 删除不需要显示在 url 的参数
|
|||
delete params["roles"]; |
|||
delete params["accessToken"]; |
|||
|
|||
const newUrl = `${location.origin}${location.pathname}${subBefore( |
|||
location.hash, |
|||
"?" |
|||
)}?${JSON.stringify(params) |
|||
.replace(/["{}]/g, "") |
|||
.replace(/:/g, "=") |
|||
.replace(/,/g, "&")}`;
|
|||
|
|||
// 替换历史记录项
|
|||
window.location.replace(newUrl); |
|||
} else { |
|||
return; |
|||
} |
|||
})(); |
@ -0,0 +1,188 @@ |
|||
/** |
|||
* @description 提取菜单树中的每一项uniqueId |
|||
* @param tree 树 |
|||
* @returns 每一项uniqueId组成的数组 |
|||
*/ |
|||
export const extractPathList = (tree: any[]): any => { |
|||
if (!Array.isArray(tree)) { |
|||
console.warn("tree must be an array"); |
|||
return []; |
|||
} |
|||
if (!tree || tree.length === 0) return []; |
|||
const expandedPaths: Array<number | string> = []; |
|||
for (const node of tree) { |
|||
const hasChildren = node.children && node.children.length > 0; |
|||
if (hasChildren) { |
|||
extractPathList(node.children); |
|||
} |
|||
expandedPaths.push(node.uniqueId); |
|||
} |
|||
return expandedPaths; |
|||
}; |
|||
|
|||
/** |
|||
* @description 如果父级下children的length为1,删除children并自动组建唯一uniqueId |
|||
* @param tree 树 |
|||
* @param pathList 每一项的id组成的数组 |
|||
* @returns 组件唯一uniqueId后的树 |
|||
*/ |
|||
export const deleteChildren = (tree: any[], pathList = []): any => { |
|||
if (!Array.isArray(tree)) { |
|||
console.warn("menuTree must be an array"); |
|||
return []; |
|||
} |
|||
if (!tree || tree.length === 0) return []; |
|||
for (const [key, node] of tree.entries()) { |
|||
if (node.children && node.children.length === 1) delete node.children; |
|||
node.id = key; |
|||
node.parentId = pathList.length ? pathList[pathList.length - 1] : null; |
|||
node.pathList = [...pathList, node.id]; |
|||
node.uniqueId = |
|||
node.pathList.length > 1 ? node.pathList.join("-") : node.pathList[0]; |
|||
const hasChildren = node.children && node.children.length > 0; |
|||
if (hasChildren) { |
|||
deleteChildren(node.children, node.pathList); |
|||
} |
|||
} |
|||
return tree; |
|||
}; |
|||
|
|||
/** |
|||
* @description 创建层级关系 |
|||
* @param tree 树 |
|||
* @param pathList 每一项的id组成的数组 |
|||
* @returns 创建层级关系后的树 |
|||
*/ |
|||
export const buildHierarchyTree = (tree: any[], pathList = []): any => { |
|||
if (!Array.isArray(tree)) { |
|||
console.warn("tree must be an array"); |
|||
return []; |
|||
} |
|||
if (!tree || tree.length === 0) return []; |
|||
for (const [key, node] of tree.entries()) { |
|||
node.id = key; |
|||
node.parentId = pathList.length ? pathList[pathList.length - 1] : null; |
|||
node.pathList = [...pathList, node.id]; |
|||
const hasChildren = node.children && node.children.length > 0; |
|||
if (hasChildren) { |
|||
buildHierarchyTree(node.children, node.pathList); |
|||
} |
|||
} |
|||
return tree; |
|||
}; |
|||
|
|||
/** |
|||
* @description 广度优先遍历,根据唯一uniqueId找当前节点信息 |
|||
* @param tree 树 |
|||
* @param uniqueId 唯一uniqueId |
|||
* @returns 当前节点信息 |
|||
*/ |
|||
export const getNodeByUniqueId = ( |
|||
tree: any[], |
|||
uniqueId: number | string |
|||
): any => { |
|||
if (!Array.isArray(tree)) { |
|||
console.warn("menuTree must be an array"); |
|||
return []; |
|||
} |
|||
if (!tree || tree.length === 0) return []; |
|||
const item = tree.find(node => node.uniqueId === uniqueId); |
|||
if (item) return item; |
|||
const childrenList = tree |
|||
.filter(node => node.children) |
|||
.map(i => i.children) |
|||
.flat(1) as unknown; |
|||
return getNodeByUniqueId(childrenList as any[], uniqueId); |
|||
}; |
|||
|
|||
/** |
|||
* @description 向当前唯一uniqueId节点中追加字段 |
|||
* @param tree 树 |
|||
* @param uniqueId 唯一uniqueId |
|||
* @param fields 需要追加的字段 |
|||
* @returns 追加字段后的树 |
|||
*/ |
|||
export const appendFieldByUniqueId = ( |
|||
tree: any[], |
|||
uniqueId: number | string, |
|||
fields: object |
|||
): any => { |
|||
if (!Array.isArray(tree)) { |
|||
console.warn("menuTree must be an array"); |
|||
return []; |
|||
} |
|||
if (!tree || tree.length === 0) return []; |
|||
for (const node of tree) { |
|||
const hasChildren = node.children && node.children.length > 0; |
|||
if ( |
|||
node.uniqueId === uniqueId && |
|||
Object.prototype.toString.call(fields) === "[object Object]" |
|||
) |
|||
Object.assign(node, fields); |
|||
if (hasChildren) { |
|||
appendFieldByUniqueId(node.children, uniqueId, fields); |
|||
} |
|||
} |
|||
return tree; |
|||
}; |
|||
|
|||
/** |
|||
* @description 构造树型结构数据 |
|||
* @param data 数据源 |
|||
* @param id id字段 默认id |
|||
* @param parentId 父节点字段,默认parentId |
|||
* @param children 子节点字段,默认children |
|||
* @returns 追加字段后的树 |
|||
*/ |
|||
export const handleTree = ( |
|||
data: any[], |
|||
id?: string, |
|||
parentId?: string, |
|||
children?: string |
|||
): any => { |
|||
if (!Array.isArray(data)) { |
|||
console.warn("data must be an array"); |
|||
return []; |
|||
} |
|||
const config = { |
|||
id: id || "id", |
|||
parentId: parentId || "parentId", |
|||
childrenList: children || "children" |
|||
}; |
|||
|
|||
const childrenListMap: any = {}; |
|||
const nodeIds: any = {}; |
|||
const tree = []; |
|||
|
|||
for (const d of data) { |
|||
const parentId = d[config.parentId]; |
|||
if (childrenListMap[parentId] == null) { |
|||
childrenListMap[parentId] = []; |
|||
} |
|||
nodeIds[d[config.id]] = d; |
|||
childrenListMap[parentId].push(d); |
|||
} |
|||
|
|||
for (const d of data) { |
|||
const parentId = d[config.parentId]; |
|||
if (nodeIds[parentId] == null) { |
|||
tree.push(d); |
|||
} |
|||
} |
|||
|
|||
for (const t of tree) { |
|||
adaptToChildrenList(t); |
|||
} |
|||
|
|||
function adaptToChildrenList(o: Record<string, any>) { |
|||
if (childrenListMap[o[config.id]] !== null) { |
|||
o[config.childrenList] = childrenListMap[o[config.id]]; |
|||
} |
|||
if (o[config.childrenList]) { |
|||
for (const c of o[config.childrenList]) { |
|||
adaptToChildrenList(c); |
|||
} |
|||
} |
|||
} |
|||
return tree; |
|||
}; |
@ -1,33 +1,74 @@ |
|||
declare interface Fn<T = any, R = T> { |
|||
(...arg: T[]): R; |
|||
} |
|||
// 此文件跟同级目录的 global.d.ts 文件一样也是全局类型声明,只不过这里存放一些零散的全局类型,无需引入直接在 .vue 、.ts 、.tsx 文件使用即可获得类型提示
|
|||
|
|||
declare interface PromiseFn<T = any, R = T> { |
|||
(...arg: T[]): Promise<R>; |
|||
} |
|||
type RefType<T> = T | null; |
|||
|
|||
declare type RefType<T> = T | null; |
|||
type EmitType = (event: string, ...args: any[]) => void; |
|||
|
|||
declare type LabelValueOptions = { |
|||
label: string; |
|||
value: any; |
|||
}[]; |
|||
type TargetContext = "_self" | "_blank"; |
|||
|
|||
declare type EmitType = (event: string, ...args: any[]) => void; |
|||
type ComponentRef<T extends HTMLElement = HTMLDivElement> = |
|||
ComponentElRef<T> | null; |
|||
|
|||
declare type TargetContext = "_self" | "_blank"; |
|||
type ElRef<T extends HTMLElement = HTMLDivElement> = Nullable<T>; |
|||
|
|||
declare interface ComponentElRef<T extends HTMLElement = HTMLDivElement> { |
|||
$el: T; |
|||
} |
|||
type ForDataType<T> = { |
|||
[P in T]?: ForDataType<T[P]>; |
|||
}; |
|||
|
|||
declare type ComponentRef<T extends HTMLElement = HTMLDivElement> = |
|||
ComponentElRef<T> | null; |
|||
type AnyFunction<T> = (...args: any[]) => T; |
|||
|
|||
declare type ElRef<T extends HTMLElement = HTMLDivElement> = Nullable<T>; |
|||
type PropType<T> = VuePropType<T>; |
|||
|
|||
declare type ForDataType<T> = { |
|||
[P in T]?: ForDataType<T[P]>; |
|||
type Writable<T> = { |
|||
-readonly [P in keyof T]: T[P]; |
|||
}; |
|||
|
|||
type Nullable<T> = T | null; |
|||
|
|||
type NonNullable<T> = T extends null | undefined ? never : T; |
|||
|
|||
type Recordable<T = any> = Record<string, T>; |
|||
|
|||
type ReadonlyRecordable<T = any> = { |
|||
readonly [key: string]: T; |
|||
}; |
|||
|
|||
type Indexable<T = any> = { |
|||
[key: string]: T; |
|||
}; |
|||
|
|||
type DeepPartial<T> = { |
|||
[P in keyof T]?: DeepPartial<T[P]>; |
|||
}; |
|||
|
|||
declare type AnyFunction<T> = (...args: any[]) => T; |
|||
type TimeoutHandle = ReturnType<typeof setTimeout>; |
|||
|
|||
type IntervalHandle = ReturnType<typeof setInterval>; |
|||
|
|||
interface ChangeEvent extends Event { |
|||
target: HTMLInputElement; |
|||
} |
|||
|
|||
interface WheelEvent { |
|||
path?: EventTarget[]; |
|||
} |
|||
|
|||
interface ImportMetaEnv extends ViteEnv { |
|||
__: unknown; |
|||
} |
|||
|
|||
interface Fn<T = any, R = T> { |
|||
(...arg: T[]): R; |
|||
} |
|||
|
|||
interface PromiseFn<T = any, R = T> { |
|||
(...arg: T[]): Promise<R>; |
|||
} |
|||
|
|||
interface ComponentElRef<T extends HTMLElement = HTMLDivElement> { |
|||
$el: T; |
|||
} |
|||
|
|||
function parseInt(s: string | number, radix?: number): number; |
|||
|
|||
function parseFloat(string: string | number): number; |
@ -1,134 +0,0 @@ |
|||
import { type FunctionalComponent } from "vue"; |
|||
import { type RouteComponent } from "vue-router"; |
|||
|
|||
export interface StorageConfigs { |
|||
version?: string; |
|||
title?: string; |
|||
fixedHeader?: boolean; |
|||
hiddenSideBar?: boolean; |
|||
multiTagsCache?: boolean; |
|||
keepAlive?: boolean; |
|||
locale?: string; |
|||
layout?: string; |
|||
theme?: string; |
|||
darkMode?: boolean; |
|||
grey?: boolean; |
|||
weak?: boolean; |
|||
hideTabs?: boolean; |
|||
sidebarStatus?: boolean; |
|||
epThemeColor?: string; |
|||
showLogo?: boolean; |
|||
showModel?: string; |
|||
mapConfigure?: { |
|||
amapKey?: string; |
|||
options: { |
|||
resizeEnable?: boolean; |
|||
center?: number[]; |
|||
zoom?: number; |
|||
}; |
|||
}; |
|||
username?: string; |
|||
} |
|||
|
|||
export interface ResponsiveStorage { |
|||
locale: { |
|||
locale?: string; |
|||
}; |
|||
layout: { |
|||
layout?: string; |
|||
theme?: string; |
|||
darkMode?: boolean; |
|||
sidebarStatus?: boolean; |
|||
epThemeColor?: string; |
|||
}; |
|||
configure: { |
|||
grey?: boolean; |
|||
weak?: boolean; |
|||
hideTabs?: boolean; |
|||
showLogo?: boolean; |
|||
showModel?: string; |
|||
multiTagsCache?: boolean; |
|||
}; |
|||
tags?: Array<any>; |
|||
} |
|||
|
|||
export interface RouteChildrenConfigsTable { |
|||
/** 子路由地址 `必填` */ |
|||
path: string; |
|||
/** 路由名字(对应不要重复,和当前组件的`name`保持一致)`必填` */ |
|||
name?: string; |
|||
/** 路由重定向 `可选` */ |
|||
redirect?: string; |
|||
/** 按需加载组件 `可选` */ |
|||
component?: RouteComponent; |
|||
meta?: { |
|||
/** 菜单名称(兼容国际化、非国际化,如何用国际化的写法就必须在根目录的`locales`文件夹下对应添加) `必填` */ |
|||
title: string; |
|||
/** 菜单图标 `可选` */ |
|||
icon?: string | FunctionalComponent; |
|||
/** 菜单名称右侧的额外图标,支持`fontawesome`、`iconfont`、`element-plus-icon` `可选` */ |
|||
extraIcon?: { |
|||
svg?: boolean; |
|||
name?: string; |
|||
}; |
|||
/** 是否在菜单中显示(默认`true`)`可选` */ |
|||
showLink?: boolean; |
|||
/** 是否显示父级菜单 `可选` */ |
|||
showParent?: boolean; |
|||
/** 页面级别权限设置 `可选` */ |
|||
roles?: Array<string>; |
|||
/** 按钮级别权限设置 `可选` */ |
|||
auths?: Array<string>; |
|||
/** 路由组件缓存(开启 `true`、关闭 `false`)`可选` */ |
|||
keepAlive?: boolean; |
|||
/** 内嵌的`iframe`链接 `可选` */ |
|||
frameSrc?: string; |
|||
/** `iframe`页是否开启首次加载动画(默认`true`)`可选` */ |
|||
frameLoading?: boolean; |
|||
/** 页面加载动画(有两种形式,一种直接采用vue内置的`transitions`动画,另一种是使用`animate.css`写进、离场动画)`可选` */ |
|||
transition?: { |
|||
/** |
|||
* @description 当前路由动画效果 |
|||
* @see {@link https://next.router.vuejs.org/guide/advanced/transitions.html#transitions}
|
|||
*/ |
|||
name?: string; |
|||
/** 进场动画 */ |
|||
enterTransition?: string; |
|||
/** 离场动画 */ |
|||
leaveTransition?: string; |
|||
}; |
|||
// 是否不添加信息到标签页,(默认`false`)
|
|||
hiddenTag?: boolean; |
|||
/** 动态路由可打开的最大数量 `可选` */ |
|||
dynamicLevel?: number; |
|||
}; |
|||
/** 子路由配置项 */ |
|||
children?: Array<RouteChildrenConfigsTable>; |
|||
} |
|||
|
|||
/** |
|||
* @description 完整路由配置表 |
|||
* @see {@link https://yiming_chang.gitee.io/pure-admin-doc/pages/782b6e/#%E4%B8%80-%E9%85%8D%E7%BD%AE%E9%A1%B9}
|
|||
*/ |
|||
export interface RouteConfigsTable { |
|||
/** 路由地址 `必填` */ |
|||
path: string; |
|||
/** 路由名字(保持唯一)`可选` */ |
|||
name?: string; |
|||
/** `Layout`组件 `可选` */ |
|||
component?: RouteComponent; |
|||
/** 路由重定向 `可选` */ |
|||
redirect?: string; |
|||
meta?: { |
|||
/** 菜单名称(兼容国际化、非国际化,如何用国际化的写法就必须在根目录的`locales`文件夹下对应添加)`必填` */ |
|||
title: string; |
|||
/** 菜单图标 `可选` */ |
|||
icon?: string | FunctionalComponent; |
|||
/** 是否在菜单中显示(默认`true`)`可选` */ |
|||
showLink?: boolean; |
|||
/** 菜单升序排序,值越高排的越后(只针对顶级路由)`可选` */ |
|||
rank?: number; |
|||
}; |
|||
/** 子路由配置项 */ |
|||
children?: Array<RouteChildrenConfigsTable>; |
|||
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue