Compare commits
63 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cdaa4cdba8 | ||
|
|
496947b524 | ||
|
|
d48058c28f | ||
|
|
820f724d5b | ||
|
|
63e2ddc171 | ||
|
|
60cdaf4697 | ||
|
|
d3cabb0f33 | ||
|
|
97193a71a6 | ||
|
|
f8694e953c | ||
|
|
55f97fce96 | ||
|
|
29144aba44 | ||
|
|
9a448143ae | ||
|
|
a85a2f9022 | ||
|
|
20fee1418c | ||
|
|
2888f8c4f1 | ||
|
|
c07e60e114 | ||
|
|
13427998f3 | ||
|
|
46355bf5bb | ||
|
|
d9b62a9e62 | ||
|
|
6911688ba6 | ||
|
|
45c08d779d | ||
|
|
41b35588c5 | ||
|
|
0a9eb30549 | ||
|
|
a7119c1cbe | ||
|
|
5a463ccfe7 | ||
|
|
73e98814e2 | ||
|
|
77049fdbd7 | ||
|
|
736f1c27cd | ||
|
|
2bac78478c | ||
|
|
bc8a0f3b35 | ||
|
|
53e19f7971 | ||
|
|
abc43dad6e | ||
|
|
f80fbbed20 | ||
|
|
12c2365a26 | ||
|
|
8a926c509f | ||
|
|
e87c38a9d2 | ||
|
|
340a79d286 | ||
|
|
45743a7c74 | ||
|
|
6887d4b1b8 | ||
|
|
b850783ca7 | ||
|
|
05e55ae9a1 | ||
|
|
f5b387231a | ||
|
|
79ebfb9284 | ||
|
|
51fd06c6a1 | ||
|
|
4bb8647990 | ||
|
|
a43d5ce865 | ||
|
|
aea8605a60 | ||
|
|
eb9d1e8238 | ||
|
|
138e0fd2e4 | ||
|
|
526023e0b0 | ||
|
|
885cbf2d9f | ||
|
|
e161102495 | ||
|
|
5300781d05 | ||
|
|
2e7e2ee3ce | ||
|
|
aa165ff70b | ||
|
|
b77ba43572 | ||
|
|
7cc69ac52d | ||
|
|
1b93ce989c | ||
|
|
9e5fe5edbc | ||
|
|
248fbf26ed | ||
|
|
d943550e10 | ||
|
|
3a7832b7fe | ||
|
|
ad3f964c32 |
@@ -8,4 +8,4 @@ VITE_ROUTER_HISTORY = "hash"
|
|||||||
VITE_PROXY_DOMAIN_REAL = ""
|
VITE_PROXY_DOMAIN_REAL = ""
|
||||||
|
|
||||||
# 是否为打包后的文件提供传统浏览器兼容性支持 支持 true 不支持 false
|
# 是否为打包后的文件提供传统浏览器兼容性支持 支持 true 不支持 false
|
||||||
VITE_LEGACY = false
|
VITE_LEGACY = false
|
||||||
14
.env.staging
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
# 预发布也需要生产环境的行为
|
||||||
|
# https://cn.vitejs.dev/guide/env-and-mode.html#modes
|
||||||
|
NODE_ENV=production
|
||||||
|
|
||||||
|
VITE_PUBLIC_PATH = /
|
||||||
|
|
||||||
|
# 线上环境路由历史模式
|
||||||
|
VITE_ROUTER_HISTORY = "hash"
|
||||||
|
|
||||||
|
# 线上环境后端地址
|
||||||
|
VITE_PROXY_DOMAIN_REAL = ""
|
||||||
|
|
||||||
|
# 是否为打包后的文件提供传统浏览器兼容性支持 支持 true 不支持 false
|
||||||
|
VITE_LEGACY = false
|
||||||
@@ -1,4 +1,10 @@
|
|||||||
public
|
public
|
||||||
dist
|
dist
|
||||||
*.d.ts
|
*.d.ts
|
||||||
package.json
|
package.json
|
||||||
|
.eslintrc.js
|
||||||
|
.prettierrc.js
|
||||||
|
commitlint.config.js
|
||||||
|
postcss.config.js
|
||||||
|
tailwind.config.js
|
||||||
|
stylelint.config.js
|
||||||
41
.eslintrc.js
@@ -37,7 +37,7 @@ module.exports = {
|
|||||||
"eslint:recommended",
|
"eslint:recommended",
|
||||||
"@vue/typescript/recommended",
|
"@vue/typescript/recommended",
|
||||||
"@vue/prettier",
|
"@vue/prettier",
|
||||||
"@vue/prettier/@typescript-eslint"
|
"@vue/eslint-config-typescript"
|
||||||
],
|
],
|
||||||
parser: "vue-eslint-parser",
|
parser: "vue-eslint-parser",
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
@@ -49,7 +49,34 @@ module.exports = {
|
|||||||
jsx: true
|
jsx: true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
overrides: [
|
||||||
|
{
|
||||||
|
files: ["*.ts", "*.vue"],
|
||||||
|
rules: {
|
||||||
|
"no-undef": "off"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
files: ["*.vue"],
|
||||||
|
parser: "vue-eslint-parser",
|
||||||
|
parserOptions: {
|
||||||
|
parser: "@typescript-eslint/parser",
|
||||||
|
extraFileExtensions: [".vue"],
|
||||||
|
ecmaVersion: "latest",
|
||||||
|
ecmaFeatures: {
|
||||||
|
jsx: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
"no-undef": "off"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
rules: {
|
rules: {
|
||||||
|
"vue/no-v-html": "off",
|
||||||
|
"vue/require-default-prop": "off",
|
||||||
|
"vue/require-explicit-emits": "off",
|
||||||
|
"vue/multi-word-component-names": "off",
|
||||||
"@typescript-eslint/no-explicit-any": "off", // any
|
"@typescript-eslint/no-explicit-any": "off", // any
|
||||||
"no-debugger": "off",
|
"no-debugger": "off",
|
||||||
"@typescript-eslint/explicit-module-boundary-types": "off", // setup()
|
"@typescript-eslint/explicit-module-boundary-types": "off", // setup()
|
||||||
@@ -57,6 +84,18 @@ module.exports = {
|
|||||||
"@typescript-eslint/ban-ts-comment": "off",
|
"@typescript-eslint/ban-ts-comment": "off",
|
||||||
"@typescript-eslint/no-empty-function": "off",
|
"@typescript-eslint/no-empty-function": "off",
|
||||||
"@typescript-eslint/no-non-null-assertion": "off",
|
"@typescript-eslint/no-non-null-assertion": "off",
|
||||||
|
"vue/html-self-closing": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
html: {
|
||||||
|
void: "always",
|
||||||
|
normal: "always",
|
||||||
|
component: "always"
|
||||||
|
},
|
||||||
|
svg: "always",
|
||||||
|
math: "always"
|
||||||
|
}
|
||||||
|
],
|
||||||
"@typescript-eslint/no-unused-vars": [
|
"@typescript-eslint/no-unused-vars": [
|
||||||
"error",
|
"error",
|
||||||
{
|
{
|
||||||
|
|||||||
2
.gitignore
vendored
@@ -4,6 +4,7 @@ dist
|
|||||||
dist-ssr
|
dist-ssr
|
||||||
*.local
|
*.local
|
||||||
.eslintcache
|
.eslintcache
|
||||||
|
report.html
|
||||||
|
|
||||||
yarn.lock
|
yarn.lock
|
||||||
npm-debug.log*
|
npm-debug.log*
|
||||||
@@ -17,3 +18,4 @@ tests/**/coverage/
|
|||||||
*.ntvs*
|
*.ntvs*
|
||||||
*.njsproj
|
*.njsproj
|
||||||
*.sln
|
*.sln
|
||||||
|
tsconfig.tsbuildinfo
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
"*.{js,jsx,ts,tsx}": ["eslint --fix", "prettier --write"],
|
"*.{js,jsx,ts,tsx}": ["eslint --fix", "prettier --write"],
|
||||||
"{!(package)*.json,.!(browserslist)*rc}": ["prettier --write--parser json"],
|
"{!(package)*.json}": ["prettier --write--parser json"],
|
||||||
"package.json": ["prettier --write"],
|
"package.json": ["prettier --write"],
|
||||||
"*.vue": ["eslint --fix", "prettier --write", "stylelint --fix"],
|
"*.vue": ["eslint --fix", "prettier --write", "stylelint --fix"],
|
||||||
"*.{vue,css,scss,postcss,less}": ["stylelint --fix", "prettier --write"],
|
"*.{vue,css,scss,postcss,less}": ["stylelint --fix", "prettier --write"],
|
||||||
|
|||||||
3
.npmrc
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
shamefully-hoist=true
|
||||||
|
strict-peer-dependencies=false
|
||||||
|
shell-emulator=true
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
bracketSpacing: true,
|
bracketSpacing: true,
|
||||||
jsxBracketSameLine: true,
|
|
||||||
singleQuote: false,
|
singleQuote: false,
|
||||||
arrowParens: "avoid",
|
arrowParens: "avoid",
|
||||||
trailingComma: "none"
|
trailingComma: "none"
|
||||||
|
|||||||
7
.vscode/extensions.json
vendored
@@ -1,16 +1,15 @@
|
|||||||
{
|
{
|
||||||
"recommendations": [
|
"recommendations": [
|
||||||
"johnsoncodehk.vscode-typescript-vue-plugin",
|
|
||||||
"voorjaar.windicss-intellisense",
|
|
||||||
"vscode-icons-team.vscode-icons",
|
"vscode-icons-team.vscode-icons",
|
||||||
"davidanson.vscode-markdownlint",
|
"davidanson.vscode-markdownlint",
|
||||||
"stylelint.vscode-stylelint",
|
"stylelint.vscode-stylelint",
|
||||||
|
"bradlc.vscode-tailwindcss",
|
||||||
"dbaeumer.vscode-eslint",
|
"dbaeumer.vscode-eslint",
|
||||||
"esbenp.prettier-vscode",
|
"esbenp.prettier-vscode",
|
||||||
"johnsoncodehk.volar",
|
|
||||||
"lokalise.i18n-ally",
|
"lokalise.i18n-ally",
|
||||||
"mikestead.dotenv",
|
"mikestead.dotenv",
|
||||||
"eamodio.gitlens",
|
"eamodio.gitlens",
|
||||||
"antfu.iconify"
|
"antfu.iconify",
|
||||||
|
"Vue.volar"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
14
.vscode/settings.json
vendored
@@ -1,15 +1,12 @@
|
|||||||
{
|
{
|
||||||
"editor.formatOnType": true,
|
"editor.formatOnType": true,
|
||||||
"editor.formatOnSave": true,
|
"editor.formatOnSave": true,
|
||||||
"javascript.updateImportsOnFileMove.enabled": "always",
|
|
||||||
"[vue]": {
|
"[vue]": {
|
||||||
"editor.defaultFormatter": "esbenp.prettier-vscode"
|
"editor.defaultFormatter": "esbenp.prettier-vscode"
|
||||||
},
|
},
|
||||||
"[javascript]": {
|
|
||||||
"editor.defaultFormatter": "vscode.typescript-language-features"
|
|
||||||
},
|
|
||||||
"editor.tabSize": 2,
|
"editor.tabSize": 2,
|
||||||
"editor.formatOnPaste": true,
|
"editor.formatOnPaste": true,
|
||||||
|
"editor.guides.bracketPairs": "active",
|
||||||
"files.autoSave": "afterDelay",
|
"files.autoSave": "afterDelay",
|
||||||
"git.confirmSync": false,
|
"git.confirmSync": false,
|
||||||
"workbench.startupEditor": "newUntitledFile",
|
"workbench.startupEditor": "newUntitledFile",
|
||||||
@@ -30,14 +27,13 @@
|
|||||||
"editor.codeActionsOnSave": {
|
"editor.codeActionsOnSave": {
|
||||||
"source.fixAll.eslint": true
|
"source.fixAll.eslint": true
|
||||||
},
|
},
|
||||||
"typescript.tsdk": "node_modules/typescript/lib",
|
"i18n-ally.localesPaths": "locales",
|
||||||
"i18n-ally.localesPaths": ["src/plugins/i18n"],
|
|
||||||
"i18n-ally.keystyle": "nested",
|
"i18n-ally.keystyle": "nested",
|
||||||
"i18n-ally.sortKeys": true,
|
"i18n-ally.sortKeys": true,
|
||||||
"i18n-ally.namespace": true,
|
"i18n-ally.namespace": true,
|
||||||
"i18n-ally.pathMatcher": "{locale}/{namespaces}.{ext}",
|
"i18n-ally.enabledParsers": ["yaml", "js"],
|
||||||
"i18n-ally.enabledParsers": ["ts"],
|
|
||||||
"i18n-ally.sourceLanguage": "en",
|
"i18n-ally.sourceLanguage": "en",
|
||||||
"i18n-ally.displayLanguage": "zh-CN",
|
"i18n-ally.displayLanguage": "zh-CN",
|
||||||
"i18n-ally.enabledFrameworks": ["vue", "react"]
|
"i18n-ally.enabledFrameworks": ["vue"],
|
||||||
|
"iconify.excludes": ["el"]
|
||||||
}
|
}
|
||||||
|
|||||||
2
LICENSE
@@ -1,6 +1,6 @@
|
|||||||
MIT License
|
MIT License
|
||||||
|
|
||||||
Copyright (c) 2021 啝裳
|
Copyright (c) 2022 啝裳
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
|||||||
@@ -6,11 +6,18 @@
|
|||||||
|
|
||||||
## introduce
|
## introduce
|
||||||
|
|
||||||
The lite version is based on the shelf extracted from https://github.com/xiaoxian521/vue-pure-admin, which contains the main functions and is more suitable for actual project development
|
The lite version is based on the shelf extracted from [vue-pure-admin](https://github.com/xiaoxian521/vue-pure-admin), which contains the main functions and is more suitable for actual project development, the packaged size is only `3MB`, and will permanently sync the full version of the code
|
||||||
|
|
||||||
|
## Supporting Video
|
||||||
|
|
||||||
|
- [Click Watch Tutorial](https://www.bilibili.com/video/BV1534y1S7HV)
|
||||||
|
- [Click Watch UI Design](https://www.bilibili.com/video/BV17g411T7rq)
|
||||||
|
|
||||||
## Docs
|
## Docs
|
||||||
|
|
||||||
<https://pure-admin-doc.vercel.app/>
|
- [Click me to view the domestic documentation site](http://yiming_chang.gitee.io/pure-admin-doc)
|
||||||
|
- [Click me to view foreign document site 1](https://xiaoxian521.github.io/pure-admin-doc)
|
||||||
|
- [Click me to view foreign document site 2](https://pure-admin-doc.vercel.app)
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@@ -32,6 +39,11 @@ I think you should fork the project first to develop, so that you can pull the u
|
|||||||
|
|
||||||
bilibili: https://www.bilibili.com/video/BV1534y1S7HV/
|
bilibili: https://www.bilibili.com/video/BV1534y1S7HV/
|
||||||
|
|
||||||
## ⚠️ Note
|
## ⚠️ Attention
|
||||||
|
|
||||||
The lite version does not accept any issues and prs. If you have any questions, please go to the full version https://github.com/xiaoxian521/vue-pure-admin/issues/new/choose to mention it, thank you! ! !
|
- The Lite version does not accept any issues and prs. If you have any questions, please go to the full version https://github.com/xiaoxian521/vue-pure-admin/issues/new/choose to mention, thank you! ! !
|
||||||
|
- Don't use the `delete-i18n` branch code, this branch is just for you to completely delete the internationalized references, it won't sync the code! ! ! [Completely remove the internationalization tutorial](https://www.bilibili.com/video/BV1Ru411B7k3/), please be sure to use the code from the `main` branch! ! !
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
In principle, no fees and copyrights are charged, and you can use it with confidence, but if you need secondary open source, please contact the author for permission!
|
||||||
|
|||||||
31
README.md
@@ -6,16 +6,18 @@
|
|||||||
|
|
||||||
## 介绍
|
## 介绍
|
||||||
|
|
||||||
精简版是基于 https://github.com/xiaoxian521/vue-pure-admin 提炼出的架子,包含主体功能,更适合实际项目开发
|
精简版是基于 [vue-pure-admin](https://github.com/xiaoxian521/vue-pure-admin) 提炼出的架子,包含主体功能,更适合实际项目开发,打包后的大小仅 `3MB`,并且会永久同步完整版的代码
|
||||||
|
|
||||||
## 配套视频
|
## 配套视频
|
||||||
|
|
||||||
教程:<https://www.bilibili.com/video/BV1534y1S7HV/>
|
- [点我查看教程](https://www.bilibili.com/video/BV1534y1S7HV)
|
||||||
UI 设计:<https://www.bilibili.com/video/BV17g411T7rq/>
|
- [点我查看 UI 设计](https://www.bilibili.com/video/BV17g411T7rq)
|
||||||
|
|
||||||
## 配套文档
|
## 配套文档
|
||||||
|
|
||||||
<https://pure-admin-doc.vercel.app/>
|
- [点我查看国内文档站](http://yiming_chang.gitee.io/pure-admin-doc)
|
||||||
|
- [点我查看国外文档站 1](https://xiaoxian521.github.io/pure-admin-doc)
|
||||||
|
- [点我查看国外文档站 2](https://pure-admin-doc.vercel.app)
|
||||||
|
|
||||||
## 维护者
|
## 维护者
|
||||||
|
|
||||||
@@ -23,15 +25,15 @@ UI 设计:<https://www.bilibili.com/video/BV17g411T7rq/>
|
|||||||
|
|
||||||
## 捐赠
|
## 捐赠
|
||||||
|
|
||||||
如果你觉得这个项目对你有帮助,你可以帮作者买一杯咖啡表示支持
|
如果你觉得这个项目对您有帮助,可以帮作者买一杯果汁 🍹 表示支持
|
||||||
|
|
||||||
<img src="http://yiming_chang.gitee.io/manages/pay.jpg" width="150px" height="150px" />
|
<img src="https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f69bf13c5b854ed5b699807cafa0e3ce~tplv-k3u1fbpfcp-zoom-in-crop-mark:1304:0:0:0.awebp?" width="150px" height="150px" />
|
||||||
|
|
||||||
## 付费咨询、需求定制
|
## QQ 交流群
|
||||||
|
|
||||||
作者精力有限,需要提供技术服务的可扫下面的二维码加微信,添加请备注来意
|
一群已满,下面是二群,群里严禁 `黄` 、 `赌` 、 `毒` 、 `vpn` 等违法行为!
|
||||||
|
|
||||||
<img src="http://yiming_chang.gitee.io/manages/wechat.jpg" width="150px" height="150px" />
|
<img src="http://yiming_chang.gitee.io/pure-admin-doc/img/support/qq.png" width="150px" height="225px" />
|
||||||
|
|
||||||
## 用法
|
## 用法
|
||||||
|
|
||||||
@@ -47,8 +49,15 @@ pnpm add 包名
|
|||||||
|
|
||||||
pnpm remove 包名
|
pnpm remove 包名
|
||||||
|
|
||||||
我认为你应该先 fork 项目去开发,以便我更新时你可以同步拉取更新!!!
|
我认为你应该先 `fork` 项目去开发,以便我更新时您可以同步拉取更新!!!
|
||||||
|
|
||||||
## ⚠️ 注意
|
## ⚠️ 注意
|
||||||
|
|
||||||
精简版不接受任何 issues 和 pr,如果有问题请到完整版 https://github.com/xiaoxian521/vue-pure-admin/issues/new/choose 去提,谢谢!!!
|
- 精简版不接受任何 `issues` 和 `pr`,如果有问题请到完整版 [issues](https://github.com/xiaoxian521/vue-pure-admin/issues/new/choose) 去提,谢谢!!!
|
||||||
|
- 不要使用 `delete-i18n` 分支代码,这个分支只是给你们完全删除国际化的参考,不会同步代码的!!! [完全删除国际化教程](https://www.bilibili.com/video/BV1Ru411B7k3/) ,请务必使用 `main` 分支的代码!!!
|
||||||
|
|
||||||
|
## 许可证
|
||||||
|
|
||||||
|
原则上不收取任何费用及版权,可以放心使用,不过如需二次开源(比如用此平台二次开发并开源)请联系作者获取许可!
|
||||||
|
|
||||||
|
[MIT © xiaoxian521-2020](./LICENSE)
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ const warpperEnv = (envConf: Recordable): ViteEnv => {
|
|||||||
|
|
||||||
// 跨域代理重写
|
// 跨域代理重写
|
||||||
const regExps = (value: string, reg: string): string => {
|
const regExps = (value: string, reg: string): string => {
|
||||||
return value.replace(new RegExp(reg, "g"), "");
|
return value.replace(new RegExp(`^${reg}`, "g"), "");
|
||||||
};
|
};
|
||||||
|
|
||||||
// 环境变量
|
// 环境变量
|
||||||
|
|||||||
50
build/info.ts
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import type { Plugin } from "vite";
|
||||||
|
import dayjs, { Dayjs } from "dayjs";
|
||||||
|
import duration from "dayjs/plugin/duration";
|
||||||
|
import { green, blue, bold } from "picocolors";
|
||||||
|
import { getPackageSize } from "@pureadmin/utils";
|
||||||
|
dayjs.extend(duration);
|
||||||
|
|
||||||
|
export function viteBuildInfo(): Plugin {
|
||||||
|
let config: { command: string };
|
||||||
|
let startTime: Dayjs;
|
||||||
|
let endTime: Dayjs;
|
||||||
|
return {
|
||||||
|
name: "vite:buildInfo",
|
||||||
|
configResolved(resolvedConfig: { command: string }) {
|
||||||
|
config = resolvedConfig;
|
||||||
|
},
|
||||||
|
buildStart() {
|
||||||
|
console.log(
|
||||||
|
bold(
|
||||||
|
green(
|
||||||
|
`👏欢迎使用${blue(
|
||||||
|
"[vue-pure-admin]"
|
||||||
|
)},如果您感觉不错,记得点击后面链接给个star哦💖 https://github.com/xiaoxian521/vue-pure-admin`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
if (config.command === "build") {
|
||||||
|
startTime = dayjs(new Date());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
closeBundle() {
|
||||||
|
if (config.command === "build") {
|
||||||
|
endTime = dayjs(new Date());
|
||||||
|
getPackageSize({
|
||||||
|
callback: (size: string) => {
|
||||||
|
console.log(
|
||||||
|
bold(
|
||||||
|
green(
|
||||||
|
`🎉恭喜打包完成(总用时${dayjs
|
||||||
|
.duration(endTime.diff(startTime))
|
||||||
|
.format("mm分ss秒")},打包后的大小为${size})`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
73
build/plugins.ts
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
import { resolve } from "path";
|
||||||
|
import vue from "@vitejs/plugin-vue";
|
||||||
|
import { viteBuildInfo } from "./info";
|
||||||
|
import svgLoader from "vite-svg-loader";
|
||||||
|
import legacy from "@vitejs/plugin-legacy";
|
||||||
|
import vueJsx from "@vitejs/plugin-vue-jsx";
|
||||||
|
import { viteMockServe } from "vite-plugin-mock";
|
||||||
|
import VueI18n from "@intlify/vite-plugin-vue-i18n";
|
||||||
|
// import ElementPlus from "unplugin-element-plus/vite";
|
||||||
|
import { visualizer } from "rollup-plugin-visualizer";
|
||||||
|
import removeConsole from "vite-plugin-remove-console";
|
||||||
|
import themePreprocessorPlugin from "@pureadmin/theme";
|
||||||
|
import { genScssMultipleScopeVars } from "/@/layout/theme";
|
||||||
|
import DefineOptions from "unplugin-vue-define-options/vite";
|
||||||
|
|
||||||
|
export function getPluginsList(command, VITE_LEGACY) {
|
||||||
|
const prodMock = true;
|
||||||
|
const lifecycle = process.env.npm_lifecycle_event;
|
||||||
|
return [
|
||||||
|
vue(),
|
||||||
|
// https://github.com/intlify/bundle-tools/tree/main/packages/vite-plugin-vue-i18n
|
||||||
|
VueI18n({
|
||||||
|
runtimeOnly: true,
|
||||||
|
compositionOnly: true,
|
||||||
|
include: [resolve("locales/**")]
|
||||||
|
}),
|
||||||
|
// jsx、tsx语法支持
|
||||||
|
vueJsx(),
|
||||||
|
DefineOptions(),
|
||||||
|
// 线上环境删除console
|
||||||
|
removeConsole({ external: ["src/assets/iconfont/iconfont.js"] }),
|
||||||
|
viteBuildInfo(),
|
||||||
|
// 自定义主题
|
||||||
|
themePreprocessorPlugin({
|
||||||
|
scss: {
|
||||||
|
multipleScopeVars: genScssMultipleScopeVars(),
|
||||||
|
// 在生产模式是否抽取独立的主题css文件,extract为true以下属性有效
|
||||||
|
extract: true,
|
||||||
|
// 会选取defaultScopeName对应的主题css文件在html添加link
|
||||||
|
themeLinkTagId: "head",
|
||||||
|
// "head"||"head-prepend" || "body" ||"body-prepend"
|
||||||
|
themeLinkTagInjectTo: "head",
|
||||||
|
// 是否对抽取的css文件内对应scopeName的权重类名移除
|
||||||
|
removeCssScopeName: false
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
// svg组件化支持
|
||||||
|
svgLoader(),
|
||||||
|
// ElementPlus({}),
|
||||||
|
// mock支持
|
||||||
|
viteMockServe({
|
||||||
|
mockPath: "mock",
|
||||||
|
localEnabled: command === "serve",
|
||||||
|
prodEnabled: command !== "serve" && prodMock,
|
||||||
|
injectCode: `
|
||||||
|
import { setupProdMockServer } from './mockProdServer';
|
||||||
|
setupProdMockServer();
|
||||||
|
`,
|
||||||
|
logger: false
|
||||||
|
}),
|
||||||
|
// 是否为打包后的文件提供传统浏览器兼容性支持
|
||||||
|
VITE_LEGACY
|
||||||
|
? legacy({
|
||||||
|
targets: ["ie >= 11"],
|
||||||
|
additionalLegacyPolyfills: ["regenerator-runtime/runtime"]
|
||||||
|
})
|
||||||
|
: null,
|
||||||
|
// 打包分析
|
||||||
|
lifecycle === "report"
|
||||||
|
? visualizer({ open: true, brotliSize: true, filename: "report.html" })
|
||||||
|
: null
|
||||||
|
];
|
||||||
|
}
|
||||||
24
index.html
@@ -2,10 +2,14 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<link rel="icon" href="/favicon.ico" />
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" />
|
||||||
<link rel="stylesheet" href="/iconfont.css" />
|
<meta name="renderer" content="webkit" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta
|
||||||
|
name="viewport"
|
||||||
|
content="width=device-width,initial-scale=1.0,minimum-scale=1.0,maximum-scale=1.0,user-scalable=0"
|
||||||
|
/>
|
||||||
<title>pure-admin-thin</title>
|
<title>pure-admin-thin</title>
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
<script>
|
<script>
|
||||||
window.process = {};
|
window.process = {};
|
||||||
</script>
|
</script>
|
||||||
@@ -14,20 +18,16 @@
|
|||||||
<body>
|
<body>
|
||||||
<div id="app">
|
<div id="app">
|
||||||
<style>
|
<style>
|
||||||
* {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
html,
|
html,
|
||||||
body {
|
body,
|
||||||
|
#app {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
position: relative;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
font-family: "Reggae One", cursive;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.loader,
|
.loader,
|
||||||
@@ -53,6 +53,8 @@
|
|||||||
transform: translateZ(0);
|
transform: translateZ(0);
|
||||||
-webkit-animation-delay: -0.16s;
|
-webkit-animation-delay: -0.16s;
|
||||||
animation-delay: -0.16s;
|
animation-delay: -0.16s;
|
||||||
|
top: 0;
|
||||||
|
transform: translate(-50%, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
.loader:before,
|
.loader:before,
|
||||||
@@ -96,7 +98,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<div class="loader">Loading...</div>
|
<div class="loader"></div>
|
||||||
</div>
|
</div>
|
||||||
<script type="module" src="/src/main.ts"></script>
|
<script type="module" src="/src/main.ts"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|||||||
39
locales/en.yaml
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
buttons:
|
||||||
|
hsLoginOut: LoginOut
|
||||||
|
hsfullscreen: FullScreen
|
||||||
|
hsexitfullscreen: ExitFullscreen
|
||||||
|
hsrefreshRoute: RefreshRoute
|
||||||
|
hslogin: Login
|
||||||
|
hsadd: Add
|
||||||
|
hsmark: Mark/Cancel
|
||||||
|
hssave: Save
|
||||||
|
hssearch: Search
|
||||||
|
hsexpendAll: Expand All
|
||||||
|
hscollapseAll: Collapse All
|
||||||
|
hssystemSet: Open ProjectConfig
|
||||||
|
hsdelete: Delete
|
||||||
|
hsreload: Reload
|
||||||
|
hscloseCurrentTab: Close CurrentTab
|
||||||
|
hscloseLeftTabs: Close LeftTabs
|
||||||
|
hscloseRightTabs: Close RightTabs
|
||||||
|
hscloseOtherTabs: Close OtherTabs
|
||||||
|
hscloseAllTabs: Close AllTabs
|
||||||
|
menus:
|
||||||
|
hshome: Home
|
||||||
|
hslogin: Login
|
||||||
|
hserror: Error Page
|
||||||
|
hsfourZeroFour: "404"
|
||||||
|
hsfourZeroOne: "403"
|
||||||
|
hsFive: "500"
|
||||||
|
permission: Permission Manage
|
||||||
|
permissionPage: Page Permission
|
||||||
|
permissionButton: Button Permission
|
||||||
|
status:
|
||||||
|
hsLoad: Loading...
|
||||||
|
login:
|
||||||
|
username: Username
|
||||||
|
password: Password
|
||||||
|
login: Login
|
||||||
|
usernameReg: Please enter username
|
||||||
|
passwordReg: Please enter password
|
||||||
|
passwordRuleReg: The password format should be any combination of 8-18 digits
|
||||||
39
locales/zh-CN.yaml
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
buttons:
|
||||||
|
hsLoginOut: 退出系统
|
||||||
|
hsfullscreen: 全屏
|
||||||
|
hsexitfullscreen: 退出全屏
|
||||||
|
hsrefreshRoute: 刷新路由
|
||||||
|
hslogin: 登录
|
||||||
|
hsadd: 新增
|
||||||
|
hsmark: 标记/取消
|
||||||
|
hssave: 保存
|
||||||
|
hssearch: 搜索
|
||||||
|
hsexpendAll: 全部展开
|
||||||
|
hscollapseAll: 全部折叠
|
||||||
|
hssystemSet: 打开项目配置
|
||||||
|
hsdelete: 删除
|
||||||
|
hsreload: 重新加载
|
||||||
|
hscloseCurrentTab: 关闭当前标签页
|
||||||
|
hscloseLeftTabs: 关闭左侧标签页
|
||||||
|
hscloseRightTabs: 关闭右侧标签页
|
||||||
|
hscloseOtherTabs: 关闭其他标签页
|
||||||
|
hscloseAllTabs: 关闭全部标签页
|
||||||
|
menus:
|
||||||
|
hshome: 首页
|
||||||
|
hslogin: 登录
|
||||||
|
hserror: 错误页面
|
||||||
|
hsfourZeroFour: "404"
|
||||||
|
hsfourZeroOne: "403"
|
||||||
|
hsFive: "500"
|
||||||
|
permission: 权限管理
|
||||||
|
permissionPage: 页面权限
|
||||||
|
permissionButton: 按钮权限
|
||||||
|
status:
|
||||||
|
hsLoad: 加载中...
|
||||||
|
login:
|
||||||
|
username: 账号
|
||||||
|
password: 密码
|
||||||
|
login: 登录
|
||||||
|
usernameReg: 请输入账号
|
||||||
|
passwordReg: 请输入密码
|
||||||
|
passwordRuleReg: 密码格式应为8-18位数字、字母、符号的任意两种组合
|
||||||
@@ -3,32 +3,24 @@ import { MockMethod } from "vite-plugin-mock";
|
|||||||
|
|
||||||
const permissionRouter = {
|
const permissionRouter = {
|
||||||
path: "/permission",
|
path: "/permission",
|
||||||
name: "permission",
|
|
||||||
redirect: "/permission/page/index",
|
|
||||||
meta: {
|
meta: {
|
||||||
title: "menus.permission",
|
title: "menus.permission",
|
||||||
icon: "lollipop",
|
icon: "lollipop",
|
||||||
i18n: true,
|
rank: 7
|
||||||
showLink: true,
|
|
||||||
rank: 3
|
|
||||||
},
|
},
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
path: "/permission/page/index",
|
path: "/permission/page/index",
|
||||||
name: "permissionPage",
|
name: "PermissionPage",
|
||||||
meta: {
|
meta: {
|
||||||
title: "menus.permissionPage",
|
title: "menus.permissionPage"
|
||||||
i18n: true,
|
|
||||||
showLink: true
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: "/permission/button/index",
|
path: "/permission/button/index",
|
||||||
name: "permissionButton",
|
name: "PermissionButton",
|
||||||
meta: {
|
meta: {
|
||||||
title: "menus.permissionButton",
|
title: "menus.permissionButton",
|
||||||
i18n: true,
|
|
||||||
showLink: true,
|
|
||||||
authority: []
|
authority: []
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
155
package.json
@@ -1,20 +1,20 @@
|
|||||||
{
|
{
|
||||||
"name": "pure-admin-thin",
|
"name": "pure-admin-thin",
|
||||||
"version": "2.8.5",
|
"version": "3.5.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"engines": {
|
|
||||||
"node": ">= 16",
|
|
||||||
"pnpm": ">= 6"
|
|
||||||
},
|
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "cross-env --max_old_space_size=4096 vite",
|
"dev": "NODE_OPTIONS=--max-old-space-size=4096 vite",
|
||||||
"serve": "pnpm dev",
|
"serve": "pnpm dev",
|
||||||
"build": "rimraf dist && cross-env vite build",
|
"build": "rimraf dist && NODE_OPTIONS=--max-old-space-size=8192 vite build",
|
||||||
|
"build:staging": "rimraf dist && vite build --mode staging",
|
||||||
|
"report": "rimraf dist && vite build",
|
||||||
"preview": "vite preview",
|
"preview": "vite preview",
|
||||||
"preview:build": "pnpm build && vite preview",
|
"preview:build": "pnpm build && vite preview",
|
||||||
|
"typecheck": "tsc --noEmit && vue-tsc --noEmit --skipLibCheck",
|
||||||
|
"cloc": "NODE_OPTIONS=--max-old-space-size=4096 cloc . --exclude-dir=node_modules --exclude-lang=YAML",
|
||||||
"clean:cache": "rm -rf node_modules && rm -rf .eslintcache && pnpm install",
|
"clean:cache": "rm -rf node_modules && rm -rf .eslintcache && pnpm install",
|
||||||
"lint:eslint": "eslint --cache --max-warnings 0 \"{src,mock}/**/*.{vue,ts,tsx}\" --fix",
|
"lint:eslint": "eslint --cache --max-warnings 0 \"{src,mock,build}/**/*.{vue,js,ts,tsx}\" --fix",
|
||||||
"lint:prettier": "prettier --write \"src/**/*.{js,json,tsx,css,less,scss,vue,html,md}\"",
|
"lint:prettier": "prettier --write \"src/**/*.{js,ts,json,tsx,css,less,scss,vue,html,md}\"",
|
||||||
"lint:stylelint": "stylelint --cache --fix \"**/*.{vue,css,scss,postcss,less}\" --cache --cache-location node_modules/.cache/stylelint/",
|
"lint:stylelint": "stylelint --cache --fix \"**/*.{vue,css,scss,postcss,less}\" --cache --cache-location node_modules/.cache/stylelint/",
|
||||||
"lint:lint-staged": "lint-staged -c ./.husky/lintstagedrc.js",
|
"lint:lint-staged": "lint-staged -c ./.husky/lintstagedrc.js",
|
||||||
"lint:pretty": "pretty-quick --staged",
|
"lint:pretty": "pretty-quick --staged",
|
||||||
@@ -28,87 +28,110 @@
|
|||||||
"not op_mini all"
|
"not op_mini all"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ctrl/tinycolor": "^3.4.0",
|
"@ctrl/tinycolor": "^3.4.1",
|
||||||
"@fortawesome/fontawesome-svg-core": "^1.2.36",
|
"@pureadmin/components": "^1.1.0",
|
||||||
"@fortawesome/free-solid-svg-icons": "^5.15.4",
|
"@pureadmin/descriptions": "^1.1.0",
|
||||||
"@fortawesome/vue-fontawesome": "^3.0.0-5",
|
"@pureadmin/table": "^1.2.0",
|
||||||
"@vueuse/core": "^7.5.3",
|
"@pureadmin/utils": "^0.1.1",
|
||||||
"@vueuse/motion": "^2.0.0-beta.9",
|
"@vueuse/core": "^9.1.1",
|
||||||
"@vueuse/shared": "^7.5.3",
|
"@vueuse/motion": "^2.0.0-beta.12",
|
||||||
|
"@vueuse/shared": "^9.1.1",
|
||||||
"animate.css": "^4.1.1",
|
"animate.css": "^4.1.1",
|
||||||
"axios": "^0.25.0",
|
"axios": "^0.27.2",
|
||||||
"css-color-function": "^1.3.3",
|
"dayjs": "^1.11.4",
|
||||||
"dayjs": "^1.10.7",
|
"echarts": "^5.3.3",
|
||||||
"element-plus": "1.3.0-beta.1",
|
"element-plus": "^2.2.16",
|
||||||
"element-resize-detector": "^1.2.3",
|
"element-resize-detector": "^1.2.3",
|
||||||
"font-awesome": "^4.7.0",
|
|
||||||
"js-cookie": "^3.0.1",
|
"js-cookie": "^3.0.1",
|
||||||
|
"lodash": "^4.17.21",
|
||||||
"lodash-es": "^4.17.21",
|
"lodash-es": "^4.17.21",
|
||||||
|
"lodash-unified": "^1.0.2",
|
||||||
"mitt": "^3.0.0",
|
"mitt": "^3.0.0",
|
||||||
"mockjs": "^1.1.0",
|
"mockjs": "^1.1.0",
|
||||||
"nprogress": "^0.2.0",
|
"nprogress": "^0.2.0",
|
||||||
"path": "^0.12.7",
|
"path": "^0.12.7",
|
||||||
"pinia": "^2.0.9",
|
"pinia": "^2.0.21",
|
||||||
"qs": "^6.10.2",
|
"qs": "^6.11.0",
|
||||||
"remixicon": "^2.5.0",
|
|
||||||
"resize-observer-polyfill": "^1.5.1",
|
"resize-observer-polyfill": "^1.5.1",
|
||||||
"responsive-storage": "^1.0.11",
|
"responsive-storage": "^2.1.0",
|
||||||
"rgb-hex": "^4.0.0",
|
"vue": "^3.2.39",
|
||||||
"vue": "^3.2.27",
|
"vue-i18n": "^9.2.2",
|
||||||
"vue-i18n": "^9.2.0-beta.26",
|
"vue-router": "^4.1.5",
|
||||||
"vue-router": "^4.0.12",
|
"vue-types": "^4.2.1",
|
||||||
"vue-types": "^4.1.1"
|
"vxe-table": "^4.3.2",
|
||||||
|
"xe-utils": "^3.5.6"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@commitlint/cli": "13.1.0",
|
"@commitlint/cli": "13.1.0",
|
||||||
"@commitlint/config-conventional": "13.1.0",
|
"@commitlint/config-conventional": "13.1.0",
|
||||||
"@iconify-icons/ep": "^1.1.3",
|
"@iconify-icons/ep": "^1.2.7",
|
||||||
"@iconify/vue": "^3.1.2",
|
"@iconify-icons/ri": "^1.2.3",
|
||||||
|
"@iconify/vue": "^3.2.1",
|
||||||
|
"@intlify/vite-plugin-vue-i18n": "^6.0.1",
|
||||||
|
"@pureadmin/theme": "^2.4.0",
|
||||||
"@types/element-resize-detector": "1.1.3",
|
"@types/element-resize-detector": "1.1.3",
|
||||||
"@types/js-cookie": "^3.0.1",
|
"@types/js-cookie": "^3.0.1",
|
||||||
|
"@types/lodash": "^4.14.180",
|
||||||
|
"@types/lodash-es": "^4.17.6",
|
||||||
"@types/mockjs": "1.0.3",
|
"@types/mockjs": "1.0.3",
|
||||||
"@types/node": "14.14.14",
|
"@types/node": "14.14.14",
|
||||||
"@types/nprogress": "0.2.0",
|
"@types/nprogress": "0.2.0",
|
||||||
|
"@types/qrcode": "^1.5.0",
|
||||||
"@types/qs": "^6.9.7",
|
"@types/qs": "^6.9.7",
|
||||||
"@typescript-eslint/eslint-plugin": "4.31.0",
|
"@typescript-eslint/eslint-plugin": "^5.10.2",
|
||||||
"@typescript-eslint/parser": "4.31.0",
|
"@typescript-eslint/parser": "^5.10.2",
|
||||||
"@vitejs/plugin-legacy": "^1.6.4",
|
"@vitejs/plugin-legacy": "^2.1.0",
|
||||||
"@vitejs/plugin-vue": "^2.0.1",
|
"@vitejs/plugin-vue": "^3.1.0",
|
||||||
"@vitejs/plugin-vue-jsx": "^1.3.3",
|
"@vitejs/plugin-vue-jsx": "^2.0.1",
|
||||||
"@vue/eslint-config-prettier": "6.0.0",
|
"@vue/eslint-config-prettier": "^7.0.0",
|
||||||
"@vue/eslint-config-typescript": "7.0.0",
|
"@vue/eslint-config-typescript": "^10.0.0",
|
||||||
"@zougt/vite-plugin-theme-preprocessor": "^1.4.4",
|
"@vue/runtime-core": "^3.2.39",
|
||||||
"autoprefixer": "10.2.4",
|
"autoprefixer": "^10.4.8",
|
||||||
"cross-env": "7.0.3",
|
"cloc": "^2.10.0",
|
||||||
"eslint": "7.30.0",
|
"cssnano": "^5.1.13",
|
||||||
"eslint-plugin-prettier": "3.4.0",
|
"eslint": "^8.8.0",
|
||||||
"eslint-plugin-vue": "7.17.0",
|
"eslint-plugin-prettier": "^4.0.0",
|
||||||
"husky": "7.0.2",
|
"eslint-plugin-vue": "^8.4.1",
|
||||||
|
"font-awesome": "^4.7.0",
|
||||||
|
"husky": "^7.0.4",
|
||||||
"lint-staged": "11.1.2",
|
"lint-staged": "11.1.2",
|
||||||
"postcss": "8.2.6",
|
"picocolors": "^1.0.0",
|
||||||
"postcss-import": "14.0.0",
|
"postcss": "^8.4.16",
|
||||||
"prettier": "2.3.2",
|
"postcss-html": "^1.5.0",
|
||||||
|
"postcss-import": "^15.0.0",
|
||||||
|
"postcss-scss": "^4.0.4",
|
||||||
|
"prettier": "^2.5.1",
|
||||||
"pretty-quick": "3.1.1",
|
"pretty-quick": "3.1.1",
|
||||||
"rimraf": "3.0.2",
|
"rimraf": "3.0.2",
|
||||||
"sass": "^1.45.0",
|
"rollup-plugin-visualizer": "^5.8.1",
|
||||||
"sass-loader": "^12.3.0",
|
"sass": "^1.53.0",
|
||||||
"stylelint": "13.13.1",
|
"sass-loader": "^13.0.2",
|
||||||
"stylelint-config-prettier": "8.0.2",
|
"stylelint": "^14.3.0",
|
||||||
"stylelint-config-standard": "22.0.0",
|
"stylelint-config-html": "^1.0.0",
|
||||||
"stylelint-order": "4.1.0",
|
"stylelint-config-prettier": "^9.0.3",
|
||||||
"typescript": "^4.5.5",
|
"stylelint-config-recommended": "^6.0.0",
|
||||||
"unplugin-element-plus": "^0.2.0",
|
"stylelint-config-standard": "^24.0.0",
|
||||||
"vite": "^2.7.13",
|
"stylelint-order": "^5.0.0",
|
||||||
"vite-plugin-live-reload": "^2.1.0",
|
"tailwindcss": "^3.1.8",
|
||||||
|
"terser": "^5.15.0",
|
||||||
|
"typescript": "^4.7.4",
|
||||||
|
"unplugin-vue-define-options": "0.7.3",
|
||||||
|
"vite": "^3.1.0",
|
||||||
"vite-plugin-mock": "^2.9.6",
|
"vite-plugin-mock": "^2.9.6",
|
||||||
"vite-plugin-remove-console": "^0.0.6",
|
"vite-plugin-remove-console": "^1.1.0",
|
||||||
"vite-plugin-style-import": "^1.4.1",
|
"vite-svg-loader": "^3.4.0",
|
||||||
"vite-plugin-windicss": "^1.6.3",
|
"vue-eslint-parser": "^8.2.0",
|
||||||
"vite-svg-loader": "2.2.0",
|
"vue-tsc": "^0.40.7"
|
||||||
"vue-eslint-parser": "7.10.0",
|
|
||||||
"windicss": "^3.4.3"
|
|
||||||
},
|
},
|
||||||
"repository": "git@github.com:xiaoxian521/vue-pure-admin.git",
|
"pnpm": {
|
||||||
|
"peerDependencyRules": {
|
||||||
|
"ignoreMissing": [
|
||||||
|
"rollup",
|
||||||
|
"webpack"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"repository": "git@github.com:xiaoxian521/pure-admin-thin.git",
|
||||||
"author": "xiaoxian521",
|
"author": "xiaoxian521",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
}
|
}
|
||||||
|
|||||||
4923
pnpm-lock.yaml
generated
@@ -1,3 +1,8 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
plugins: [require("autoprefixer"), require("postcss-import")]
|
plugins: {
|
||||||
|
"postcss-import": {},
|
||||||
|
tailwindcss: {},
|
||||||
|
autoprefixer: {},
|
||||||
|
...(process.env.NODE_ENV === "production" ? { cssnano: {} } : {})
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,18 +0,0 @@
|
|||||||
@font-face {
|
|
||||||
font-family: "iconfont"; /* project id 1098500 */
|
|
||||||
src: url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.eot");
|
|
||||||
src: url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.eot?#iefix")
|
|
||||||
format("embedded-opentype"),
|
|
||||||
url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.woff2") format("woff2"),
|
|
||||||
url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.woff") format("woff"),
|
|
||||||
url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.ttf") format("truetype"),
|
|
||||||
url("//at.alicdn.com/t/font_1098500_3d6un9zwltz.svg#iconfont") format("svg");
|
|
||||||
}
|
|
||||||
|
|
||||||
.iconfont {
|
|
||||||
font-family: "iconfont" !important;
|
|
||||||
font-size: 16px;
|
|
||||||
font-style: normal;
|
|
||||||
-webkit-font-smoothing: antialiased;
|
|
||||||
-moz-osx-font-smoothing: grayscale;
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"Version": "2.8.5",
|
"Version": "3.5.0",
|
||||||
"Title": "PureAdmin",
|
"Title": "PureAdmin",
|
||||||
"FixedHeader": true,
|
"FixedHeader": true,
|
||||||
"HiddenSideBar": false,
|
"HiddenSideBar": false,
|
||||||
@@ -15,13 +15,5 @@
|
|||||||
"SidebarStatus": true,
|
"SidebarStatus": true,
|
||||||
"EpThemeColor": "#409EFF",
|
"EpThemeColor": "#409EFF",
|
||||||
"ShowLogo": true,
|
"ShowLogo": true,
|
||||||
"ShowModel": "smart",
|
"ShowModel": "smart"
|
||||||
"MapConfigure": {
|
|
||||||
"amapKey": "97b3248d1553172e81f168cf94ea667e",
|
|
||||||
"options": {
|
|
||||||
"resizeEnable": true,
|
|
||||||
"center": [113.6401, 34.72468],
|
|
||||||
"zoom": 12
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,10 @@
|
|||||||
import { http } from "../utils/http";
|
import { http } from "../utils/http";
|
||||||
|
|
||||||
export const getAsyncRoutes = (params?: object) => {
|
type Result = {
|
||||||
return http.request("get", "/getAsyncRoutes", { params });
|
code: number;
|
||||||
|
info: Array<any>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAsyncRoutes = (params?: object) => {
|
||||||
|
return http.request<Result>("get", "/getAsyncRoutes", { params });
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
import { http } from "../utils/http";
|
import { http } from "../utils/http";
|
||||||
|
|
||||||
interface userType extends Promise<any> {
|
type Result = {
|
||||||
svg?: string;
|
svg?: string;
|
||||||
code?: number;
|
code?: number;
|
||||||
info?: object;
|
info?: object;
|
||||||
}
|
};
|
||||||
|
|
||||||
// 获取验证码
|
// 获取验证码
|
||||||
export const getVerify = (): userType => {
|
export const getVerify = () => {
|
||||||
return http.request("get", "/captcha");
|
return http.request<Result>("get", "/captcha");
|
||||||
};
|
};
|
||||||
|
|
||||||
// 登录
|
// 登录
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 680 KiB |
|
Before Width: | Height: | Size: 96 KiB |
|
Before Width: | Height: | Size: 4.7 KiB |
|
Before Width: | Height: | Size: 1.4 KiB |
|
Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 20 KiB |
|
Before Width: | Height: | Size: 17 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" data-name="Layer 1" width="500" height="380" viewBox="0 0 896 529.1129" xmlns:xlink="http://www.w3.org/1999/xlink"><path d="M832.06729,623.22778s-26.37759,9.89441-38.806,32.94348S787.06,706.69574,787.06,706.69574s26.37759-9.89447,38.806-32.94348S832.06729,623.22778,832.06729,623.22778Z" transform="translate(-158 -185.8871)" fill="#3f3d56"/><path d="M867.5,657.59637s-8.64182,26.814-31.0802,40.31373-50.17651,8.57293-50.17651,8.57293,8.64175-26.81408,31.08017-40.31378S867.5,657.59637,867.5,657.59637Z" transform="translate(-158 -185.8871)" fill="#5392f0"/><rect y="527.1129" width="896" height="2" fill="#2f2e41"/><path d="M519.87238,620.97461a95.44448,95.44448,0,0,1-35.748-14.44629L485.306,604.915a93.36283,93.36283,0,0,0,34.999,14.10547c18.93164,3.40137,47.26075,1.73144,74.707-25.52735,53.41358-53.04785,104.39307-58.39062,104.90186-58.43847l.18652,1.99219c-.50146.04687-50.76806,5.31738-103.67822,57.86621-21.61328,21.46386-43.792,27.40234-61.71777,27.40234A83.49962,83.49962,0,0,1,519.87238,620.97461Z" transform="translate(-158 -185.8871)" fill="#2f2e41"/><circle cx="515.15271" cy="381.1129" r="12" fill="#2f2e41"/><circle cx="430.15271" cy="437.1129" r="12" fill="#2f2e41"/><path d="M841.5,714s-17.46191-5.41315-52.26129-10.84192L790,692.5c6-60-34-150-34-150a401.561,401.561,0,0,1,21.4693,139.0246C772.13214,672.2124,761.82056,662.16638,742,656c0,0,25.77765,22.106,33.15918,45.10175a997.84042,997.84042,0,0,0-102.02258-8.21589L682,672.5l-17,17s-7-51-22-53l11,50s-13-10-16-9l7.39746,14.79486c-49.819-.51654-109.08453,1.7356-177.76581,8.95227L476,682l-17,17s-7-51-22-53l11,50s-13-10-16-9l8.64288,17.28583Q406.9763,708.2897,370.5,714Z" transform="translate(-158 -185.8871)" fill="#3f3d56"/><path d="M565.64813,230.37817c-10.89964,11.74783,17.59745,40.25959,17.59745,40.25959s-57.70662,9.73051-53.12783,9.14083,2.20622-49.13151,2.20622-49.13151S576.54777,218.63035,565.64813,230.37817Z" transform="translate(-158 -185.8871)" fill="#a0616a"/><path d="M605.81236,356.10945l-50.139,25.6141-27.22969,15.6059s-32.09862,40.43116-38.08709,64.39234,25.92963,68.247,29.54371,72.82286a54.36088,54.36088,0,0,1,4.98908,7.42355c1.24727,1.85589,12.02944-.541,23.80342-3.06554S547.13,518.93875,547.13,518.93875s-15.02732-38.39505-16.14686-39.25912c-1.04554-.807-4.60093-7.44631-2.04309-10.35234a25.94993,25.94993,0,0,0,5.44489-8.89825,30.09064,30.09064,0,0,1,4.18709-7.94151s45.361-36.83645,59.52776-49.37835,51.82952-4.65839,51.82952-4.65839-17.78167,68.20027-22.22979,72.80616-4.929,8.70085-2.91535,16.50759,28.28157.39078,28.28157.39078L662.766,461.6996s15.74879-34.2925,24.29946-69.67451c4.27533-17.691-3.88828-28.23462-13.12073-34.35549a41.39094,41.39094,0,0,0-30.02983-5.97766l-46.34848,8.1308,32.14706-13.84923Z" transform="translate(-158 -185.8871)" fill="#2f2e41"/><path d="M420.87777,290.19133,361.02366,271.685s-24.179-31.16689-12.78824-36.6669,25.65172,26.94419,25.65172,26.94419l41.686,2.69751Z" transform="translate(-158 -185.8871)" fill="#a0616a"/><path d="M672.49431,257.78673l53.2121-33.06768s15.49333-36.27612,3.0807-38.71059-17.98787,32.56435-17.98787,32.56435l-39.64232,13.17143Z" transform="translate(-158 -185.8871)" fill="#a0616a"/><path d="M682.45318,220.40023l1.01427,39.19147-89.68779,16.025c13.19231,28.22441,9.84118,60.34675,43.04725,74.4259L524.9027,404.78717c4.9871-43.03806-15.81748-75.456-35.263-115.75876-23.68547-8.58589-51.19594-2.29078-80.33649,10.34619l-5.237-40.66416,123.87841-8.896,20.34848,7.77932,21.81842-9.17677C602.17891,238.88953,648.22076,220.77584,682.45318,220.40023Z" transform="translate(-158 -185.8871)" fill="#5392f0"/><path d="M626.64006,486.51727c-2.72,2.36681-5.25213,21.84984-5.34982,28.92023s9.21178,8.89624,14.29855,9.2494,4.47816,3.45631,7.83678,6.04854,14.39625,2.179,28.89019-2.71238-9.75274-20.92568-11.86409-21.662-11-22.78156-11-22.78156S629.36,484.15046,626.64006,486.51727Z" transform="translate(-158 -185.8871)" fill="#2f2e41"/><path d="M547.368,531.00717c3.23089,1.60043,10.61681,19.80614,12.50274,26.62107s-6.65716,10.93994-11.48848,12.5704-3.45631,4.47816-6.04855,7.83678-13.3744,5.75546-28.63472,4.696,4.13258-22.71391,5.98847-23.96118,4.86893-24.82526,4.86893-24.82526S544.13715,529.40674,547.368,531.00717Z" transform="translate(-158 -185.8871)" fill="#2f2e41"/><circle cx="389.47074" cy="35.42904" r="23.99585" fill="#a0616a"/><path d="M519.73448,218.90923a22.82668,22.82668,0,0,1-.83378-18.59281c2.35891-5.8153,7.59174-11.65569,18.87309-13.4,24.61387-3.80572,37.71267,13.43543,37.02452,19.07449s-3.99294,19.27051-3.99294,19.27051,1.47587-12.90619-4.85883-13.362-30.90178-2.37835-37.12217,4.145a14.23268,14.23268,0,0,0-3.71042,13.82977Z" transform="translate(-158 -185.8871)" fill="#2f2e41"/></svg>
|
|
||||||
|
Before Width: | Height: | Size: 4.6 KiB |
|
Before Width: | Height: | Size: 11 KiB |
|
Before Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 29 KiB |
|
Before Width: | Height: | Size: 11 KiB |
1
src/assets/status/403.svg
Normal file
|
After Width: | Height: | Size: 12 KiB |
1
src/assets/status/404.svg
Normal file
|
After Width: | Height: | Size: 13 KiB |
1
src/assets/status/500.svg
Normal file
|
After Width: | Height: | Size: 15 KiB |
1
src/assets/svg/enter_outlined.svg
Normal file
@@ -0,0 +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>
|
||||||
|
After Width: | Height: | Size: 448 B |
1
src/assets/svg/mdi_keyboard_esc.svg
Normal file
@@ -0,0 +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>
|
||||||
|
After Width: | Height: | Size: 477 B |
@@ -1,103 +1,12 @@
|
|||||||
import { App, defineComponent } from "vue";
|
|
||||||
import icon from "./src/Icon.vue";
|
|
||||||
import { FontAwesomeIcon } from "@fortawesome/vue-fontawesome";
|
|
||||||
import iconifyIconOffline from "./src/iconifyIconOffline";
|
import iconifyIconOffline from "./src/iconifyIconOffline";
|
||||||
import iconifyIconOnline from "./src/iconifyIconOnline";
|
import iconifyIconOnline from "./src/iconifyIconOnline";
|
||||||
|
import fontIcon from "./src/iconfont";
|
||||||
|
|
||||||
/**
|
/** 离线图标组件 */
|
||||||
* find icon component
|
const IconifyIconOffline = iconifyIconOffline;
|
||||||
* @param icon icon图标
|
/** 在线图标组件 */
|
||||||
* @returns component
|
const IconifyIconOnline = iconifyIconOnline;
|
||||||
*/
|
/** iconfont组件 */
|
||||||
export function findIconReg(icon: string) {
|
const FontIcon = fontIcon;
|
||||||
// fontawesome4
|
|
||||||
const fa4Reg = /^fa-/;
|
|
||||||
// fontawesome5+
|
|
||||||
const fa5Reg = /^FA-/;
|
|
||||||
// iconfont
|
|
||||||
const iFReg = /^IF-/;
|
|
||||||
// remixicon
|
|
||||||
const riReg = /^RI-/;
|
|
||||||
// typeof icon === "function" 属于SVG
|
|
||||||
if (fa5Reg.test(icon)) {
|
|
||||||
const text = icon.split(fa5Reg)[1];
|
|
||||||
return findIcon(
|
|
||||||
text.slice(0, text.indexOf(" ") == -1 ? text.length : text.indexOf(" ")),
|
|
||||||
"FA",
|
|
||||||
text.slice(text.indexOf(" ") + 1, text.length)
|
|
||||||
);
|
|
||||||
} else if (fa4Reg.test(icon)) {
|
|
||||||
return findIcon(icon.split(fa4Reg)[1], "fa");
|
|
||||||
} else if (iFReg.test(icon)) {
|
|
||||||
return findIcon(icon.split(iFReg)[1], "IF");
|
|
||||||
} else if (typeof icon === "function") {
|
|
||||||
return findIcon(icon, "SVG");
|
|
||||||
} else if (riReg.test(icon)) {
|
|
||||||
return findIcon(icon.split(riReg)[1], "RI");
|
|
||||||
} else {
|
|
||||||
return findIcon(icon, "EL");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 支持fontawesome、iconfont、remixicon、element-plus/icons、自定义svg
|
export { IconifyIconOffline, IconifyIconOnline, FontIcon };
|
||||||
export function findIcon(icon: String, type = "EL", property?: string) {
|
|
||||||
if (type === "FA") {
|
|
||||||
return defineComponent({
|
|
||||||
name: "FaIcon",
|
|
||||||
setup() {
|
|
||||||
return { icon, property };
|
|
||||||
},
|
|
||||||
components: { FontAwesomeIcon },
|
|
||||||
template: `<font-awesome-icon :icon="icon" v-bind:[property]="true" />`
|
|
||||||
});
|
|
||||||
} else if (type === "fa") {
|
|
||||||
return defineComponent({
|
|
||||||
name: "faIcon",
|
|
||||||
data() {
|
|
||||||
return { icon: `fa ${icon}` };
|
|
||||||
},
|
|
||||||
template: `<i :class="icon" />`
|
|
||||||
});
|
|
||||||
} else if (type === "IF") {
|
|
||||||
return defineComponent({
|
|
||||||
name: "IfIcon",
|
|
||||||
data() {
|
|
||||||
return { icon: `iconfont ${icon}` };
|
|
||||||
},
|
|
||||||
template: `<i :class="icon" />`
|
|
||||||
});
|
|
||||||
} else if (type === "RI") {
|
|
||||||
return defineComponent({
|
|
||||||
name: "RiIcon",
|
|
||||||
data() {
|
|
||||||
return { icon: `ri-${icon}` };
|
|
||||||
},
|
|
||||||
template: `<i :class="icon" />`
|
|
||||||
});
|
|
||||||
} else if (type === "EL") {
|
|
||||||
return defineComponent({
|
|
||||||
name: "ElIcon",
|
|
||||||
data() {
|
|
||||||
return { icon };
|
|
||||||
},
|
|
||||||
template: `<IconifyIconOffline :icon="icon" />`
|
|
||||||
});
|
|
||||||
} else if (type === "SVG") {
|
|
||||||
return icon;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const Icon = Object.assign(icon, {
|
|
||||||
install(app: App) {
|
|
||||||
app.component(icon.name, icon);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
export const IconifyIconOffline = iconifyIconOffline;
|
|
||||||
export const IconifyIconOnline = iconifyIconOnline;
|
|
||||||
|
|
||||||
export default {
|
|
||||||
Icon,
|
|
||||||
IconifyIconOffline,
|
|
||||||
IconifyIconOnline
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -1,97 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
export default {
|
|
||||||
name: "Icon"
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref, computed } from "vue";
|
|
||||||
|
|
||||||
const props = defineProps({
|
|
||||||
content: {
|
|
||||||
type: String,
|
|
||||||
default: ""
|
|
||||||
},
|
|
||||||
size: {
|
|
||||||
type: Number,
|
|
||||||
default: 18
|
|
||||||
},
|
|
||||||
width: {
|
|
||||||
type: Number,
|
|
||||||
default: 20
|
|
||||||
},
|
|
||||||
height: {
|
|
||||||
type: Number,
|
|
||||||
default: 20
|
|
||||||
},
|
|
||||||
color: {
|
|
||||||
type: String,
|
|
||||||
default: ""
|
|
||||||
},
|
|
||||||
svg: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
(e: "click"): void;
|
|
||||||
}>();
|
|
||||||
|
|
||||||
let text = ref("");
|
|
||||||
|
|
||||||
let className = computed(() => {
|
|
||||||
if (props.content.indexOf("fa-") > -1) {
|
|
||||||
return props.content.indexOf("fa ") === 0
|
|
||||||
? props.content
|
|
||||||
: ["fa", props.content];
|
|
||||||
} else if (props.content.indexOf("el-icon-") > -1) {
|
|
||||||
return props.content;
|
|
||||||
} else if (props.content.indexOf("#") > -1) {
|
|
||||||
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
|
||||||
text.value = props.content;
|
|
||||||
return "iconfont";
|
|
||||||
} else {
|
|
||||||
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
|
||||||
text.value = props.content;
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let iconStyle = computed(() => {
|
|
||||||
return (
|
|
||||||
"font-size: " +
|
|
||||||
props.size +
|
|
||||||
"px; color: " +
|
|
||||||
props.color +
|
|
||||||
"; width: " +
|
|
||||||
props.width +
|
|
||||||
"px; height: " +
|
|
||||||
props.height +
|
|
||||||
"px; font-style: normal;"
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
const clickHandle = () => {
|
|
||||||
emit("click");
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<i
|
|
||||||
v-if="!props.svg"
|
|
||||||
:class="className"
|
|
||||||
:style="iconStyle"
|
|
||||||
v-html="text"
|
|
||||||
@click="clickHandle"
|
|
||||||
></i>
|
|
||||||
<svg
|
|
||||||
class="icon-svg"
|
|
||||||
v-if="props.svg"
|
|
||||||
aria-hidden="true"
|
|
||||||
:style="iconStyle"
|
|
||||||
@click="clickHandle"
|
|
||||||
>
|
|
||||||
<use :xlink:href="`#${props.content}`" />
|
|
||||||
</svg>
|
|
||||||
</template>
|
|
||||||
49
src/components/ReIcon/src/hooks.ts
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import { iconType } from "./types";
|
||||||
|
import { h, defineComponent, Component } from "vue";
|
||||||
|
import { IconifyIconOnline, IconifyIconOffline, FontIcon } from "../index";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 支持fontawesome4、5+、iconfont、remixicon、element-plus的icons、自定义svg
|
||||||
|
* @param icon 必传 图标
|
||||||
|
* @param attrs 可选 iconType 属性
|
||||||
|
* @returns Component
|
||||||
|
*/
|
||||||
|
export function useRenderIcon(icon: any, attrs?: iconType): Component {
|
||||||
|
// iconfont
|
||||||
|
const ifReg = /^IF-/;
|
||||||
|
// typeof icon === "function" 属于SVG
|
||||||
|
if (ifReg.test(icon)) {
|
||||||
|
// iconfont
|
||||||
|
const name = icon.split(ifReg)[1];
|
||||||
|
const iconName = name.slice(
|
||||||
|
0,
|
||||||
|
name.indexOf(" ") == -1 ? name.length : name.indexOf(" ")
|
||||||
|
);
|
||||||
|
const iconType = name.slice(name.indexOf(" ") + 1, name.length);
|
||||||
|
return defineComponent({
|
||||||
|
name: "FontIcon",
|
||||||
|
render() {
|
||||||
|
return h(FontIcon, {
|
||||||
|
icon: iconName,
|
||||||
|
iconType,
|
||||||
|
...attrs
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (typeof icon === "function" || typeof icon?.render === "function") {
|
||||||
|
// svg
|
||||||
|
return icon;
|
||||||
|
} else {
|
||||||
|
return defineComponent({
|
||||||
|
name: "Icon",
|
||||||
|
render() {
|
||||||
|
const IconifyIcon =
|
||||||
|
attrs && attrs["online"] ? IconifyIconOnline : IconifyIconOffline;
|
||||||
|
return h(IconifyIcon, {
|
||||||
|
icon: icon,
|
||||||
|
...attrs
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
48
src/components/ReIcon/src/iconfont.ts
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
import { h, defineComponent } from "vue";
|
||||||
|
|
||||||
|
// 封装iconfont组件,默认`font-class`引用模式,支持`unicode`引用、`font-class`引用、`symbol`引用 (https://www.iconfont.cn/help/detail?spm=a313x.7781069.1998910419.20&helptype=code)
|
||||||
|
export default defineComponent({
|
||||||
|
name: "FontIcon",
|
||||||
|
props: {
|
||||||
|
icon: {
|
||||||
|
type: String,
|
||||||
|
default: ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
render() {
|
||||||
|
const attrs = this.$attrs;
|
||||||
|
if (Object.keys(attrs).includes("uni") || attrs?.iconType === "uni") {
|
||||||
|
return h(
|
||||||
|
"i",
|
||||||
|
{
|
||||||
|
class: "iconfont",
|
||||||
|
...attrs
|
||||||
|
},
|
||||||
|
this.icon
|
||||||
|
);
|
||||||
|
} else if (
|
||||||
|
Object.keys(attrs).includes("svg") ||
|
||||||
|
attrs?.iconType === "svg"
|
||||||
|
) {
|
||||||
|
return h(
|
||||||
|
"svg",
|
||||||
|
{
|
||||||
|
class: "icon-svg",
|
||||||
|
"aria-hidden": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
default: () => [
|
||||||
|
h("use", {
|
||||||
|
"xlink:href": `#${this.icon}`
|
||||||
|
})
|
||||||
|
]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return h("i", {
|
||||||
|
class: `iconfont ${this.icon}`,
|
||||||
|
...attrs
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -1,54 +1,53 @@
|
|||||||
import { h, defineComponent } from "vue";
|
import { h, defineComponent } from "vue";
|
||||||
import { Icon as IconifyIcon, addIcon } from "@iconify/vue/dist/offline";
|
import { Icon as IconifyIcon, addIcon } from "@iconify/vue/dist/offline";
|
||||||
|
|
||||||
|
// element-plus icon
|
||||||
import Check from "@iconify-icons/ep/check";
|
import Check from "@iconify-icons/ep/check";
|
||||||
import Menu from "@iconify-icons/ep/menu";
|
|
||||||
import HomeFilled from "@iconify-icons/ep/home-filled";
|
import HomeFilled from "@iconify-icons/ep/home-filled";
|
||||||
import SetUp from "@iconify-icons/ep/set-up";
|
|
||||||
import Edit from "@iconify-icons/ep/edit";
|
|
||||||
import Setting from "@iconify-icons/ep/setting";
|
import Setting from "@iconify-icons/ep/setting";
|
||||||
import Lollipop from "@iconify-icons/ep/lollipop";
|
import Lollipop from "@iconify-icons/ep/lollipop";
|
||||||
import Link from "@iconify-icons/ep/link";
|
|
||||||
import Position from "@iconify-icons/ep/position";
|
|
||||||
import Histogram from "@iconify-icons/ep/histogram";
|
|
||||||
import RefreshRight from "@iconify-icons/ep/refresh-right";
|
import RefreshRight from "@iconify-icons/ep/refresh-right";
|
||||||
import ArrowDown from "@iconify-icons/ep/arrow-down";
|
import ArrowDown from "@iconify-icons/ep/arrow-down";
|
||||||
import Close from "@iconify-icons/ep/close";
|
|
||||||
import CloseBold from "@iconify-icons/ep/close-bold";
|
import CloseBold from "@iconify-icons/ep/close-bold";
|
||||||
import Bell from "@iconify-icons/ep/bell";
|
import Bell from "@iconify-icons/ep/bell";
|
||||||
import Guide from "@iconify-icons/ep/guide";
|
import Search from "@iconify-icons/ep/search";
|
||||||
import User from "@iconify-icons/ep/user";
|
|
||||||
import Iphone from "@iconify-icons/ep/iphone";
|
|
||||||
import Location from "@iconify-icons/ep/location";
|
|
||||||
import Tickets from "@iconify-icons/ep/tickets";
|
|
||||||
import OfficeBuilding from "@iconify-icons/ep/office-building";
|
|
||||||
import Notebook from "@iconify-icons/ep/notebook";
|
|
||||||
addIcon("check", Check);
|
addIcon("check", Check);
|
||||||
addIcon("menu", Menu);
|
|
||||||
addIcon("home-filled", HomeFilled);
|
addIcon("home-filled", HomeFilled);
|
||||||
addIcon("set-up", SetUp);
|
|
||||||
addIcon("edit", Edit);
|
|
||||||
addIcon("setting", Setting);
|
addIcon("setting", Setting);
|
||||||
addIcon("lollipop", Lollipop);
|
addIcon("lollipop", Lollipop);
|
||||||
addIcon("link", Link);
|
|
||||||
addIcon("position", Position);
|
|
||||||
addIcon("histogram", Histogram);
|
|
||||||
addIcon("refresh-right", RefreshRight);
|
addIcon("refresh-right", RefreshRight);
|
||||||
addIcon("arrow-down", ArrowDown);
|
addIcon("arrow-down", ArrowDown);
|
||||||
addIcon("close", Close);
|
|
||||||
addIcon("close-bold", CloseBold);
|
addIcon("close-bold", CloseBold);
|
||||||
addIcon("bell", Bell);
|
addIcon("bell", Bell);
|
||||||
addIcon("guide", Guide);
|
addIcon("search", Search);
|
||||||
addIcon("user", User);
|
|
||||||
addIcon("iphone", Iphone);
|
|
||||||
addIcon("location", Location);
|
|
||||||
addIcon("tickets", Tickets);
|
|
||||||
addIcon("office-building", OfficeBuilding);
|
|
||||||
addIcon("notebook", Notebook);
|
|
||||||
|
|
||||||
// Iconify Icon在Vue里离线使用(用于内网环境)
|
// remixicon
|
||||||
// https://docs.iconify.design/icon-components/vue/offline.html
|
import ArrowRightSLine from "@iconify-icons/ri/arrow-right-s-line";
|
||||||
|
import ArrowLeftSLine from "@iconify-icons/ri/arrow-left-s-line";
|
||||||
|
import LogoutCircleRLine from "@iconify-icons/ri/logout-circle-r-line";
|
||||||
|
import InformationLine from "@iconify-icons/ri/information-line";
|
||||||
|
import ArrowUpLine from "@iconify-icons/ri/arrow-up-line";
|
||||||
|
import ArrowDownLine from "@iconify-icons/ri/arrow-down-line";
|
||||||
|
import Bookmark2Line from "@iconify-icons/ri/bookmark-2-line";
|
||||||
|
import User from "@iconify-icons/ri/user-3-fill";
|
||||||
|
import Lock from "@iconify-icons/ri/lock-fill";
|
||||||
|
import MenuUnfold from "@iconify-icons/ri/menu-unfold-fill";
|
||||||
|
import MenuFold from "@iconify-icons/ri/menu-fold-fill";
|
||||||
|
addIcon("arrow-right-s-line", ArrowRightSLine);
|
||||||
|
addIcon("arrow-left-s-line", ArrowLeftSLine);
|
||||||
|
addIcon("logout-circle-r-line", LogoutCircleRLine);
|
||||||
|
addIcon("information-line", InformationLine);
|
||||||
|
addIcon("arrow-up-line", ArrowUpLine);
|
||||||
|
addIcon("arrow-down-line", ArrowDownLine);
|
||||||
|
addIcon("bookmark-2-line", Bookmark2Line);
|
||||||
|
addIcon("user", User);
|
||||||
|
addIcon("lock", Lock);
|
||||||
|
addIcon("menu-unfold", MenuUnfold);
|
||||||
|
addIcon("menu-fold", MenuFold);
|
||||||
|
|
||||||
|
// Iconify Icon在Vue里离线使用(用于内网环境)https://docs.iconify.design/icon-components/vue/offline.html
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: "IconifyIcon",
|
name: "IconifyIconOffline",
|
||||||
components: { IconifyIcon },
|
components: { IconifyIcon },
|
||||||
props: {
|
props: {
|
||||||
icon: {
|
icon: {
|
||||||
@@ -57,10 +56,15 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
|
const attrs = this.$attrs;
|
||||||
return h(
|
return h(
|
||||||
IconifyIcon,
|
IconifyIcon,
|
||||||
{
|
{
|
||||||
icon: `${this.icon}`
|
icon: `${this.icon}`,
|
||||||
|
style: attrs?.style
|
||||||
|
? Object.assign(attrs.style, { outline: "none" })
|
||||||
|
: { outline: "none" },
|
||||||
|
...attrs
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
default: () => []
|
default: () => []
|
||||||
|
|||||||
@@ -2,25 +2,25 @@ import { h, defineComponent } from "vue";
|
|||||||
import { Icon as IconifyIcon } from "@iconify/vue";
|
import { Icon as IconifyIcon } from "@iconify/vue";
|
||||||
|
|
||||||
// Iconify Icon在Vue里在线使用(用于外网环境)
|
// Iconify Icon在Vue里在线使用(用于外网环境)
|
||||||
// https://docs.iconify.design/icon-components/vue/offline.html
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: "IconifyIcon",
|
name: "IconifyIconOnline",
|
||||||
components: { IconifyIcon },
|
components: { IconifyIcon },
|
||||||
props: {
|
props: {
|
||||||
icon: {
|
icon: {
|
||||||
type: String,
|
type: String,
|
||||||
default: ""
|
default: ""
|
||||||
},
|
|
||||||
type: {
|
|
||||||
type: String,
|
|
||||||
default: "ep:"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
|
const attrs = this.$attrs;
|
||||||
return h(
|
return h(
|
||||||
IconifyIcon,
|
IconifyIcon,
|
||||||
{
|
{
|
||||||
icon: `${this.type}${this.icon}`
|
icon: `${this.icon}`,
|
||||||
|
style: attrs?.style
|
||||||
|
? Object.assign(attrs.style, { outline: "none" })
|
||||||
|
: { outline: "none" },
|
||||||
|
...attrs
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
default: () => []
|
default: () => []
|
||||||
|
|||||||
20
src/components/ReIcon/src/types.ts
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
export interface iconType {
|
||||||
|
// iconify (https://docs.iconify.design/icon-components/vue/#properties)
|
||||||
|
inline?: boolean;
|
||||||
|
width?: string | number;
|
||||||
|
height?: string | number;
|
||||||
|
horizontalFlip?: boolean;
|
||||||
|
verticalFlip?: boolean;
|
||||||
|
flip?: string;
|
||||||
|
rotate?: number | string;
|
||||||
|
color?: string;
|
||||||
|
horizontalAlign?: boolean;
|
||||||
|
verticalAlign?: boolean;
|
||||||
|
align?: string;
|
||||||
|
online?: boolean;
|
||||||
|
onLoad?: Function;
|
||||||
|
includes?: Function;
|
||||||
|
|
||||||
|
// all icon
|
||||||
|
style?: object;
|
||||||
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { Directive } from "vue";
|
import { Directive, type DirectiveBinding, type VNode } from "vue";
|
||||||
import type { DirectiveBinding, VNode } from "vue";
|
|
||||||
import elementResizeDetectorMaker from "element-resize-detector";
|
import elementResizeDetectorMaker from "element-resize-detector";
|
||||||
import type { Erd } from "element-resize-detector";
|
import type { Erd } from "element-resize-detector";
|
||||||
|
import { optimizeFps } from "@pureadmin/utils";
|
||||||
import { emitter } from "/@/utils/mitt";
|
import { emitter } from "/@/utils/mitt";
|
||||||
|
|
||||||
const erd: Erd = elementResizeDetectorMaker({
|
const erd: Erd = elementResizeDetectorMaker({
|
||||||
@@ -14,7 +14,9 @@ export const resize: Directive = {
|
|||||||
const width = elem.offsetWidth;
|
const width = elem.offsetWidth;
|
||||||
const height = elem.offsetHeight;
|
const height = elem.offsetHeight;
|
||||||
if (binding?.instance) {
|
if (binding?.instance) {
|
||||||
emitter.emit("resize", { detail: { width, height } });
|
optimizeFps(() => {
|
||||||
|
emitter.emit("resize", { detail: { width, height } });
|
||||||
|
})();
|
||||||
} else {
|
} else {
|
||||||
vnode.el.dispatchEvent(
|
vnode.el.dispatchEvent(
|
||||||
new CustomEvent("resize", { detail: { width, height } })
|
new CustomEvent("resize", { detail: { width, height } })
|
||||||
|
|||||||
@@ -1,24 +1,18 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
import { useGlobal } from "@pureadmin/utils";
|
||||||
h,
|
|
||||||
ref,
|
|
||||||
computed,
|
|
||||||
Transition,
|
|
||||||
defineComponent,
|
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
|
||||||
import { RouterView } from "vue-router";
|
|
||||||
import backTop from "/@/assets/svg/back_top.svg?component";
|
import backTop from "/@/assets/svg/back_top.svg?component";
|
||||||
|
import { h, computed, Transition, defineComponent } from "vue";
|
||||||
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
fixedHeader: Boolean
|
fixedHeader: Boolean
|
||||||
});
|
});
|
||||||
const keepAlive: Boolean = ref(
|
|
||||||
getCurrentInstance().appContext.config.globalProperties.$config?.KeepAlive
|
const { $storage, $config } = useGlobal<GlobalPropertiesApi>();
|
||||||
);
|
|
||||||
const instance =
|
const keepAlive = computed(() => {
|
||||||
getCurrentInstance().appContext.app.config.globalProperties.$storage;
|
return $config?.KeepAlive;
|
||||||
|
});
|
||||||
|
|
||||||
const transitions = computed(() => {
|
const transitions = computed(() => {
|
||||||
return route => {
|
return route => {
|
||||||
@@ -27,11 +21,11 @@ const transitions = computed(() => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const hideTabs = computed(() => {
|
const hideTabs = computed(() => {
|
||||||
return instance?.configure.hideTabs;
|
return $storage?.configure.hideTabs;
|
||||||
});
|
});
|
||||||
|
|
||||||
const layout = computed(() => {
|
const layout = computed(() => {
|
||||||
return instance?.layout.layout === "vertical";
|
return $storage?.layout.layout === "vertical";
|
||||||
});
|
});
|
||||||
|
|
||||||
const getSectionStyle = computed(() => {
|
const getSectionStyle = computed(() => {
|
||||||
|
|||||||
@@ -1,138 +1,114 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed } from "vue";
|
import Search from "./search/index.vue";
|
||||||
import { useI18n } from "vue-i18n";
|
|
||||||
import { emitter } from "/@/utils/mitt";
|
|
||||||
import Notice from "./notice/index.vue";
|
import Notice from "./notice/index.vue";
|
||||||
|
import mixNav from "./sidebar/mixNav.vue";
|
||||||
import avatars from "/@/assets/avatars.jpg";
|
import avatars from "/@/assets/avatars.jpg";
|
||||||
import { transformI18n } from "/@/plugins/i18n";
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import Hamburger from "./sidebar/hamBurger.vue";
|
|
||||||
import { useRouter, useRoute } from "vue-router";
|
|
||||||
import { storageSession } from "/@/utils/storage";
|
|
||||||
import Breadcrumb from "./sidebar/breadCrumb.vue";
|
import Breadcrumb from "./sidebar/breadCrumb.vue";
|
||||||
import { useAppStoreHook } from "/@/store/modules/app";
|
import { deviceDetection } from "@pureadmin/utils";
|
||||||
import { unref, watch, getCurrentInstance } from "vue";
|
import topCollapse from "./sidebar/topCollapse.vue";
|
||||||
import { deviceDetection } from "/@/utils/deviceDetection";
|
|
||||||
import screenfull from "../components/screenfull/index.vue";
|
import screenfull from "../components/screenfull/index.vue";
|
||||||
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
import { useTranslationLang } from "../hooks/useTranslationLang";
|
||||||
import globalization from "/@/assets/svg/globalization.svg?component";
|
import globalization from "/@/assets/svg/globalization.svg?component";
|
||||||
|
|
||||||
const instance =
|
const {
|
||||||
getCurrentInstance().appContext.config.globalProperties.$storage;
|
layout,
|
||||||
const pureApp = useAppStoreHook();
|
device,
|
||||||
const router = useRouter();
|
logout,
|
||||||
const route = useRoute();
|
onPanel,
|
||||||
let usename = storageSession.getItem("info")?.username;
|
pureApp,
|
||||||
const { locale } = useI18n();
|
username,
|
||||||
|
avatarsStyle,
|
||||||
|
toggleSideBar,
|
||||||
|
getDropdownItemStyle,
|
||||||
|
getDropdownItemClass
|
||||||
|
} = useNav();
|
||||||
|
|
||||||
const getDropdownItemStyle = computed(() => {
|
const { t, locale, translationCh, translationEn } = useTranslationLang();
|
||||||
return t => {
|
|
||||||
return {
|
|
||||||
background: locale.value === t ? useEpThemeStoreHook().epThemeColor : "",
|
|
||||||
color: locale.value === t ? "#f4f4f5" : "#000"
|
|
||||||
};
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
watch(
|
|
||||||
() => locale.value,
|
|
||||||
() => {
|
|
||||||
//@ts-ignore
|
|
||||||
document.title = transformI18n(
|
|
||||||
//@ts-ignore
|
|
||||||
unref(route.meta.title),
|
|
||||||
unref(route.meta.i18n)
|
|
||||||
); // 动态title
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
// 退出登录
|
|
||||||
const logout = (): void => {
|
|
||||||
storageSession.removeItem("info");
|
|
||||||
router.push("/login");
|
|
||||||
};
|
|
||||||
|
|
||||||
function onPanel() {
|
|
||||||
emitter.emit("openPanel");
|
|
||||||
}
|
|
||||||
|
|
||||||
function toggleSideBar() {
|
|
||||||
pureApp.toggleSideBar();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 简体中文
|
|
||||||
function translationCh() {
|
|
||||||
instance.locale = { locale: "zh" };
|
|
||||||
locale.value = "zh";
|
|
||||||
}
|
|
||||||
|
|
||||||
// English
|
|
||||||
function translationEn() {
|
|
||||||
instance.locale = { locale: "en" };
|
|
||||||
locale.value = "en";
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="navbar">
|
<div
|
||||||
<Hamburger
|
class="navbar bg-[#fff] shadow-sm shadow-[rgba(0, 21, 41, 0.08)] dark:shadow-[#0d0d0d]"
|
||||||
:is-active="pureApp.sidebar.opened"
|
>
|
||||||
|
<topCollapse
|
||||||
|
v-if="device === 'mobile'"
|
||||||
class="hamburger-container"
|
class="hamburger-container"
|
||||||
|
:is-active="pureApp.sidebar.opened"
|
||||||
@toggleClick="toggleSideBar"
|
@toggleClick="toggleSideBar"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Breadcrumb class="breadcrumb-container" />
|
<Breadcrumb
|
||||||
|
v-if="layout !== 'mix' && device !== 'mobile'"
|
||||||
|
class="breadcrumb-container"
|
||||||
|
/>
|
||||||
|
|
||||||
<div class="vertical-header-right">
|
<mixNav v-if="layout === 'mix'" />
|
||||||
|
|
||||||
|
<div v-if="layout === 'vertical'" class="vertical-header-right">
|
||||||
|
<!-- 菜单搜索 -->
|
||||||
|
<Search />
|
||||||
<!-- 通知 -->
|
<!-- 通知 -->
|
||||||
<Notice id="header-notice" />
|
<Notice id="header-notice" />
|
||||||
<!-- 全屏 -->
|
<!-- 全屏 -->
|
||||||
<screenfull id="header-screenfull" v-show="!deviceDetection()" />
|
<screenfull id="header-screenfull" v-show="!deviceDetection()" />
|
||||||
<!-- 国际化 -->
|
<!-- 国际化 -->
|
||||||
<el-dropdown id="header-translation" trigger="click">
|
<el-dropdown id="header-translation" trigger="click">
|
||||||
<globalization />
|
<globalization
|
||||||
|
class="navbar-bg-hover w-[40px] h-[48px] p-[11px] cursor-pointer outline-none"
|
||||||
|
/>
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu class="translation">
|
<el-dropdown-menu class="translation">
|
||||||
<el-dropdown-item
|
<el-dropdown-item
|
||||||
:style="getDropdownItemStyle('zh')"
|
:style="getDropdownItemStyle(locale, 'zh')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
|
||||||
@click="translationCh"
|
@click="translationCh"
|
||||||
><IconifyIconOffline
|
>
|
||||||
|
<IconifyIconOffline
|
||||||
class="check-zh"
|
class="check-zh"
|
||||||
v-show="locale === 'zh'"
|
v-show="locale === 'zh'"
|
||||||
icon="check"
|
icon="check"
|
||||||
/>简体中文</el-dropdown-item
|
/>
|
||||||
>
|
简体中文
|
||||||
|
</el-dropdown-item>
|
||||||
<el-dropdown-item
|
<el-dropdown-item
|
||||||
:style="getDropdownItemStyle('en')"
|
:style="getDropdownItemStyle(locale, 'en')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
|
||||||
@click="translationEn"
|
@click="translationEn"
|
||||||
><el-icon class="check-en" v-show="locale === 'en'"
|
|
||||||
><IconifyIconOffline icon="check" /></el-icon
|
|
||||||
>English</el-dropdown-item
|
|
||||||
>
|
>
|
||||||
|
<span class="check-en" v-show="locale === 'en'">
|
||||||
|
<IconifyIconOffline icon="check" />
|
||||||
|
</span>
|
||||||
|
English
|
||||||
|
</el-dropdown-item>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
<!-- 退出登陆 -->
|
<!-- 退出登录 -->
|
||||||
<el-dropdown trigger="click">
|
<el-dropdown trigger="click">
|
||||||
<span class="el-dropdown-link">
|
<span class="el-dropdown-link navbar-bg-hover">
|
||||||
<img :src="avatars" />
|
<img v-if="avatars" :src="avatars" :style="avatarsStyle" />
|
||||||
<p>{{ usename }}</p>
|
<p v-if="username" class="dark:text-white">{{ username }}</p>
|
||||||
</span>
|
</span>
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu class="logout">
|
<el-dropdown-menu class="logout">
|
||||||
<el-dropdown-item @click="logout">
|
<el-dropdown-item @click="logout">
|
||||||
<i class="ri-logout-circle-r-line"></i
|
<IconifyIconOffline
|
||||||
>{{ $t("buttons.hsLoginOut") }}</el-dropdown-item
|
icon="logout-circle-r-line"
|
||||||
>
|
style="margin: 5px"
|
||||||
|
/>
|
||||||
|
{{ t("buttons.hsLoginOut") }}
|
||||||
|
</el-dropdown-item>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
<el-icon
|
<span
|
||||||
class="el-icon-setting"
|
class="el-icon-setting navbar-bg-hover"
|
||||||
:title="$t('buttons.hssystemSet')"
|
:title="t('buttons.hssystemSet')"
|
||||||
@click="onPanel"
|
@click="onPanel"
|
||||||
>
|
>
|
||||||
<IconifyIconOffline icon="setting" />
|
<IconifyIconOffline icon="setting" />
|
||||||
</el-icon>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -142,16 +118,12 @@ function translationEn() {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 48px;
|
height: 48px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
background: #fff;
|
|
||||||
box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08);
|
|
||||||
|
|
||||||
.hamburger-container {
|
.hamburger-container {
|
||||||
line-height: 48px;
|
line-height: 48px;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
float: left;
|
float: left;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: background 0.3s;
|
|
||||||
-webkit-tap-highlight-color: transparent;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.vertical-header-right {
|
.vertical-header-right {
|
||||||
@@ -162,33 +134,7 @@ function translationEn() {
|
|||||||
color: #000000d9;
|
color: #000000d9;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
|
|
||||||
:deep(.dropdown-badge) {
|
|
||||||
&:hover {
|
|
||||||
background: #f6f6f6;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.screen-full {
|
|
||||||
cursor: pointer;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: #f6f6f6;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.globalization {
|
|
||||||
height: 48px;
|
|
||||||
width: 40px;
|
|
||||||
padding: 11px;
|
|
||||||
cursor: pointer;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: #f6f6f6;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.el-dropdown-link {
|
.el-dropdown-link {
|
||||||
width: 100px;
|
|
||||||
height: 48px;
|
height: 48px;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -197,10 +143,6 @@ function translationEn() {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
color: #000000d9;
|
color: #000000d9;
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: #f6f6f6;
|
|
||||||
}
|
|
||||||
|
|
||||||
p {
|
p {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
@@ -219,27 +161,18 @@ function translationEn() {
|
|||||||
display: flex;
|
display: flex;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background: #f6f6f6;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumb-container {
|
.breadcrumb-container {
|
||||||
float: left;
|
float: left;
|
||||||
|
margin-left: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.translation {
|
.translation {
|
||||||
.el-dropdown-menu__item {
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
padding: 5px 40px !important;
|
padding: 5px 40px;
|
||||||
}
|
|
||||||
|
|
||||||
.el-dropdown-menu__item:focus,
|
|
||||||
.el-dropdown-menu__item:not(.is-disabled):hover {
|
|
||||||
color: #606266;
|
|
||||||
background: #f0f0f0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.check-zh {
|
.check-zh {
|
||||||
@@ -256,16 +189,10 @@ function translationEn() {
|
|||||||
.logout {
|
.logout {
|
||||||
max-width: 120px;
|
max-width: 120px;
|
||||||
|
|
||||||
.el-dropdown-menu__item {
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
min-width: 100%;
|
min-width: 100%;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.el-dropdown-menu__item:focus,
|
|
||||||
.el-dropdown-menu__item:not(.is-disabled):hover {
|
|
||||||
color: #606266;
|
|
||||||
background: #f0f0f0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -108,7 +108,7 @@ export const noticesData: TabItem[] = [
|
|||||||
{
|
{
|
||||||
avatar: "",
|
avatar: "",
|
||||||
title: "任务名称",
|
title: "任务名称",
|
||||||
description: "任务需要在 2021-11-16 20:00 前启动",
|
description: "任务需要在 2022-11-16 20:00 前启动",
|
||||||
datetime: "",
|
datetime: "",
|
||||||
extra: "未开始",
|
extra: "未开始",
|
||||||
status: "info",
|
status: "info",
|
||||||
@@ -118,7 +118,7 @@ export const noticesData: TabItem[] = [
|
|||||||
avatar: "",
|
avatar: "",
|
||||||
title: "第三方紧急代码变更",
|
title: "第三方紧急代码变更",
|
||||||
description:
|
description:
|
||||||
"一拳提交于 2021-11-16,需在 2021-11-18 前完成代码变更任务",
|
"一拳提交于 2022-11-16,需在 2022-11-18 前完成代码变更任务",
|
||||||
datetime: "",
|
datetime: "",
|
||||||
extra: "马上到期",
|
extra: "马上到期",
|
||||||
status: "danger",
|
status: "danger",
|
||||||
@@ -127,7 +127,7 @@ export const noticesData: TabItem[] = [
|
|||||||
{
|
{
|
||||||
avatar: "",
|
avatar: "",
|
||||||
title: "信息安全考试",
|
title: "信息安全考试",
|
||||||
description: "指派小仙于 2021-12-12 前完成更新并发布",
|
description: "指派小仙于 2022-12-12 前完成更新并发布",
|
||||||
datetime: "",
|
datetime: "",
|
||||||
extra: "已耗时 8 天",
|
extra: "已耗时 8 天",
|
||||||
status: "warning",
|
status: "warning",
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref } from "vue";
|
import { ref } from "vue";
|
||||||
import NoticeList from "./noticeList.vue";
|
|
||||||
import { noticesData } from "./data";
|
import { noticesData } from "./data";
|
||||||
|
import NoticeList from "./noticeList.vue";
|
||||||
|
import { templateRef } from "@vueuse/core";
|
||||||
|
import { Tabs, TabPane } from "@pureadmin/components";
|
||||||
|
|
||||||
|
const dropdownDom = templateRef<ElRef | null>("dropdownDom", null);
|
||||||
const activeName = ref(noticesData[0].name);
|
const activeName = ref(noticesData[0].name);
|
||||||
const notices = ref(noticesData);
|
const notices = ref(noticesData);
|
||||||
|
|
||||||
@@ -10,38 +13,50 @@ let noticesNum = ref(0);
|
|||||||
notices.value.forEach(notice => {
|
notices.value.forEach(notice => {
|
||||||
noticesNum.value += notice.list.length;
|
noticesNum.value += notice.list.length;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function tabClick() {
|
||||||
|
(dropdownDom as any).value.handleOpen();
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<el-dropdown trigger="click" placement="bottom-end">
|
<el-dropdown ref="dropdownDom" trigger="click" placement="bottom-end">
|
||||||
<span class="dropdown-badge">
|
<span class="dropdown-badge navbar-bg-hover select-none">
|
||||||
<el-badge :value="noticesNum" :max="99">
|
<el-badge :value="noticesNum" :max="99">
|
||||||
<el-icon class="header-notice-icon"
|
<span class="header-notice-icon">
|
||||||
><IconifyIconOffline icon="bell"
|
<IconifyIconOffline icon="bell" />
|
||||||
/></el-icon>
|
</span>
|
||||||
</el-badge>
|
</el-badge>
|
||||||
</span>
|
</span>
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu>
|
<el-dropdown-menu>
|
||||||
<el-tabs v-model="activeName" class="dropdown-tabs">
|
<Tabs
|
||||||
|
centered
|
||||||
|
class="dropdown-tabs"
|
||||||
|
v-model:activeName="activeName"
|
||||||
|
@tabClick="tabClick"
|
||||||
|
>
|
||||||
<template v-for="item in notices" :key="item.key">
|
<template v-for="item in notices" :key="item.key">
|
||||||
<el-tab-pane
|
<TabPane :tab="`${item.name}(${item.list.length})`">
|
||||||
:label="`${item.name}(${item.list.length})`"
|
|
||||||
:name="item.name"
|
|
||||||
>
|
|
||||||
<el-scrollbar max-height="330px">
|
<el-scrollbar max-height="330px">
|
||||||
<div class="noticeList-container">
|
<div class="noticeList-container">
|
||||||
<NoticeList :list="item.list" />
|
<NoticeList :list="item.list" />
|
||||||
</div>
|
</div>
|
||||||
</el-scrollbar>
|
</el-scrollbar>
|
||||||
</el-tab-pane>
|
</TabPane>
|
||||||
</template>
|
</template>
|
||||||
</el-tabs>
|
</Tabs>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.ant-tabs-dropdown {
|
||||||
|
z-index: 2900 !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.dropdown-badge {
|
.dropdown-badge {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -79,4 +94,8 @@ notices.value.forEach(notice => {
|
|||||||
padding: 15px 24px 0 24px;
|
padding: 15px 24px 0 24px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:deep(.ant-tabs-nav) {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ const props = defineProps({
|
|||||||
});
|
});
|
||||||
|
|
||||||
const titleRef = ref(null);
|
const titleRef = ref(null);
|
||||||
const descriptionRef = ref(null);
|
|
||||||
const titleTooltip = ref(false);
|
const titleTooltip = ref(false);
|
||||||
|
const descriptionRef = ref(null);
|
||||||
const descriptionTooltip = ref(false);
|
const descriptionTooltip = ref(false);
|
||||||
|
|
||||||
function hoverTitle() {
|
function hoverTitle() {
|
||||||
@@ -44,15 +44,17 @@ function hoverDescription(event, description) {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="notice-container">
|
<div
|
||||||
|
class="notice-container border-b-[1px] border-solid border-[#f0f0f0] dark:border-[#303030]"
|
||||||
|
>
|
||||||
<el-avatar
|
<el-avatar
|
||||||
v-if="props.noticeItem.avatar"
|
v-if="props.noticeItem.avatar"
|
||||||
:size="30"
|
:size="30"
|
||||||
:src="props.noticeItem.avatar"
|
:src="props.noticeItem.avatar"
|
||||||
class="notice-container-avatar"
|
class="notice-container-avatar"
|
||||||
></el-avatar>
|
/>
|
||||||
<div class="notice-container-text">
|
<div class="notice-container-text">
|
||||||
<div class="notice-text-title">
|
<div class="notice-text-title text-[#000000d9] dark:text-white">
|
||||||
<el-tooltip
|
<el-tooltip
|
||||||
popper-class="notice-title-popper"
|
popper-class="notice-title-popper"
|
||||||
:disabled="!titleTooltip"
|
:disabled="!titleTooltip"
|
||||||
@@ -72,7 +74,8 @@ function hoverDescription(event, description) {
|
|||||||
:type="props.noticeItem?.status"
|
:type="props.noticeItem?.status"
|
||||||
size="small"
|
size="small"
|
||||||
class="notice-title-extra"
|
class="notice-title-extra"
|
||||||
>{{ props.noticeItem?.extra }}
|
>
|
||||||
|
{{ props.noticeItem?.extra }}
|
||||||
</el-tag>
|
</el-tag>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -90,7 +93,7 @@ function hoverDescription(event, description) {
|
|||||||
{{ props.noticeItem.description }}
|
{{ props.noticeItem.description }}
|
||||||
</div>
|
</div>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<div class="notice-text-datetime">
|
<div class="notice-text-datetime text-[#00000073] dark:text-white">
|
||||||
{{ props.noticeItem.datetime }}
|
{{ props.noticeItem.datetime }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -108,7 +111,7 @@ function hoverDescription(event, description) {
|
|||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 12px 0;
|
padding: 12px 0;
|
||||||
border-bottom: 1px solid #f0f0f0;
|
// border-bottom: 1px solid #f0f0f0;
|
||||||
|
|
||||||
.notice-container-avatar {
|
.notice-container-avatar {
|
||||||
margin-right: 16px;
|
margin-right: 16px;
|
||||||
@@ -127,7 +130,6 @@ function hoverDescription(event, description) {
|
|||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 1.5715;
|
line-height: 1.5715;
|
||||||
color: rgba(0, 0, 0, 0.85);
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
||||||
.notice-title-content {
|
.notice-title-content {
|
||||||
@@ -149,7 +151,6 @@ function hoverDescription(event, description) {
|
|||||||
.notice-text-datetime {
|
.notice-text-datetime {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
line-height: 1.5715;
|
line-height: 1.5715;
|
||||||
color: rgba(0, 0, 0, 0.45);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.notice-text-description {
|
.notice-text-description {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { PropType } from "vue";
|
import { PropType } from "vue";
|
||||||
import NoticeItem from "./noticeItem.vue";
|
|
||||||
import { ListItem } from "./data";
|
import { ListItem } from "./data";
|
||||||
|
import NoticeItem from "./noticeItem.vue";
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
list: {
|
list: {
|
||||||
@@ -17,7 +17,7 @@ const props = defineProps({
|
|||||||
v-for="(item, index) in props.list"
|
v-for="(item, index) in props.list"
|
||||||
:noticeItem="item"
|
:noticeItem="item"
|
||||||
:key="index"
|
:key="index"
|
||||||
></NoticeItem>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<el-empty v-else description="暂无数据"></el-empty>
|
<el-empty v-else description="暂无数据" />
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { emitter } from "/@/utils/mitt";
|
|||||||
|
|
||||||
let show = ref<Boolean>(false);
|
let show = ref<Boolean>(false);
|
||||||
const target = ref(null);
|
const target = ref(null);
|
||||||
onClickOutside(target, event => {
|
onClickOutside(target, (event: any) => {
|
||||||
if (event.clientX > target.value.offsetLeft) return;
|
if (event.clientX > target.value.offsetLeft) return;
|
||||||
show.value = false;
|
show.value = false;
|
||||||
});
|
});
|
||||||
@@ -18,15 +18,17 @@ emitter.on("openPanel", () => {
|
|||||||
<template>
|
<template>
|
||||||
<div :class="{ show: show }" class="right-panel-container">
|
<div :class="{ show: show }" class="right-panel-container">
|
||||||
<div class="right-panel-background" />
|
<div class="right-panel-background" />
|
||||||
<div ref="target" class="right-panel">
|
<div ref="target" class="right-panel bg-white dark:bg-dark">
|
||||||
<div class="right-panel-items">
|
<div class="right-panel-items">
|
||||||
<div class="project-configuration">
|
<div class="project-configuration">
|
||||||
<h3>项目配置</h3>
|
<h3 class="dark:text-white">项目配置</h3>
|
||||||
<el-icon title="关闭配置" class="el-icon-close" @click="show = !show">
|
<el-icon title="关闭配置" class="el-icon-close" @click="show = !show">
|
||||||
<IconifyIconOffline icon="close" />
|
<IconifyIconOffline icon="close" />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</div>
|
</div>
|
||||||
<div style="border-bottom: 1px solid #dcdfe6"></div>
|
<div
|
||||||
|
class="border-b-[1px] border-solid border-[#dcdfe6] dark:border-[#303030]"
|
||||||
|
/>
|
||||||
<slot />
|
<slot />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -54,7 +56,7 @@ emitter.on("openPanel", () => {
|
|||||||
|
|
||||||
.right-panel {
|
.right-panel {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 300px;
|
max-width: 315px;
|
||||||
height: 100vh;
|
height: 100vh;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
@@ -62,7 +64,7 @@ emitter.on("openPanel", () => {
|
|||||||
box-shadow: 0 0 15px 0 rgba(0, 0, 0, 0.05);
|
box-shadow: 0 0 15px 0 rgba(0, 0, 0, 0.05);
|
||||||
transition: all 0.25s cubic-bezier(0.7, 0.3, 0.1, 1);
|
transition: all 0.25s cubic-bezier(0.7, 0.3, 0.1, 1);
|
||||||
transform: translate(100%);
|
transform: translate(100%);
|
||||||
background: #fff;
|
// background: #fff;
|
||||||
z-index: 40000;
|
z-index: 40000;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -105,8 +107,8 @@ emitter.on("openPanel", () => {
|
|||||||
|
|
||||||
.right-panel-items {
|
.right-panel-items {
|
||||||
margin-top: 60px;
|
margin-top: 60px;
|
||||||
height: 100vh;
|
height: calc(100vh - 60px);
|
||||||
overflow: auto;
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.project-configuration {
|
.project-configuration {
|
||||||
@@ -125,7 +127,7 @@ emitter.on("openPanel", () => {
|
|||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
color: red;
|
color: var(--el-color-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,31 +1,21 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
import { useFullscreen } from "@vueuse/core";
|
import { useFullscreen } from "@vueuse/core";
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
const { isFullscreen, toggle } = useFullscreen();
|
const { isFullscreen, toggle } = useFullscreen();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="screen-full" @click="toggle">
|
<div
|
||||||
<i
|
class="screen-full w-[36px] h-[48px] flex-ac cursor-pointer navbar-bg-hover"
|
||||||
|
@click="toggle"
|
||||||
|
>
|
||||||
|
<FontIcon
|
||||||
:title="
|
:title="
|
||||||
isFullscreen
|
isFullscreen ? t('buttons.hsexitfullscreen') : t('buttons.hsfullscreen')
|
||||||
? $t('buttons.hsexitfullscreen')
|
|
||||||
: $t('buttons.hsfullscreen')
|
|
||||||
"
|
"
|
||||||
:class="
|
:icon="isFullscreen ? 'team-iconexit-fullscreen' : 'team-iconfullscreen'"
|
||||||
isFullscreen
|
/>
|
||||||
? 'iconfont team-iconexit-fullscreen'
|
|
||||||
: 'iconfont team-iconfullscreen'
|
|
||||||
"
|
|
||||||
></i>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
|
||||||
.screen-full {
|
|
||||||
width: 36px;
|
|
||||||
height: 48px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-around;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
41
src/layout/components/search/components/SearchFooter.vue
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<template>
|
||||||
|
<div class="search-footer text-[#333] dark:text-white">
|
||||||
|
<span class="search-footer-item">
|
||||||
|
<enterOutlined class="icon" />
|
||||||
|
确认
|
||||||
|
</span>
|
||||||
|
<span class="search-footer-item">
|
||||||
|
<IconifyIconOffline icon="arrow-up-line" class="icon" />
|
||||||
|
<IconifyIconOffline icon="arrow-down-line" class="icon" />
|
||||||
|
切换
|
||||||
|
</span>
|
||||||
|
<span class="search-footer-item">
|
||||||
|
<mdiKeyboardEsc class="icon" />
|
||||||
|
关闭
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import enterOutlined from "/@/assets/svg/enter_outlined.svg?component";
|
||||||
|
import mdiKeyboardEsc from "/@/assets/svg/mdi_keyboard_esc.svg?component";
|
||||||
|
</script>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.search-footer {
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
.search-footer-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-right: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
padding: 2px;
|
||||||
|
margin-right: 3px;
|
||||||
|
font-size: 20px;
|
||||||
|
box-shadow: inset 0 -2px #cdcde6, inset 0 0 1px 1px #fff,
|
||||||
|
0 1px 2px 1px #1e235a66;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
172
src/layout/components/search/components/SearchModal.vue
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { useRouter } from "vue-router";
|
||||||
|
import SearchResult from "./SearchResult.vue";
|
||||||
|
import SearchFooter from "./SearchFooter.vue";
|
||||||
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
|
import { transformI18n } from "/@/plugins/i18n";
|
||||||
|
import { deleteChildren } from "@pureadmin/utils";
|
||||||
|
import { useDebounceFn, onKeyStroke } from "@vueuse/core";
|
||||||
|
import { ref, watch, computed, nextTick, shallowRef } from "vue";
|
||||||
|
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
/** 弹窗显隐 */
|
||||||
|
value: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Emits {
|
||||||
|
(e: "update:value", val: boolean): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { device } = useNav();
|
||||||
|
const emit = defineEmits<Emits>();
|
||||||
|
const props = withDefaults(defineProps<Props>(), {});
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const keyword = ref("");
|
||||||
|
const activePath = ref("");
|
||||||
|
const inputRef = ref<HTMLInputElement | null>(null);
|
||||||
|
const resultOptions = shallowRef([]);
|
||||||
|
const handleSearch = useDebounceFn(search, 300);
|
||||||
|
|
||||||
|
/** 菜单树形结构 */
|
||||||
|
const menusData = computed(() => {
|
||||||
|
return deleteChildren(usePermissionStoreHook().menusTree);
|
||||||
|
});
|
||||||
|
|
||||||
|
const show = computed({
|
||||||
|
get() {
|
||||||
|
return props.value;
|
||||||
|
},
|
||||||
|
set(val: boolean) {
|
||||||
|
emit("update:value", val);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(show, async val => {
|
||||||
|
if (val) {
|
||||||
|
/** 自动聚焦 */
|
||||||
|
await nextTick();
|
||||||
|
inputRef.value?.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/** 将菜单树形结构扁平化为一维数组,用于菜单查询 */
|
||||||
|
function flatTree(arr) {
|
||||||
|
const res = [];
|
||||||
|
function deep(arr) {
|
||||||
|
arr.forEach(item => {
|
||||||
|
res.push(item);
|
||||||
|
item.children && deep(item.children);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
deep(arr);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 查询 */
|
||||||
|
function search() {
|
||||||
|
const flatMenusData = flatTree(menusData.value);
|
||||||
|
resultOptions.value = flatMenusData.filter(
|
||||||
|
menu =>
|
||||||
|
keyword.value &&
|
||||||
|
transformI18n(menu.meta?.title)
|
||||||
|
.toLocaleLowerCase()
|
||||||
|
.includes(keyword.value.toLocaleLowerCase().trim())
|
||||||
|
);
|
||||||
|
if (resultOptions.value?.length > 0) {
|
||||||
|
activePath.value = resultOptions.value[0].path;
|
||||||
|
} else {
|
||||||
|
activePath.value = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleClose() {
|
||||||
|
show.value = false;
|
||||||
|
/** 延时处理防止用户看到某些操作 */
|
||||||
|
setTimeout(() => {
|
||||||
|
resultOptions.value = [];
|
||||||
|
keyword.value = "";
|
||||||
|
}, 200);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** key up */
|
||||||
|
function handleUp() {
|
||||||
|
const { length } = resultOptions.value;
|
||||||
|
if (length === 0) return;
|
||||||
|
const index = resultOptions.value.findIndex(
|
||||||
|
item => item.path === activePath.value
|
||||||
|
);
|
||||||
|
if (index === 0) {
|
||||||
|
activePath.value = resultOptions.value[length - 1].path;
|
||||||
|
} else {
|
||||||
|
activePath.value = resultOptions.value[index - 1].path;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** key down */
|
||||||
|
function handleDown() {
|
||||||
|
const { length } = resultOptions.value;
|
||||||
|
if (length === 0) return;
|
||||||
|
const index = resultOptions.value.findIndex(
|
||||||
|
item => item.path === activePath.value
|
||||||
|
);
|
||||||
|
if (index + 1 === length) {
|
||||||
|
activePath.value = resultOptions.value[0].path;
|
||||||
|
} else {
|
||||||
|
activePath.value = resultOptions.value[index + 1].path;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** key enter */
|
||||||
|
function handleEnter() {
|
||||||
|
const { length } = resultOptions.value;
|
||||||
|
if (length === 0 || activePath.value === "") return;
|
||||||
|
router.push(activePath.value);
|
||||||
|
handleClose();
|
||||||
|
}
|
||||||
|
|
||||||
|
onKeyStroke("Enter", handleEnter);
|
||||||
|
onKeyStroke("ArrowUp", handleUp);
|
||||||
|
onKeyStroke("ArrowDown", handleDown);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<el-dialog
|
||||||
|
top="5vh"
|
||||||
|
:width="device === 'mobile' ? '80vw' : '50vw'"
|
||||||
|
v-model="show"
|
||||||
|
:before-close="handleClose"
|
||||||
|
>
|
||||||
|
<el-input
|
||||||
|
ref="inputRef"
|
||||||
|
v-model="keyword"
|
||||||
|
clearable
|
||||||
|
placeholder="请输入关键词搜索"
|
||||||
|
@input="handleSearch"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<span class="el-input__icon">
|
||||||
|
<IconifyIconOffline icon="search" />
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</el-input>
|
||||||
|
<div class="search-result-container">
|
||||||
|
<el-empty v-if="resultOptions.length === 0" description="暂无搜索结果" />
|
||||||
|
<SearchResult
|
||||||
|
v-else
|
||||||
|
v-model:value="activePath"
|
||||||
|
:options="resultOptions"
|
||||||
|
@click="handleEnter"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<SearchFooter />
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</template>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.search-result-container {
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
100
src/layout/components/search/components/SearchResult.vue
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { computed } from "vue";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
||||||
|
import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
|
||||||
|
import enterOutlined from "/@/assets/svg/enter_outlined.svg?component";
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
interface optionsItem {
|
||||||
|
path: string;
|
||||||
|
meta?: {
|
||||||
|
icon?: string;
|
||||||
|
title?: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
value: string;
|
||||||
|
options: Array<optionsItem>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Emits {
|
||||||
|
(e: "update:value", val: string): void;
|
||||||
|
(e: "enter"): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<Props>(), {});
|
||||||
|
const emit = defineEmits<Emits>();
|
||||||
|
|
||||||
|
const itemStyle = computed(() => {
|
||||||
|
return item => {
|
||||||
|
return {
|
||||||
|
background:
|
||||||
|
item?.path === active.value ? useEpThemeStoreHook().epThemeColor : "",
|
||||||
|
color: item.path === active.value ? "#fff" : "",
|
||||||
|
fontSize: item.path === active.value ? "16px" : "14px"
|
||||||
|
};
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const active = computed({
|
||||||
|
get() {
|
||||||
|
return props.value;
|
||||||
|
},
|
||||||
|
set(val: string) {
|
||||||
|
emit("update:value", val);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/** 鼠标移入 */
|
||||||
|
async function handleMouse(item) {
|
||||||
|
active.value = item.path;
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleTo() {
|
||||||
|
emit("enter");
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="result">
|
||||||
|
<template v-for="item in options" :key="item.path">
|
||||||
|
<div
|
||||||
|
class="result-item dark:bg-[#1d1d1d]"
|
||||||
|
:style="itemStyle(item)"
|
||||||
|
@click="handleTo"
|
||||||
|
@mouseenter="handleMouse(item)"
|
||||||
|
>
|
||||||
|
<component :is="useRenderIcon(item.meta?.icon ?? 'bookmark-2-line')" />
|
||||||
|
<span class="result-item-title">{{ t(item.meta?.title) }}</span>
|
||||||
|
<enterOutlined />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.result {
|
||||||
|
padding-bottom: 12px;
|
||||||
|
|
||||||
|
&-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
height: 56px;
|
||||||
|
margin-top: 8px;
|
||||||
|
padding: 14px;
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 0.1px solid #ccc;
|
||||||
|
transition: all 0.3s;
|
||||||
|
|
||||||
|
&-title {
|
||||||
|
display: flex;
|
||||||
|
flex: 1;
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
3
src/layout/components/search/components/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import SearchModal from "./SearchModal.vue";
|
||||||
|
|
||||||
|
export { SearchModal };
|
||||||
18
src/layout/components/search/index.vue
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { SearchModal } from "./components";
|
||||||
|
import { useBoolean } from "../../hooks/useBoolean";
|
||||||
|
const { bool: show, toggle } = useBoolean();
|
||||||
|
function handleSearch() {
|
||||||
|
toggle();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="search-container w-[40px] h-[48px] flex-c cursor-pointer navbar-bg-hover"
|
||||||
|
@click="handleSearch"
|
||||||
|
>
|
||||||
|
<IconifyIconOffline icon="search" />
|
||||||
|
</div>
|
||||||
|
<SearchModal v-model:value="show" />
|
||||||
|
</template>
|
||||||
@@ -1,75 +1,57 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
import {
|
||||||
reactive,
|
|
||||||
ref,
|
ref,
|
||||||
unref,
|
unref,
|
||||||
watch,
|
watch,
|
||||||
|
reactive,
|
||||||
computed,
|
computed,
|
||||||
nextTick,
|
nextTick,
|
||||||
useCssModule,
|
useCssModule
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
} from "vue";
|
||||||
import rgbHex from "rgb-hex";
|
|
||||||
import { find } from "lodash-es";
|
|
||||||
import { getConfig } from "/@/config";
|
import { getConfig } from "/@/config";
|
||||||
import { useRouter } from "vue-router";
|
import { useRouter } from "vue-router";
|
||||||
import panel from "../panel/index.vue";
|
import panel from "../panel/index.vue";
|
||||||
import { emitter } from "/@/utils/mitt";
|
import { emitter } from "/@/utils/mitt";
|
||||||
|
import { resetRouter } from "/@/router";
|
||||||
import { templateRef } from "@vueuse/core";
|
import { templateRef } from "@vueuse/core";
|
||||||
import { debounce } from "/@/utils/debounce";
|
import { routerArrays } from "/@/layout/types";
|
||||||
import { themeColorsType } from "../../types";
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import { useAppStoreHook } from "/@/store/modules/app";
|
import { useAppStoreHook } from "/@/store/modules/app";
|
||||||
import { shadeBgColor } from "../../theme/element-plus";
|
|
||||||
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
|
||||||
import { storageLocal, storageSession } from "/@/utils/storage";
|
|
||||||
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
||||||
import { createNewStyle, writeNewStyle } from "../../theme/element-plus";
|
import { useDataThemeChange } from "/@/layout/hooks/useDataThemeChange";
|
||||||
import { toggleTheme } from "@zougt/vite-plugin-theme-preprocessor/dist/browser-utils";
|
import {
|
||||||
|
useDark,
|
||||||
|
debounce,
|
||||||
|
useGlobal,
|
||||||
|
storageLocal,
|
||||||
|
storageSession
|
||||||
|
} from "@pureadmin/utils";
|
||||||
|
import { toggleTheme } from "@pureadmin/theme/dist/browser-utils";
|
||||||
|
|
||||||
import dayIcon from "/@/assets/svg/day.svg?component";
|
import dayIcon from "/@/assets/svg/day.svg?component";
|
||||||
import darkIcon from "/@/assets/svg/dark.svg?component";
|
import darkIcon from "/@/assets/svg/dark.svg?component";
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
const { device } = useNav();
|
||||||
|
const { isDark } = useDark();
|
||||||
const { isSelect } = useCssModule();
|
const { isSelect } = useCssModule();
|
||||||
const body = document.documentElement as HTMLElement;
|
const { $storage } = useGlobal<GlobalPropertiesApi>();
|
||||||
const instance =
|
|
||||||
getCurrentInstance().appContext.app.config.globalProperties.$storage;
|
|
||||||
|
|
||||||
const instanceConfig =
|
|
||||||
getCurrentInstance().appContext.app.config.globalProperties.$config;
|
|
||||||
|
|
||||||
let themeColors = ref<Array<themeColorsType>>([
|
|
||||||
// 道奇蓝(默认)
|
|
||||||
{ rgb: "27, 42, 71", themeColor: "default" },
|
|
||||||
// 亮白色
|
|
||||||
{ rgb: "255, 255, 255", themeColor: "light" },
|
|
||||||
// 猩红色
|
|
||||||
{ rgb: "245, 34, 45", themeColor: "dusk" },
|
|
||||||
// 橙红色
|
|
||||||
{ rgb: "250, 84, 28", themeColor: "volcano" },
|
|
||||||
// 金色
|
|
||||||
{ rgb: "250, 219, 20", themeColor: "yellow" },
|
|
||||||
// 绿宝石
|
|
||||||
{ rgb: "19, 194, 194", themeColor: "mingQing" },
|
|
||||||
// 酸橙绿
|
|
||||||
{ rgb: "82, 196, 26", themeColor: "auroraGreen" },
|
|
||||||
// 深粉色
|
|
||||||
{ rgb: "235, 47, 150", themeColor: "pink" },
|
|
||||||
// 深紫罗兰色
|
|
||||||
{ rgb: "114, 46, 209", themeColor: "saucePurple" }
|
|
||||||
]);
|
|
||||||
|
|
||||||
|
const mixRef = templateRef<HTMLElement | null>("mixRef", null);
|
||||||
const verticalRef = templateRef<HTMLElement | null>("verticalRef", null);
|
const verticalRef = templateRef<HTMLElement | null>("verticalRef", null);
|
||||||
const horizontalRef = templateRef<HTMLElement | null>("horizontalRef", null);
|
const horizontalRef = templateRef<HTMLElement | null>("horizontalRef", null);
|
||||||
|
|
||||||
let layoutTheme =
|
const {
|
||||||
ref(storageLocal.getItem("responsive-layout")) ||
|
body,
|
||||||
ref({
|
dataTheme,
|
||||||
layout: instanceConfig?.Layout ?? "vertical",
|
layoutTheme,
|
||||||
theme: instanceConfig?.Theme ?? "default"
|
themeColors,
|
||||||
});
|
dataThemeChange,
|
||||||
|
setEpThemeColor,
|
||||||
|
setLayoutThemeColor
|
||||||
|
} = useDataThemeChange();
|
||||||
|
|
||||||
// body添加layout属性,作用于src/style/sidebar.scss
|
/* body添加layout属性,作用于src/style/sidebar.scss */
|
||||||
if (unref(layoutTheme)) {
|
if (unref(layoutTheme)) {
|
||||||
let layout = unref(layoutTheme).layout;
|
let layout = unref(layoutTheme).layout;
|
||||||
let theme = unref(layoutTheme).theme;
|
let theme = unref(layoutTheme).theme;
|
||||||
@@ -79,106 +61,99 @@ if (unref(layoutTheme)) {
|
|||||||
setLayoutModel(layout);
|
setLayoutModel(layout);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 默认灵动模式
|
/** 默认灵动模式 */
|
||||||
const markValue = ref(instance.configure?.showModel ?? "smart");
|
const markValue = ref($storage.configure?.showModel ?? "smart");
|
||||||
|
|
||||||
const logoVal = ref(instance.configure?.showLogo ?? true);
|
const logoVal = ref($storage.configure?.showLogo ?? true);
|
||||||
|
|
||||||
const epThemeColor = ref(useEpThemeStoreHook().getEpThemeColor);
|
|
||||||
|
|
||||||
const settings = reactive({
|
const settings = reactive({
|
||||||
greyVal: instance.configure.grey,
|
greyVal: $storage.configure.grey,
|
||||||
weakVal: instance.configure.weak,
|
weakVal: $storage.configure.weak,
|
||||||
tabsVal: instance.configure.hideTabs,
|
tabsVal: $storage.configure.hideTabs,
|
||||||
showLogo: instance.configure.showLogo,
|
showLogo: $storage.configure.showLogo,
|
||||||
showModel: instance.configure.showModel,
|
showModel: $storage.configure.showModel,
|
||||||
multiTagsCache: instance.configure.multiTagsCache
|
multiTagsCache: $storage.configure.multiTagsCache
|
||||||
});
|
});
|
||||||
|
|
||||||
const getThemeColorStyle = computed(() => {
|
const getThemeColorStyle = computed(() => {
|
||||||
return rgb => {
|
return color => {
|
||||||
return { background: `rgb(${rgb})` };
|
return { background: color };
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
function changeStorageConfigure(key, val) {
|
/** 当网页为暗黑模式时不显示亮白色切换选项 */
|
||||||
const storageConfigure = instance.configure;
|
const showThemeColors = computed(() => {
|
||||||
|
return themeColor => {
|
||||||
|
return themeColor === "light" && isDark.value ? false : true;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
function storageConfigureChange<T>(key: string, val: T): void {
|
||||||
|
const storageConfigure = $storage.configure;
|
||||||
storageConfigure[key] = val;
|
storageConfigure[key] = val;
|
||||||
instance.configure = storageConfigure;
|
$storage.configure = storageConfigure;
|
||||||
}
|
}
|
||||||
|
|
||||||
function toggleClass(flag: boolean, clsName: string, target?: HTMLElement) {
|
function toggleClass(flag: boolean, clsName: string, target?: HTMLElement) {
|
||||||
const targetEl = target || document.body;
|
const targetEl = target || document.body;
|
||||||
let { className } = targetEl;
|
let { className } = targetEl;
|
||||||
className = className.replace(clsName, "");
|
className = className.replace(clsName, "").trim();
|
||||||
targetEl.className = flag ? `${className} ${clsName} ` : className;
|
targetEl.className = flag ? `${className} ${clsName} ` : className;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 灰色模式设置
|
/** 灰色模式设置 */
|
||||||
const greyChange = (value): void => {
|
const greyChange = (value): void => {
|
||||||
toggleClass(settings.greyVal, "html-grey", document.querySelector("html"));
|
toggleClass(settings.greyVal, "html-grey", document.querySelector("html"));
|
||||||
changeStorageConfigure("grey", value);
|
storageConfigureChange("grey", value);
|
||||||
};
|
};
|
||||||
|
|
||||||
// 色弱模式设置
|
/** 色弱模式设置 */
|
||||||
const weekChange = (value): void => {
|
const weekChange = (value): void => {
|
||||||
toggleClass(
|
toggleClass(
|
||||||
settings.weakVal,
|
settings.weakVal,
|
||||||
"html-weakness",
|
"html-weakness",
|
||||||
document.querySelector("html")
|
document.querySelector("html")
|
||||||
);
|
);
|
||||||
changeStorageConfigure("weak", value);
|
storageConfigureChange("weak", value);
|
||||||
};
|
};
|
||||||
|
|
||||||
const tagsChange = () => {
|
const tagsChange = () => {
|
||||||
let showVal = settings.tabsVal;
|
let showVal = settings.tabsVal;
|
||||||
changeStorageConfigure("hideTabs", showVal);
|
storageConfigureChange("hideTabs", showVal);
|
||||||
emitter.emit("tagViewsChange", showVal);
|
emitter.emit("tagViewsChange", showVal as unknown as string);
|
||||||
};
|
};
|
||||||
|
|
||||||
const multiTagsCacheChange = () => {
|
const multiTagsCacheChange = () => {
|
||||||
let multiTagsCache = settings.multiTagsCache;
|
let multiTagsCache = settings.multiTagsCache;
|
||||||
changeStorageConfigure("multiTagsCache", multiTagsCache);
|
storageConfigureChange("multiTagsCache", multiTagsCache);
|
||||||
useMultiTagsStoreHook().multiTagsCacheChange(multiTagsCache);
|
useMultiTagsStoreHook().multiTagsCacheChange(multiTagsCache);
|
||||||
};
|
};
|
||||||
|
|
||||||
// 清空缓存并返回登录页
|
/** 清空缓存并返回登录页 */
|
||||||
function onReset() {
|
function onReset() {
|
||||||
toggleClass(getConfig().Grey, "html-grey", document.querySelector("html"));
|
router.push("/login");
|
||||||
toggleClass(
|
const { Grey, Weak, MultiTagsCache, EpThemeColor, Layout } = getConfig();
|
||||||
getConfig().Weak,
|
useAppStoreHook().setLayout(Layout);
|
||||||
"html-weakness",
|
setEpThemeColor(EpThemeColor);
|
||||||
document.querySelector("html")
|
useMultiTagsStoreHook().multiTagsCacheChange(MultiTagsCache);
|
||||||
);
|
toggleClass(Grey, "html-grey", document.querySelector("html"));
|
||||||
useMultiTagsStoreHook().handleTags("equal", [
|
toggleClass(Weak, "html-weakness", document.querySelector("html"));
|
||||||
{
|
useMultiTagsStoreHook().handleTags("equal", [...routerArrays]);
|
||||||
path: "/welcome",
|
|
||||||
parentPath: "/",
|
|
||||||
meta: {
|
|
||||||
title: "menus.hshome",
|
|
||||||
icon: "home-filled",
|
|
||||||
i18n: true,
|
|
||||||
showLink: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
useMultiTagsStoreHook().multiTagsCacheChange(getConfig().MultiTagsCache);
|
|
||||||
useEpThemeStoreHook().setEpThemeColor(getConfig().EpThemeColor);
|
|
||||||
storageLocal.clear();
|
storageLocal.clear();
|
||||||
storageSession.clear();
|
storageSession.clear();
|
||||||
router.push("/login");
|
resetRouter();
|
||||||
}
|
}
|
||||||
|
|
||||||
function onChange(label) {
|
function onChange(label) {
|
||||||
changeStorageConfigure("showModel", label);
|
storageConfigureChange("showModel", label);
|
||||||
emitter.emit("tagViewsShowModel", label);
|
emitter.emit("tagViewsShowModel", label);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 侧边栏Logo
|
/** 侧边栏Logo */
|
||||||
function logoChange() {
|
function logoChange() {
|
||||||
unref(logoVal)
|
unref(logoVal)
|
||||||
? changeStorageConfigure("showLogo", true)
|
? storageConfigureChange("showLogo", true)
|
||||||
: changeStorageConfigure("showLogo", false);
|
: storageConfigureChange("showLogo", false);
|
||||||
emitter.emit("logoChange", unref(logoVal));
|
emitter.emit("logoChange", unref(logoVal));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,20 +163,29 @@ function setFalse(Doms): any {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
watch(instance, ({ layout }) => {
|
watch($storage, ({ layout }) => {
|
||||||
|
/* 设置wangeditorV5主题色 */
|
||||||
|
body.style.setProperty("--w-e-toolbar-active-color", layout["epThemeColor"]);
|
||||||
switch (layout["layout"]) {
|
switch (layout["layout"]) {
|
||||||
case "vertical":
|
case "vertical":
|
||||||
toggleClass(true, isSelect, unref(verticalRef));
|
toggleClass(true, isSelect, unref(verticalRef));
|
||||||
debounce(setFalse([horizontalRef]), 50);
|
debounce(setFalse([horizontalRef]), 50);
|
||||||
|
debounce(setFalse([mixRef]), 50);
|
||||||
break;
|
break;
|
||||||
case "horizontal":
|
case "horizontal":
|
||||||
toggleClass(true, isSelect, unref(horizontalRef));
|
toggleClass(true, isSelect, unref(horizontalRef));
|
||||||
debounce(setFalse([verticalRef]), 50);
|
debounce(setFalse([verticalRef]), 50);
|
||||||
|
debounce(setFalse([mixRef]), 50);
|
||||||
|
break;
|
||||||
|
case "mix":
|
||||||
|
toggleClass(true, isSelect, unref(mixRef));
|
||||||
|
debounce(setFalse([verticalRef]), 50);
|
||||||
|
debounce(setFalse([horizontalRef]), 50);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// 主题色 激活选择项
|
/** 主题色 激活选择项 */
|
||||||
const getThemeColor = computed(() => {
|
const getThemeColor = computed(() => {
|
||||||
return current => {
|
return current => {
|
||||||
if (
|
if (
|
||||||
@@ -220,83 +204,27 @@ const getThemeColor = computed(() => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 设置导航模式
|
/** 设置导航模式 */
|
||||||
function setLayoutModel(layout: string) {
|
function setLayoutModel(layout: string) {
|
||||||
layoutTheme.value.layout = layout;
|
layoutTheme.value.layout = layout;
|
||||||
window.document.body.setAttribute("layout", layout);
|
window.document.body.setAttribute("layout", layout);
|
||||||
instance.layout = {
|
$storage.layout = {
|
||||||
layout,
|
layout,
|
||||||
theme: layoutTheme.value.theme,
|
theme: layoutTheme.value.theme,
|
||||||
darkMode: instance.layout.darkMode,
|
darkMode: $storage.layout?.darkMode,
|
||||||
sidebarStatus: instance.layout.sidebarStatus,
|
sidebarStatus: $storage.layout?.sidebarStatus,
|
||||||
epThemeColor: instance.layout.epThemeColor
|
epThemeColor: $storage.layout?.epThemeColor
|
||||||
};
|
};
|
||||||
useAppStoreHook().setLayout(layout);
|
useAppStoreHook().setLayout(layout);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 存放夜间主题切换前的导航主题色
|
/* 初始化项目配置 */
|
||||||
let tempLayoutThemeColor;
|
|
||||||
|
|
||||||
// 设置导航主题色
|
|
||||||
function setLayoutThemeColor(theme: string) {
|
|
||||||
tempLayoutThemeColor = instance.layout.theme;
|
|
||||||
layoutTheme.value.theme = theme;
|
|
||||||
toggleTheme({
|
|
||||||
scopeName: `layout-theme-${theme}`
|
|
||||||
});
|
|
||||||
instance.layout = {
|
|
||||||
layout: useAppStoreHook().layout,
|
|
||||||
theme,
|
|
||||||
darkMode: dataTheme.value,
|
|
||||||
sidebarStatus: instance.layout.sidebarStatus,
|
|
||||||
epThemeColor: instance.layout.epThemeColor
|
|
||||||
};
|
|
||||||
|
|
||||||
if (theme === "default" || theme === "light") {
|
|
||||||
setEpThemeColor(getConfig().EpThemeColor);
|
|
||||||
} else {
|
|
||||||
const colors = find(themeColors.value, { themeColor: theme });
|
|
||||||
const color = "#" + rgbHex(colors.rgb);
|
|
||||||
setEpThemeColor(color);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 设置ep主题色
|
|
||||||
const setEpThemeColor = (color: string) => {
|
|
||||||
writeNewStyle(createNewStyle(color));
|
|
||||||
useEpThemeStoreHook().setEpThemeColor(color);
|
|
||||||
body.style.setProperty("--el-color-primary-active", shadeBgColor(color));
|
|
||||||
};
|
|
||||||
|
|
||||||
let dataTheme = ref<boolean>(instance.layout.darkMode);
|
|
||||||
|
|
||||||
// 日间、夜间主题切换
|
|
||||||
function dataThemeChange() {
|
|
||||||
if (dataTheme.value) {
|
|
||||||
body.setAttribute("data-theme", "dark");
|
|
||||||
setLayoutThemeColor("light");
|
|
||||||
} else {
|
|
||||||
body.setAttribute("data-theme", "");
|
|
||||||
tempLayoutThemeColor && setLayoutThemeColor(tempLayoutThemeColor);
|
|
||||||
instance.layout = {
|
|
||||||
layout: useAppStoreHook().layout,
|
|
||||||
theme: instance.layout.theme,
|
|
||||||
darkMode: dataTheme.value,
|
|
||||||
sidebarStatus: instance.layout.sidebarStatus,
|
|
||||||
epThemeColor: instance.layout.epThemeColor
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//初始化项目配置
|
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
settings.greyVal &&
|
settings.greyVal &&
|
||||||
document.querySelector("html")?.setAttribute("class", "html-grey");
|
document.querySelector("html")?.setAttribute("class", "html-grey");
|
||||||
settings.weakVal &&
|
settings.weakVal &&
|
||||||
document.querySelector("html")?.setAttribute("class", "html-weakness");
|
document.querySelector("html")?.setAttribute("class", "html-weakness");
|
||||||
settings.tabsVal && tagsChange();
|
settings.tabsVal && tagsChange();
|
||||||
|
|
||||||
writeNewStyle(createNewStyle(epThemeColor.value));
|
|
||||||
dataThemeChange();
|
dataThemeChange();
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
@@ -311,40 +239,61 @@ nextTick(() => {
|
|||||||
:active-icon="dayIcon"
|
:active-icon="dayIcon"
|
||||||
:inactive-icon="darkIcon"
|
:inactive-icon="darkIcon"
|
||||||
@change="dataThemeChange"
|
@change="dataThemeChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
|
|
||||||
<el-divider>导航栏模式</el-divider>
|
<el-divider>导航栏模式</el-divider>
|
||||||
<ul class="pure-theme">
|
<ul class="pure-theme">
|
||||||
<el-tooltip class="item" content="左侧菜单模式" placement="bottom">
|
<el-tooltip class="item" content="左侧模式" placement="bottom">
|
||||||
<li
|
<li
|
||||||
:class="layoutTheme.layout === 'vertical' ? $style.isSelect : ''"
|
:class="layoutTheme.layout === 'vertical' ? $style.isSelect : ''"
|
||||||
ref="verticalRef"
|
ref="verticalRef"
|
||||||
@click="setLayoutModel('vertical')"
|
@click="setLayoutModel('vertical')"
|
||||||
>
|
>
|
||||||
<div></div>
|
<div />
|
||||||
<div></div>
|
<div />
|
||||||
</li>
|
</li>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
|
|
||||||
<el-tooltip class="item" content="顶部菜单模式" placement="bottom">
|
<el-tooltip
|
||||||
|
v-if="device !== 'mobile'"
|
||||||
|
class="item"
|
||||||
|
content="顶部模式"
|
||||||
|
placement="bottom"
|
||||||
|
>
|
||||||
<li
|
<li
|
||||||
:class="layoutTheme.layout === 'horizontal' ? $style.isSelect : ''"
|
:class="layoutTheme.layout === 'horizontal' ? $style.isSelect : ''"
|
||||||
ref="horizontalRef"
|
ref="horizontalRef"
|
||||||
@click="setLayoutModel('horizontal')"
|
@click="setLayoutModel('horizontal')"
|
||||||
>
|
>
|
||||||
<div></div>
|
<div />
|
||||||
<div></div>
|
<div />
|
||||||
|
</li>
|
||||||
|
</el-tooltip>
|
||||||
|
|
||||||
|
<el-tooltip
|
||||||
|
v-if="device !== 'mobile'"
|
||||||
|
class="item"
|
||||||
|
content="混合模式"
|
||||||
|
placement="bottom"
|
||||||
|
>
|
||||||
|
<li
|
||||||
|
:class="layoutTheme.layout === 'mix' ? $style.isSelect : ''"
|
||||||
|
ref="mixRef"
|
||||||
|
@click="setLayoutModel('mix')"
|
||||||
|
>
|
||||||
|
<div />
|
||||||
|
<div />
|
||||||
</li>
|
</li>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<el-divider v-show="!dataTheme">主题色</el-divider>
|
<el-divider>主题色</el-divider>
|
||||||
<ul class="theme-color" v-show="!dataTheme">
|
<ul class="theme-color">
|
||||||
<li
|
<li
|
||||||
v-for="(item, index) in themeColors"
|
v-for="(item, index) in themeColors"
|
||||||
:key="index"
|
:key="index"
|
||||||
:style="getThemeColorStyle(item.rgb)"
|
v-show="showThemeColors(item.themeColor)"
|
||||||
|
:style="getThemeColorStyle(item.color)"
|
||||||
@click="setLayoutThemeColor(item.themeColor)"
|
@click="setLayoutThemeColor(item.themeColor)"
|
||||||
>
|
>
|
||||||
<el-icon
|
<el-icon
|
||||||
@@ -359,8 +308,8 @@ nextTick(() => {
|
|||||||
|
|
||||||
<el-divider>界面显示</el-divider>
|
<el-divider>界面显示</el-divider>
|
||||||
<ul class="setting">
|
<ul class="setting">
|
||||||
<li v-show="!dataTheme">
|
<li>
|
||||||
<span>灰色模式</span>
|
<span class="dark:text-white">灰色模式</span>
|
||||||
<el-switch
|
<el-switch
|
||||||
v-model="settings.greyVal"
|
v-model="settings.greyVal"
|
||||||
inline-prompt
|
inline-prompt
|
||||||
@@ -368,11 +317,10 @@ nextTick(() => {
|
|||||||
active-text="开"
|
active-text="开"
|
||||||
inactive-text="关"
|
inactive-text="关"
|
||||||
@change="greyChange"
|
@change="greyChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
</li>
|
</li>
|
||||||
<li v-show="!dataTheme">
|
<li>
|
||||||
<span>色弱模式</span>
|
<span class="dark:text-white">色弱模式</span>
|
||||||
<el-switch
|
<el-switch
|
||||||
v-model="settings.weakVal"
|
v-model="settings.weakVal"
|
||||||
inline-prompt
|
inline-prompt
|
||||||
@@ -380,11 +328,10 @@ nextTick(() => {
|
|||||||
active-text="开"
|
active-text="开"
|
||||||
inactive-text="关"
|
inactive-text="关"
|
||||||
@change="weekChange"
|
@change="weekChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span>隐藏标签页</span>
|
<span class="dark:text-white">隐藏标签页</span>
|
||||||
<el-switch
|
<el-switch
|
||||||
v-model="settings.tabsVal"
|
v-model="settings.tabsVal"
|
||||||
inline-prompt
|
inline-prompt
|
||||||
@@ -392,11 +339,10 @@ nextTick(() => {
|
|||||||
active-text="开"
|
active-text="开"
|
||||||
inactive-text="关"
|
inactive-text="关"
|
||||||
@change="tagsChange"
|
@change="tagsChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span>侧边栏Logo</span>
|
<span class="dark:text-white">侧边栏Logo</span>
|
||||||
<el-switch
|
<el-switch
|
||||||
v-model="logoVal"
|
v-model="logoVal"
|
||||||
inline-prompt
|
inline-prompt
|
||||||
@@ -406,11 +352,10 @@ nextTick(() => {
|
|||||||
active-text="开"
|
active-text="开"
|
||||||
inactive-text="关"
|
inactive-text="关"
|
||||||
@change="logoChange"
|
@change="logoChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<span>标签页持久化</span>
|
<span class="dark:text-white">标签页持久化</span>
|
||||||
<el-switch
|
<el-switch
|
||||||
v-model="settings.multiTagsCache"
|
v-model="settings.multiTagsCache"
|
||||||
inline-prompt
|
inline-prompt
|
||||||
@@ -418,12 +363,11 @@ nextTick(() => {
|
|||||||
active-text="开"
|
active-text="开"
|
||||||
inactive-text="关"
|
inactive-text="关"
|
||||||
@change="multiTagsCacheChange"
|
@change="multiTagsCacheChange"
|
||||||
>
|
/>
|
||||||
</el-switch>
|
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
<li>
|
<li>
|
||||||
<span>标签风格</span>
|
<span class="dark:text-white">标签风格</span>
|
||||||
<el-radio-group v-model="markValue" size="small" @change="onChange">
|
<el-radio-group v-model="markValue" size="small" @change="onChange">
|
||||||
<el-radio label="card">卡片</el-radio>
|
<el-radio label="card">卡片</el-radio>
|
||||||
<el-radio label="smart">灵动</el-radio>
|
<el-radio label="smart">灵动</el-radio>
|
||||||
@@ -437,9 +381,14 @@ nextTick(() => {
|
|||||||
style="width: 90%; margin: 24px 15px"
|
style="width: 90%; margin: 24px 15px"
|
||||||
@click="onReset"
|
@click="onReset"
|
||||||
>
|
>
|
||||||
<i class="fa fa-sign-out"></i>
|
<IconifyIconOffline
|
||||||
清空缓存并返回登录页</el-button
|
icon="fa-sign-out"
|
||||||
>
|
width="15"
|
||||||
|
height="15"
|
||||||
|
style="margin-right: 4px"
|
||||||
|
/>
|
||||||
|
清空缓存并返回登录页
|
||||||
|
</el-button>
|
||||||
</panel>
|
</panel>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -477,15 +426,14 @@ nextTick(() => {
|
|||||||
.pure-theme {
|
.pure-theme {
|
||||||
margin-top: 25px;
|
margin-top: 25px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100px;
|
height: 50px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: space-around;
|
justify-content: space-around;
|
||||||
|
|
||||||
li {
|
li {
|
||||||
margin: 10px;
|
width: 18%;
|
||||||
width: 36%;
|
height: 45px;
|
||||||
height: 70px;
|
|
||||||
background: #f0f2f5;
|
background: #f0f2f5;
|
||||||
position: relative;
|
position: relative;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
@@ -523,6 +471,27 @@ nextTick(() => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&:nth-child(3) {
|
||||||
|
div {
|
||||||
|
&:nth-child(1) {
|
||||||
|
width: 100%;
|
||||||
|
height: 30%;
|
||||||
|
background: #1b2a47;
|
||||||
|
box-shadow: 0 0 1px #888;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(2) {
|
||||||
|
width: 30%;
|
||||||
|
height: 70%;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 0 1px #888;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, watch } from "vue";
|
import { isEqual } from "lodash-unified";
|
||||||
import { isEqual } from "lodash-es";
|
|
||||||
import { transformI18n } from "/@/plugins/i18n";
|
import { transformI18n } from "/@/plugins/i18n";
|
||||||
|
import { ref, watch, onMounted, toRaw } from "vue";
|
||||||
import { getParentPaths, findRouteByPath } from "/@/router/utils";
|
import { getParentPaths, findRouteByPath } from "/@/router/utils";
|
||||||
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
||||||
import { useRoute, useRouter, RouteLocationMatched } from "vue-router";
|
import { useRoute, useRouter, RouteLocationMatched } from "vue-router";
|
||||||
@@ -9,24 +9,29 @@ import { useRoute, useRouter, RouteLocationMatched } from "vue-router";
|
|||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const levelList = ref([]);
|
const levelList = ref([]);
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const routes = router.options.routes;
|
const routes: any = router.options.routes;
|
||||||
const multiTags = useMultiTagsStoreHook().multiTags;
|
const multiTags: any = useMultiTagsStoreHook().multiTags;
|
||||||
|
|
||||||
const isDashboard = (route: RouteLocationMatched): boolean | string => {
|
const isDashboard = (route: RouteLocationMatched): boolean | string => {
|
||||||
const name = route && (route.name as string);
|
const name = route && (route.name as string);
|
||||||
if (!name) {
|
if (!name) return false;
|
||||||
return false;
|
return name.trim().toLocaleLowerCase() === "Welcome".toLocaleLowerCase();
|
||||||
}
|
|
||||||
return name.trim().toLocaleLowerCase() === "welcome".toLocaleLowerCase();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getBreadcrumb = (): void => {
|
const getBreadcrumb = (): void => {
|
||||||
// 当前路由信息
|
// 当前路由信息
|
||||||
let currentRoute;
|
let currentRoute;
|
||||||
|
|
||||||
if (Object.keys(route.query).length > 0) {
|
if (Object.keys(route.query).length > 0) {
|
||||||
multiTags.forEach(item => {
|
multiTags.forEach(item => {
|
||||||
if (isEqual(route.query, item?.query)) {
|
if (isEqual(route.query, item?.query)) {
|
||||||
currentRoute = item;
|
currentRoute = toRaw(item);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (Object.keys(route.params).length > 0) {
|
||||||
|
multiTags.forEach(item => {
|
||||||
|
if (isEqual(route.params, item?.params)) {
|
||||||
|
currentRoute = toRaw(item);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
@@ -38,91 +43,66 @@ const getBreadcrumb = (): void => {
|
|||||||
let matched = [];
|
let matched = [];
|
||||||
// 获取每个父级路径对应的路由信息
|
// 获取每个父级路径对应的路由信息
|
||||||
parentRoutes.forEach(path => {
|
parentRoutes.forEach(path => {
|
||||||
if (path !== "/") {
|
if (path !== "/") matched.push(findRouteByPath(path, routes));
|
||||||
matched.push(findRouteByPath(path, routes));
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
if (router.currentRoute.value.meta?.refreshRedirect) {
|
|
||||||
matched.unshift(
|
|
||||||
findRouteByPath(
|
|
||||||
router.currentRoute.value.meta.refreshRedirect as string,
|
|
||||||
routes
|
|
||||||
)
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
// 过滤与子级相同标题的父级路由
|
|
||||||
matched = matched.filter(item => {
|
|
||||||
return !item.redirect || (item.redirect && item.children.length !== 1);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (currentRoute?.path !== "/welcome") {
|
|
||||||
matched.push(currentRoute);
|
|
||||||
}
|
|
||||||
|
|
||||||
const first = matched[0];
|
if (currentRoute?.path !== "/welcome") matched.push(currentRoute);
|
||||||
if (!isDashboard(first)) {
|
|
||||||
|
if (!isDashboard(matched[0])) {
|
||||||
matched = [
|
matched = [
|
||||||
{
|
{
|
||||||
path: "/welcome",
|
path: "/welcome",
|
||||||
parentPath: "/",
|
parentPath: "/",
|
||||||
meta: { title: "menus.hshome", i18n: true }
|
meta: { title: "menus.hshome" }
|
||||||
} as unknown as RouteLocationMatched
|
} as unknown as RouteLocationMatched
|
||||||
].concat(matched);
|
].concat(matched);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
matched.forEach((item, index) => {
|
||||||
|
if (currentRoute?.query || currentRoute?.params) return;
|
||||||
|
if (item?.children) {
|
||||||
|
item.children.forEach(v => {
|
||||||
|
if (v?.meta?.title === item?.meta?.title) {
|
||||||
|
matched.splice(index, 1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
levelList.value = matched.filter(
|
levelList.value = matched.filter(
|
||||||
item => item?.meta && item?.meta.title !== false
|
item => item?.meta && item?.meta.title !== false
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
getBreadcrumb();
|
const handleLink = (item: RouteLocationMatched): void => {
|
||||||
|
const { redirect, path } = item;
|
||||||
|
if (redirect) {
|
||||||
|
router.push(redirect as any);
|
||||||
|
} else {
|
||||||
|
router.push(path);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
getBreadcrumb();
|
||||||
|
});
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => route.path,
|
() => route.path,
|
||||||
() => getBreadcrumb()
|
() => {
|
||||||
);
|
getBreadcrumb();
|
||||||
|
|
||||||
watch(
|
|
||||||
() => route.query,
|
|
||||||
() => getBreadcrumb()
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleLink = (item: RouteLocationMatched): any => {
|
|
||||||
const { redirect, path } = item;
|
|
||||||
if (redirect) {
|
|
||||||
router.push(redirect.toString());
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
router.push(path);
|
);
|
||||||
};
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<el-breadcrumb class="app-breadcrumb" separator="/">
|
<el-breadcrumb class="!leading-[50px] select-none" separator="/">
|
||||||
<transition-group appear name="breadcrumb">
|
<transition-group appear name="breadcrumb">
|
||||||
<el-breadcrumb-item v-for="(item, index) in levelList" :key="item.path">
|
<el-breadcrumb-item v-for="item in levelList" :key="item.path">
|
||||||
<span
|
<a @click.prevent="handleLink(item)">
|
||||||
v-if="item.redirect === 'noRedirect' || index == levelList.length - 1"
|
{{ transformI18n(item.meta.title) }}
|
||||||
class="no-redirect"
|
|
||||||
>{{ transformI18n(item.meta.title, item.meta.i18n) }}</span
|
|
||||||
>
|
|
||||||
<a v-else @click.prevent="handleLink(item)">
|
|
||||||
{{ transformI18n(item.meta.title, item.meta.i18n) }}
|
|
||||||
</a>
|
</a>
|
||||||
</el-breadcrumb-item>
|
</el-breadcrumb-item>
|
||||||
</transition-group>
|
</transition-group>
|
||||||
</el-breadcrumb>
|
</el-breadcrumb>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
|
||||||
.app-breadcrumb.el-breadcrumb {
|
|
||||||
display: inline-block;
|
|
||||||
font-size: 14px;
|
|
||||||
line-height: 50px;
|
|
||||||
|
|
||||||
.no-redirect {
|
|
||||||
color: #97a8be;
|
|
||||||
cursor: text;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
@@ -1,63 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
import { ref } from "vue";
|
|
||||||
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
|
||||||
export interface Props {
|
|
||||||
isActive: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = withDefaults(defineProps<Props>(), {
|
|
||||||
isActive: false
|
|
||||||
});
|
|
||||||
|
|
||||||
const fillColor = ref<string>("");
|
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
(e: "toggleClick"): void;
|
|
||||||
}>();
|
|
||||||
|
|
||||||
const toggleClick = () => {
|
|
||||||
emit("toggleClick");
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<div
|
|
||||||
:class="classes.container"
|
|
||||||
:title="props.isActive ? '点击折叠' : '点击展开'"
|
|
||||||
@click="toggleClick"
|
|
||||||
@mouseenter="fillColor = useEpThemeStoreHook().epThemeColor"
|
|
||||||
@mouseleave="fillColor = ''"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
:fill="fillColor"
|
|
||||||
:class="['hamburger', props.isActive ? 'is-active' : '']"
|
|
||||||
viewBox="0 0 1024 1024"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="64"
|
|
||||||
height="64"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M408 442h480c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8H408c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8zm-8 204c0 4.4 3.6 8 8 8h480c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8H408c-4.4 0-8 3.6-8 8v56zm504-486H120c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8h784c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8zm0 632H120c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8h784c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8zM142.4 642.1L298.7 519a8.84 8.84 0 0 0 0-13.9L142.4 381.9c-5.8-4.6-14.4-.5-14.4 6.9v246.3a8.9 8.9 0 0 0 14.4 7z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style module="classes" scoped>
|
|
||||||
.container {
|
|
||||||
padding: 0 15px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.hamburger {
|
|
||||||
display: inline-block;
|
|
||||||
vertical-align: middle;
|
|
||||||
width: 20px;
|
|
||||||
height: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.is-active {
|
|
||||||
transform: rotate(180deg);
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -1,144 +1,54 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
import { ref, watch } from "vue";
|
||||||
computed,
|
import Search from "../search/index.vue";
|
||||||
unref,
|
|
||||||
watch,
|
|
||||||
nextTick,
|
|
||||||
onMounted,
|
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
|
||||||
import { useI18n } from "vue-i18n";
|
|
||||||
import { emitter } from "/@/utils/mitt";
|
|
||||||
import Notice from "../notice/index.vue";
|
import Notice from "../notice/index.vue";
|
||||||
import { templateRef } from "@vueuse/core";
|
|
||||||
import SidebarItem from "./sidebarItem.vue";
|
import SidebarItem from "./sidebarItem.vue";
|
||||||
import avatars from "/@/assets/avatars.jpg";
|
import avatars from "/@/assets/avatars.jpg";
|
||||||
import { algorithm } from "/@/utils/algorithm";
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import screenfull from "../screenfull/index.vue";
|
import screenfull from "../screenfull/index.vue";
|
||||||
import { useRoute, useRouter } from "vue-router";
|
import { deviceDetection } from "@pureadmin/utils";
|
||||||
import { storageSession } from "/@/utils/storage";
|
import { useTranslationLang } from "../../hooks/useTranslationLang";
|
||||||
import Icon from "/@/components/ReIcon/src/Icon.vue";
|
|
||||||
import { deviceDetection } from "/@/utils/deviceDetection";
|
|
||||||
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
||||||
import globalization from "/@/assets/svg/globalization.svg?component";
|
import globalization from "/@/assets/svg/globalization.svg?component";
|
||||||
|
|
||||||
const instance =
|
const menuRef = ref();
|
||||||
getCurrentInstance().appContext.config.globalProperties.$storage;
|
|
||||||
|
|
||||||
const title =
|
const { t, route, locale, translationCh, translationEn } =
|
||||||
getCurrentInstance().appContext.config.globalProperties.$config?.Title;
|
useTranslationLang(menuRef);
|
||||||
|
const {
|
||||||
const menuRef = templateRef<ElRef | null>("menu", null);
|
title,
|
||||||
const route = useRoute();
|
routers,
|
||||||
const router = useRouter();
|
logout,
|
||||||
const routers = useRouter().options.routes;
|
backHome,
|
||||||
let usename = storageSession.getItem("info")?.username;
|
onPanel,
|
||||||
const { locale, t } = useI18n();
|
menuSelect,
|
||||||
|
username,
|
||||||
const getDropdownItemStyle = computed(() => {
|
avatarsStyle,
|
||||||
return t => {
|
getDropdownItemStyle,
|
||||||
return {
|
getDropdownItemClass
|
||||||
background: locale.value === t ? "#1b2a47" : "",
|
} = useNav();
|
||||||
color: locale.value === t ? "#f4f4f5" : "#000"
|
|
||||||
};
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => locale.value,
|
() => route.path,
|
||||||
() => {
|
() => {
|
||||||
//@ts-ignore
|
menuSelect(route.path, routers);
|
||||||
// 动态title
|
|
||||||
document.title = t(unref(route.meta.title));
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
// 退出登录
|
|
||||||
const logout = (): void => {
|
|
||||||
storageSession.removeItem("info");
|
|
||||||
router.push("/login");
|
|
||||||
};
|
|
||||||
|
|
||||||
function onPanel() {
|
|
||||||
emitter.emit("openPanel");
|
|
||||||
}
|
|
||||||
|
|
||||||
const activeMenu = computed((): string => {
|
|
||||||
const { meta, path } = route;
|
|
||||||
if (meta.activeMenu) {
|
|
||||||
// @ts-ignore
|
|
||||||
return meta.activeMenu;
|
|
||||||
}
|
|
||||||
return path;
|
|
||||||
});
|
|
||||||
|
|
||||||
const menuSelect = (indexPath: string): void => {
|
|
||||||
let parentPath = "";
|
|
||||||
let parentPathIndex = indexPath.lastIndexOf("/");
|
|
||||||
if (parentPathIndex > 0) {
|
|
||||||
parentPath = indexPath.slice(0, parentPathIndex);
|
|
||||||
}
|
|
||||||
// 找到当前路由的信息
|
|
||||||
function findCurrentRoute(routes) {
|
|
||||||
return routes.map(item => {
|
|
||||||
if (item.path === indexPath) {
|
|
||||||
// 切换左侧菜单 通知标签页
|
|
||||||
emitter.emit("changLayoutRoute", {
|
|
||||||
indexPath,
|
|
||||||
parentPath
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
if (item.children) findCurrentRoute(item.children);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
findCurrentRoute(algorithm.increaseIndexes(routers));
|
|
||||||
};
|
|
||||||
|
|
||||||
function backHome() {
|
|
||||||
router.push("/welcome");
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleResize() {
|
|
||||||
// @ts-ignore
|
|
||||||
menuRef.value.handleResize();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 简体中文
|
|
||||||
function translationCh() {
|
|
||||||
instance.locale = { locale: "zh" };
|
|
||||||
locale.value = "zh";
|
|
||||||
handleResize();
|
|
||||||
}
|
|
||||||
|
|
||||||
// English
|
|
||||||
function translationEn() {
|
|
||||||
instance.locale = { locale: "en" };
|
|
||||||
locale.value = "en";
|
|
||||||
handleResize();
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => {
|
|
||||||
nextTick(() => {
|
|
||||||
handleResize();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="horizontal-header">
|
<div class="horizontal-header">
|
||||||
<div class="horizontal-header-left" @click="backHome">
|
<div class="horizontal-header-left" @click="backHome">
|
||||||
<Icon svg :width="35" :height="35" content="team-iconlogo" />
|
<FontIcon icon="team-iconlogo" svg style="width: 35px; height: 35px" />
|
||||||
<h4>{{ title }}</h4>
|
<h4>{{ title }}</h4>
|
||||||
</div>
|
</div>
|
||||||
<el-menu
|
<el-menu
|
||||||
ref="menu"
|
|
||||||
:default-active="activeMenu"
|
|
||||||
unique-opened
|
|
||||||
router
|
router
|
||||||
class="horizontal-header-menu"
|
ref="menuRef"
|
||||||
mode="horizontal"
|
mode="horizontal"
|
||||||
@select="menuSelect"
|
class="horizontal-header-menu"
|
||||||
|
:default-active="route.path"
|
||||||
|
@select="indexPath => menuSelect(indexPath, routers)"
|
||||||
>
|
>
|
||||||
<sidebar-item
|
<sidebar-item
|
||||||
v-for="route in usePermissionStoreHook().wholeMenus"
|
v-for="route in usePermissionStoreHook().wholeMenus"
|
||||||
@@ -148,68 +58,75 @@ onMounted(() => {
|
|||||||
/>
|
/>
|
||||||
</el-menu>
|
</el-menu>
|
||||||
<div class="horizontal-header-right">
|
<div class="horizontal-header-right">
|
||||||
|
<!-- 菜单搜索 -->
|
||||||
|
<Search />
|
||||||
<!-- 通知 -->
|
<!-- 通知 -->
|
||||||
<Notice id="header-notice" />
|
<Notice id="header-notice" />
|
||||||
<!-- 全屏 -->
|
<!-- 全屏 -->
|
||||||
<screenfull id="header-screenfull" v-show="!deviceDetection()" />
|
<screenfull id="header-screenfull" v-show="!deviceDetection()" />
|
||||||
<!-- 国际化 -->
|
<!-- 国际化 -->
|
||||||
<el-dropdown id="header-translation" trigger="click">
|
<el-dropdown id="header-translation" trigger="click">
|
||||||
<globalization />
|
<globalization
|
||||||
|
class="navbar-bg-hover w-[40px] h-[48px] p-[11px] cursor-pointer outline-none"
|
||||||
|
/>
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu class="translation">
|
<el-dropdown-menu class="translation">
|
||||||
<el-dropdown-item
|
<el-dropdown-item
|
||||||
:style="getDropdownItemStyle('zh')"
|
:style="getDropdownItemStyle(locale, 'zh')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
|
||||||
@click="translationCh"
|
@click="translationCh"
|
||||||
><el-icon class="check-zh" v-show="locale === 'zh'"
|
|
||||||
><IconifyIconOffline icon="check" /></el-icon
|
|
||||||
>简体中文</el-dropdown-item
|
|
||||||
>
|
>
|
||||||
|
<span class="check-zh" v-show="locale === 'zh'">
|
||||||
|
<IconifyIconOffline icon="check" />
|
||||||
|
</span>
|
||||||
|
简体中文
|
||||||
|
</el-dropdown-item>
|
||||||
<el-dropdown-item
|
<el-dropdown-item
|
||||||
:style="getDropdownItemStyle('en')"
|
:style="getDropdownItemStyle(locale, 'en')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
|
||||||
@click="translationEn"
|
@click="translationEn"
|
||||||
><el-icon class="check-en" v-show="locale === 'en'"
|
|
||||||
><IconifyIconOffline icon="check" /></el-icon
|
|
||||||
>English</el-dropdown-item
|
|
||||||
>
|
>
|
||||||
|
<span class="check-en" v-show="locale === 'en'">
|
||||||
|
<IconifyIconOffline icon="check" />
|
||||||
|
</span>
|
||||||
|
English
|
||||||
|
</el-dropdown-item>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
<!-- 退出登陆 -->
|
<!-- 退出登录 -->
|
||||||
<el-dropdown trigger="click">
|
<el-dropdown trigger="click">
|
||||||
<span class="el-dropdown-link">
|
<span class="el-dropdown-link navbar-bg-hover">
|
||||||
<img :src="avatars" />
|
<img v-if="avatars" :src="avatars" :style="avatarsStyle" />
|
||||||
<p>{{ usename }}</p>
|
<p v-if="username" class="dark:text-white">{{ username }}</p>
|
||||||
</span>
|
</span>
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu class="logout">
|
<el-dropdown-menu class="logout">
|
||||||
<el-dropdown-item @click="logout">
|
<el-dropdown-item @click="logout">
|
||||||
<i class="ri-logout-circle-r-line"></i
|
<IconifyIconOffline
|
||||||
>{{ $t("buttons.hsLoginOut") }}</el-dropdown-item
|
icon="logout-circle-r-line"
|
||||||
>
|
style="margin: 5px"
|
||||||
|
/>
|
||||||
|
{{ t("buttons.hsLoginOut") }}
|
||||||
|
</el-dropdown-item>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
<el-icon
|
<span
|
||||||
class="el-icon-setting"
|
class="el-icon-setting navbar-bg-hover"
|
||||||
:title="$t('buttons.hssystemSet')"
|
:title="t('buttons.hssystemSet')"
|
||||||
@click="onPanel"
|
@click="onPanel"
|
||||||
>
|
>
|
||||||
<IconifyIconOffline icon="setting" />
|
<IconifyIconOffline icon="setting" />
|
||||||
</el-icon>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.translation {
|
.translation {
|
||||||
.el-dropdown-menu__item {
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
padding: 5px 40px !important;
|
padding: 5px 40px;
|
||||||
}
|
|
||||||
|
|
||||||
.el-dropdown-menu__item:focus,
|
|
||||||
.el-dropdown-menu__item:not(.is-disabled):hover {
|
|
||||||
color: #606266;
|
|
||||||
background: #f0f0f0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.check-zh {
|
.check-zh {
|
||||||
@@ -226,16 +143,10 @@ onMounted(() => {
|
|||||||
.logout {
|
.logout {
|
||||||
max-width: 120px;
|
max-width: 120px;
|
||||||
|
|
||||||
.el-dropdown-menu__item {
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
min-width: 100%;
|
min-width: 100%;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.el-dropdown-menu__item:focus,
|
|
||||||
.el-dropdown-menu__item:not(.is-disabled):hover {
|
|
||||||
color: #606266;
|
|
||||||
background: #f0f0f0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
47
src/layout/components/sidebar/leftCollapse.vue
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { useDark } from "@pureadmin/utils";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
isActive: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<Props>(), {
|
||||||
|
isActive: false
|
||||||
|
});
|
||||||
|
const { isDark } = useDark();
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: "toggleClick"): void;
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const toggleClick = () => {
|
||||||
|
emit("toggleClick");
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="container">
|
||||||
|
<el-tooltip
|
||||||
|
placement="right"
|
||||||
|
:effect="isDark ? 'dark' : 'light'"
|
||||||
|
:content="props.isActive ? '点击折叠' : '点击展开'"
|
||||||
|
>
|
||||||
|
<IconifyIconOffline
|
||||||
|
:icon="props.isActive ? 'menu-fold' : 'menu-unfold'"
|
||||||
|
class="cursor-pointer inline-block align-middle text-primary hover:text-primary dark:hover:!text-white w-[16px] h-[16px] ml-4 mb-1"
|
||||||
|
@click="toggleClick"
|
||||||
|
/>
|
||||||
|
</el-tooltip>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.container {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
box-shadow: 0 0 6px -2px var(--el-color-primary);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -1,12 +1,11 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { getCurrentInstance } from "vue";
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import Icon from "/@/components/ReIcon/src/Icon.vue";
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
collapse: Boolean
|
collapse: Boolean
|
||||||
});
|
});
|
||||||
|
|
||||||
const title =
|
const { title } = useNav();
|
||||||
getCurrentInstance().appContext.config.globalProperties.$config?.Title;
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -19,7 +18,7 @@ const title =
|
|||||||
class="sidebar-logo-link"
|
class="sidebar-logo-link"
|
||||||
to="/"
|
to="/"
|
||||||
>
|
>
|
||||||
<Icon svg :width="35" :height="35" content="team-iconlogo" />
|
<FontIcon icon="team-iconlogo" svg style="width: 35px; height: 35px" />
|
||||||
<span class="sidebar-title">{{ title }}</span>
|
<span class="sidebar-title">{{ title }}</span>
|
||||||
</router-link>
|
</router-link>
|
||||||
<router-link
|
<router-link
|
||||||
@@ -29,7 +28,7 @@ const title =
|
|||||||
class="sidebar-logo-link"
|
class="sidebar-logo-link"
|
||||||
to="/"
|
to="/"
|
||||||
>
|
>
|
||||||
<Icon svg :width="35" :height="35" content="team-iconlogo" />
|
<FontIcon icon="team-iconlogo" svg style="width: 35px; height: 35px" />
|
||||||
<span class="sidebar-title">{{ title }}</span>
|
<span class="sidebar-title">{{ title }}</span>
|
||||||
</router-link>
|
</router-link>
|
||||||
</transition>
|
</transition>
|
||||||
|
|||||||
184
src/layout/components/sidebar/mixNav.vue
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import Search from "../search/index.vue";
|
||||||
|
import Notice from "../notice/index.vue";
|
||||||
|
import avatars from "/@/assets/avatars.jpg";
|
||||||
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
|
import { transformI18n } from "/@/plugins/i18n";
|
||||||
|
import screenfull from "../screenfull/index.vue";
|
||||||
|
import { deviceDetection } from "@pureadmin/utils";
|
||||||
|
import { ref, toRaw, watch, onMounted } from "vue";
|
||||||
|
import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
|
||||||
|
import { getParentPaths, findRouteByPath } from "/@/router/utils";
|
||||||
|
import { useTranslationLang } from "../../hooks/useTranslationLang";
|
||||||
|
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
||||||
|
import globalization from "/@/assets/svg/globalization.svg?component";
|
||||||
|
|
||||||
|
const menuRef = ref();
|
||||||
|
let defaultActive = ref(null);
|
||||||
|
|
||||||
|
const { t, route, locale, translationCh, translationEn } =
|
||||||
|
useTranslationLang(menuRef);
|
||||||
|
const {
|
||||||
|
device,
|
||||||
|
routers,
|
||||||
|
logout,
|
||||||
|
onPanel,
|
||||||
|
menuSelect,
|
||||||
|
resolvePath,
|
||||||
|
username,
|
||||||
|
avatarsStyle,
|
||||||
|
getDropdownItemStyle,
|
||||||
|
getDropdownItemClass
|
||||||
|
} = useNav();
|
||||||
|
|
||||||
|
function getDefaultActive(routePath) {
|
||||||
|
const wholeMenus = usePermissionStoreHook().wholeMenus;
|
||||||
|
/** 当前路由的父级路径 */
|
||||||
|
const parentRoutes = getParentPaths(routePath, wholeMenus)[0];
|
||||||
|
defaultActive.value = findRouteByPath(
|
||||||
|
parentRoutes,
|
||||||
|
wholeMenus
|
||||||
|
)?.children[0]?.path;
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
getDefaultActive(route.path);
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
getDefaultActive(route.path);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div v-if="device !== 'mobile'" class="horizontal-header">
|
||||||
|
<el-menu
|
||||||
|
router
|
||||||
|
ref="menuRef"
|
||||||
|
mode="horizontal"
|
||||||
|
class="horizontal-header-menu"
|
||||||
|
:default-active="defaultActive"
|
||||||
|
@select="indexPath => menuSelect(indexPath, routers)"
|
||||||
|
>
|
||||||
|
<el-menu-item
|
||||||
|
v-for="route in usePermissionStoreHook().wholeMenus"
|
||||||
|
:key="route.path"
|
||||||
|
:index="resolvePath(route) || route.redirect"
|
||||||
|
>
|
||||||
|
<template #title>
|
||||||
|
<div
|
||||||
|
v-if="toRaw(route.meta.icon)"
|
||||||
|
:class="['sub-menu-icon', route.meta.icon]"
|
||||||
|
>
|
||||||
|
<component
|
||||||
|
:is="useRenderIcon(route.meta && toRaw(route.meta.icon))"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span class="select-none">{{ transformI18n(route.meta.title) }}</span>
|
||||||
|
<FontIcon
|
||||||
|
v-if="route.meta.extraIcon"
|
||||||
|
width="30px"
|
||||||
|
height="30px"
|
||||||
|
style="position: absolute; right: 10px"
|
||||||
|
:icon="route.meta.extraIcon.name"
|
||||||
|
:svg="route.meta.extraIcon.svg ? true : false"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</el-menu-item>
|
||||||
|
</el-menu>
|
||||||
|
<div class="horizontal-header-right">
|
||||||
|
<!-- 菜单搜索 -->
|
||||||
|
<Search />
|
||||||
|
<!-- 通知 -->
|
||||||
|
<Notice id="header-notice" />
|
||||||
|
<!-- 全屏 -->
|
||||||
|
<screenfull id="header-screenfull" v-show="!deviceDetection()" />
|
||||||
|
<!-- 国际化 -->
|
||||||
|
<el-dropdown id="header-translation" trigger="click">
|
||||||
|
<globalization
|
||||||
|
class="navbar-bg-hover w-[40px] h-[48px] p-[11px] cursor-pointer outline-none"
|
||||||
|
/>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu class="translation">
|
||||||
|
<el-dropdown-item
|
||||||
|
:style="getDropdownItemStyle(locale, 'zh')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
|
||||||
|
@click="translationCh"
|
||||||
|
>
|
||||||
|
<span class="check-zh" v-show="locale === 'zh'">
|
||||||
|
<IconifyIconOffline icon="check" />
|
||||||
|
</span>
|
||||||
|
简体中文
|
||||||
|
</el-dropdown-item>
|
||||||
|
<el-dropdown-item
|
||||||
|
:style="getDropdownItemStyle(locale, 'en')"
|
||||||
|
:class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
|
||||||
|
@click="translationEn"
|
||||||
|
>
|
||||||
|
<span class="check-en" v-show="locale === 'en'">
|
||||||
|
<IconifyIconOffline icon="check" />
|
||||||
|
</span>
|
||||||
|
English
|
||||||
|
</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<!-- 退出登录 -->
|
||||||
|
<el-dropdown trigger="click">
|
||||||
|
<span class="el-dropdown-link navbar-bg-hover">
|
||||||
|
<img v-if="avatars" :src="avatars" :style="avatarsStyle" />
|
||||||
|
<p v-if="username" class="dark:text-white">{{ username }}</p>
|
||||||
|
</span>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu class="logout">
|
||||||
|
<el-dropdown-item @click="logout">
|
||||||
|
<IconifyIconOffline
|
||||||
|
icon="logout-circle-r-line"
|
||||||
|
style="margin: 5px"
|
||||||
|
/>
|
||||||
|
{{ t("buttons.hsLoginOut") }}
|
||||||
|
</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<span
|
||||||
|
class="el-icon-setting navbar-bg-hover"
|
||||||
|
:title="t('buttons.hssystemSet')"
|
||||||
|
@click="onPanel"
|
||||||
|
>
|
||||||
|
<IconifyIconOffline icon="setting" />
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.translation {
|
||||||
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
|
padding: 5px 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.check-zh {
|
||||||
|
position: absolute;
|
||||||
|
left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.check-en {
|
||||||
|
position: absolute;
|
||||||
|
left: 20px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.logout {
|
||||||
|
max-width: 120px;
|
||||||
|
|
||||||
|
::v-deep(.el-dropdown-menu__item) {
|
||||||
|
min-width: 100%;
|
||||||
|
display: inline-flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -1,22 +1,12 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
|
||||||
ref,
|
|
||||||
PropType,
|
|
||||||
nextTick,
|
|
||||||
computed,
|
|
||||||
CSSProperties,
|
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import { childrenType } from "../../types";
|
import { childrenType } from "../../types";
|
||||||
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import { transformI18n } from "/@/plugins/i18n";
|
import { transformI18n } from "/@/plugins/i18n";
|
||||||
import { findIconReg } from "/@/components/ReIcon";
|
import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
|
||||||
import Icon from "/@/components/ReIcon/src/Icon.vue";
|
import { ref, toRaw, PropType, nextTick, computed, CSSProperties } from "vue";
|
||||||
import { useAppStoreHook } from "/@/store/modules/app";
|
|
||||||
|
|
||||||
const instance = getCurrentInstance().appContext.app.config.globalProperties;
|
const { layout, isCollapse } = useNav();
|
||||||
const menuMode = instance.$storage.layout?.layout === "vertical";
|
|
||||||
const pureApp = useAppStoreHook();
|
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
item: {
|
item: {
|
||||||
@@ -32,6 +22,19 @@ const props = defineProps({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const getExtraIconStyle = computed((): CSSProperties => {
|
||||||
|
if (!isCollapse.value) {
|
||||||
|
return {
|
||||||
|
position: "absolute",
|
||||||
|
right: "10px"
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
position: "static"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const getNoDropdownStyle = computed((): CSSProperties => {
|
const getNoDropdownStyle = computed((): CSSProperties => {
|
||||||
return {
|
return {
|
||||||
display: "flex",
|
display: "flex",
|
||||||
@@ -41,7 +44,7 @@ const getNoDropdownStyle = computed((): CSSProperties => {
|
|||||||
|
|
||||||
const getDivStyle = computed((): CSSProperties => {
|
const getDivStyle = computed((): CSSProperties => {
|
||||||
return {
|
return {
|
||||||
width: pureApp.sidebar.opened ? "" : "100%",
|
width: !isCollapse.value ? "" : "100%",
|
||||||
display: "flex",
|
display: "flex",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
@@ -49,9 +52,8 @@ const getDivStyle = computed((): CSSProperties => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
const getMenuTextStyle = computed((): CSSProperties => {
|
const getMenuTextStyle = computed(() => {
|
||||||
return {
|
return {
|
||||||
width: pureApp.sidebar.opened ? "125px" : "",
|
|
||||||
overflow: "hidden",
|
overflow: "hidden",
|
||||||
textOverflow: "ellipsis",
|
textOverflow: "ellipsis",
|
||||||
outline: "none"
|
outline: "none"
|
||||||
@@ -60,14 +62,14 @@ const getMenuTextStyle = computed((): CSSProperties => {
|
|||||||
|
|
||||||
const getSubTextStyle = computed((): CSSProperties => {
|
const getSubTextStyle = computed((): CSSProperties => {
|
||||||
return {
|
return {
|
||||||
width: pureApp.sidebar.opened ? "125px" : "",
|
width: !isCollapse.value ? "210px" : "",
|
||||||
display: "inline-block",
|
display: "inline-block",
|
||||||
overflow: "hidden",
|
overflow: "hidden",
|
||||||
textOverflow: "ellipsis"
|
textOverflow: "ellipsis"
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
const getSpanStyle = computed((): CSSProperties => {
|
const getSpanStyle = computed(() => {
|
||||||
return {
|
return {
|
||||||
overflow: "hidden",
|
overflow: "hidden",
|
||||||
textOverflow: "ellipsis"
|
textOverflow: "ellipsis"
|
||||||
@@ -93,7 +95,6 @@ function hoverMenu(key) {
|
|||||||
: Object.assign(key, {
|
: Object.assign(key, {
|
||||||
showTooltip: false
|
showTooltip: false
|
||||||
});
|
});
|
||||||
|
|
||||||
hoverMenuMap.set(key, true);
|
hoverMenuMap.set(key, true);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -107,6 +108,10 @@ function hasOneShowingChild(
|
|||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (showingChildren[0]?.meta?.showParent) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (showingChildren.length === 1) {
|
if (showingChildren.length === 1) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -120,8 +125,8 @@ function hasOneShowingChild(
|
|||||||
|
|
||||||
function resolvePath(routePath) {
|
function resolvePath(routePath) {
|
||||||
const httpReg = /^http(s?):\/\//;
|
const httpReg = /^http(s?):\/\//;
|
||||||
if (httpReg.test(routePath)) {
|
if (httpReg.test(routePath) || httpReg.test(props.basePath)) {
|
||||||
return props.basePath + "/" + routePath;
|
return routePath || props.basePath;
|
||||||
} else {
|
} else {
|
||||||
return path.resolve(props.basePath, routePath);
|
return path.resolve(props.basePath, routePath);
|
||||||
}
|
}
|
||||||
@@ -140,21 +145,43 @@ function resolvePath(routePath) {
|
|||||||
:class="{ 'submenu-title-noDropdown': !isNest }"
|
:class="{ 'submenu-title-noDropdown': !isNest }"
|
||||||
:style="getNoDropdownStyle"
|
:style="getNoDropdownStyle"
|
||||||
>
|
>
|
||||||
<el-icon v-show="props.item.meta.icon">
|
<div class="sub-menu-icon" v-if="toRaw(props.item.meta.icon)">
|
||||||
<component
|
<component
|
||||||
:is="
|
:is="
|
||||||
findIconReg(
|
useRenderIcon(
|
||||||
onlyOneChild.meta.icon ||
|
toRaw(onlyOneChild.meta.icon) ||
|
||||||
(props.item.meta && props.item.meta.icon)
|
(props.item.meta && toRaw(props.item.meta.icon))
|
||||||
)
|
)
|
||||||
"
|
"
|
||||||
></component>
|
/>
|
||||||
</el-icon>
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="
|
||||||
|
isCollapse &&
|
||||||
|
layout === 'vertical' &&
|
||||||
|
props.item?.pathList?.length === 1
|
||||||
|
"
|
||||||
|
:style="getDivStyle"
|
||||||
|
>
|
||||||
|
<span :style="getMenuTextStyle">
|
||||||
|
{{ transformI18n(onlyOneChild.meta.title) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="
|
||||||
|
isCollapse && layout === 'mix' && props.item?.pathList?.length === 2
|
||||||
|
"
|
||||||
|
:style="getDivStyle"
|
||||||
|
>
|
||||||
|
<span :style="getMenuTextStyle">
|
||||||
|
{{ transformI18n(onlyOneChild.meta.title) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
<template #title>
|
<template #title>
|
||||||
<div :style="getDivStyle">
|
<div :style="getDivStyle">
|
||||||
<span v-if="!menuMode">{{
|
<span v-if="layout === 'horizontal'">
|
||||||
transformI18n(onlyOneChild.meta.title, onlyOneChild.meta.i18n)
|
{{ transformI18n(onlyOneChild.meta.title) }}
|
||||||
}}</span>
|
</span>
|
||||||
<el-tooltip
|
<el-tooltip
|
||||||
v-else
|
v-else
|
||||||
placement="top"
|
placement="top"
|
||||||
@@ -162,53 +189,47 @@ function resolvePath(routePath) {
|
|||||||
:disabled="!onlyOneChild.showTooltip"
|
:disabled="!onlyOneChild.showTooltip"
|
||||||
>
|
>
|
||||||
<template #content>
|
<template #content>
|
||||||
{{
|
{{ transformI18n(onlyOneChild.meta.title) }}
|
||||||
transformI18n(onlyOneChild.meta.title, onlyOneChild.meta.i18n)
|
|
||||||
}}
|
|
||||||
</template>
|
</template>
|
||||||
<span
|
<span
|
||||||
ref="menuTextRef"
|
ref="menuTextRef"
|
||||||
:style="getMenuTextStyle"
|
:style="getMenuTextStyle"
|
||||||
@mouseover="hoverMenu(onlyOneChild)"
|
@mouseover="hoverMenu(onlyOneChild)"
|
||||||
>
|
>
|
||||||
{{
|
{{ transformI18n(onlyOneChild.meta.title) }}
|
||||||
transformI18n(onlyOneChild.meta.title, onlyOneChild.meta.i18n)
|
|
||||||
}}
|
|
||||||
</span>
|
</span>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<Icon
|
<FontIcon
|
||||||
v-if="onlyOneChild.meta.extraIcon"
|
v-if="onlyOneChild.meta.extraIcon"
|
||||||
|
width="30px"
|
||||||
|
height="30px"
|
||||||
|
:style="getExtraIconStyle"
|
||||||
|
:icon="onlyOneChild.meta.extraIcon.name"
|
||||||
:svg="onlyOneChild.meta.extraIcon.svg ? true : false"
|
:svg="onlyOneChild.meta.extraIcon.svg ? true : false"
|
||||||
:content="`${onlyOneChild.meta.extraIcon.name}`"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</el-menu-item>
|
</el-menu-item>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<el-sub-menu
|
<el-sub-menu v-else ref="subMenu" :index="resolvePath(props.item.path)">
|
||||||
v-else
|
|
||||||
ref="subMenu"
|
|
||||||
:index="resolvePath(props.item.path)"
|
|
||||||
popper-append-to-body
|
|
||||||
>
|
|
||||||
<template #title>
|
<template #title>
|
||||||
<el-icon v-show="props.item.meta.icon" :class="props.item.meta.icon">
|
<div v-if="toRaw(props.item.meta.icon)" class="sub-menu-icon">
|
||||||
<component
|
<component
|
||||||
:is="findIconReg(props.item.meta && props.item.meta.icon)"
|
:is="useRenderIcon(props.item.meta && toRaw(props.item.meta.icon))"
|
||||||
></component>
|
/>
|
||||||
</el-icon>
|
</div>
|
||||||
<span v-if="!menuMode">{{
|
<span v-if="layout === 'horizontal'">
|
||||||
transformI18n(props.item.meta.title, props.item.meta.i18n)
|
{{ transformI18n(props.item.meta.title) }}
|
||||||
}}</span>
|
</span>
|
||||||
<el-tooltip
|
<el-tooltip
|
||||||
v-else
|
v-else
|
||||||
placement="top"
|
placement="top"
|
||||||
:offset="-10"
|
:offset="-10"
|
||||||
:disabled="!pureApp.sidebar.opened || !props.item.showTooltip"
|
:disabled="!isCollapse || !props.item.showTooltip"
|
||||||
>
|
>
|
||||||
<template #content>
|
<template #content>
|
||||||
{{ transformI18n(props.item.meta.title, props.item.meta.i18n) }}
|
{{ transformI18n(props.item.meta.title) }}
|
||||||
</template>
|
</template>
|
||||||
<div
|
<div
|
||||||
ref="menuTextRef"
|
ref="menuTextRef"
|
||||||
@@ -216,14 +237,17 @@ function resolvePath(routePath) {
|
|||||||
@mouseover="hoverMenu(props.item)"
|
@mouseover="hoverMenu(props.item)"
|
||||||
>
|
>
|
||||||
<span :style="getSpanStyle">
|
<span :style="getSpanStyle">
|
||||||
{{ transformI18n(props.item.meta.title, props.item.meta.i18n) }}
|
{{ transformI18n(props.item.meta.title) }}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<Icon
|
<FontIcon
|
||||||
v-if="props.item.meta.extraIcon"
|
v-if="props.item.meta.extraIcon"
|
||||||
|
width="30px"
|
||||||
|
height="30px"
|
||||||
|
style="position: absolute; right: 10px"
|
||||||
|
:icon="props.item.meta.extraIcon.name"
|
||||||
:svg="props.item.meta.extraIcon.svg ? true : false"
|
:svg="props.item.meta.extraIcon.svg ? true : false"
|
||||||
:content="`${props.item.meta.extraIcon.name}`"
|
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
<sidebar-item
|
<sidebar-item
|
||||||
|
|||||||
30
src/layout/components/sidebar/topCollapse.vue
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
interface Props {
|
||||||
|
isActive: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<Props>(), {
|
||||||
|
isActive: false
|
||||||
|
});
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: "toggleClick"): void;
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const toggleClick = () => {
|
||||||
|
emit("toggleClick");
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="px-3 mr-1 navbar-bg-hover"
|
||||||
|
:title="props.isActive ? '点击折叠' : '点击展开'"
|
||||||
|
@click="toggleClick"
|
||||||
|
>
|
||||||
|
<IconifyIconOffline
|
||||||
|
:icon="props.isActive ? 'menu-fold' : 'menu-unfold'"
|
||||||
|
class="inline-block align-middle hover:text-primary dark:hover:!text-white"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@@ -1,85 +1,93 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import Logo from "./logo.vue";
|
import Logo from "./logo.vue";
|
||||||
|
import { useRoute } from "vue-router";
|
||||||
import { emitter } from "/@/utils/mitt";
|
import { emitter } from "/@/utils/mitt";
|
||||||
import SidebarItem from "./sidebarItem.vue";
|
import SidebarItem from "./sidebarItem.vue";
|
||||||
import { algorithm } from "/@/utils/algorithm";
|
import leftCollapse from "./leftCollapse.vue";
|
||||||
import { storageLocal } from "/@/utils/storage";
|
import type { StorageConfigs } from "/#/index";
|
||||||
import { useRoute, useRouter } from "vue-router";
|
import { useNav } from "/@/layout/hooks/useNav";
|
||||||
import { computed, ref, onBeforeMount } from "vue";
|
import { storageLocal } from "@pureadmin/utils";
|
||||||
import { useAppStoreHook } from "/@/store/modules/app";
|
import { ref, computed, watch, onBeforeMount } from "vue";
|
||||||
|
import { findRouteByPath, getParentPaths } from "/@/router/utils";
|
||||||
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const pureApp = useAppStoreHook();
|
|
||||||
const router = useRouter().options.routes;
|
|
||||||
const showLogo = ref(
|
const showLogo = ref(
|
||||||
storageLocal.getItem("responsive-configure")?.showLogo ?? true
|
storageLocal.getItem<StorageConfigs>("responsive-configure")?.showLogo ?? true
|
||||||
);
|
);
|
||||||
const isCollapse = computed(() => {
|
|
||||||
return !pureApp.getSidebarStatus;
|
const { routers, device, pureApp, isCollapse, menuSelect, toggleSideBar } =
|
||||||
});
|
useNav();
|
||||||
const activeMenu = computed((): string => {
|
|
||||||
const { meta, path } = route;
|
let subMenuData = ref([]);
|
||||||
if (meta.activeMenu) {
|
|
||||||
// @ts-ignore
|
const menuData = computed(() => {
|
||||||
return meta.activeMenu;
|
return pureApp.layout === "mix" && device.value !== "mobile"
|
||||||
}
|
? subMenuData.value
|
||||||
return path;
|
: usePermissionStoreHook().wholeMenus;
|
||||||
});
|
});
|
||||||
|
|
||||||
const menuSelect = (indexPath: string): void => {
|
function getSubMenuData(path) {
|
||||||
let parentPath = "";
|
// path的上级路由组成的数组
|
||||||
let parentPathIndex = indexPath.lastIndexOf("/");
|
const parentPathArr = getParentPaths(
|
||||||
if (parentPathIndex > 0) {
|
path,
|
||||||
parentPath = indexPath.slice(0, parentPathIndex);
|
usePermissionStoreHook().wholeMenus
|
||||||
}
|
);
|
||||||
// 找到当前路由的信息
|
// 当前路由的父级路由信息
|
||||||
// eslint-disable-next-line no-inner-declarations
|
const parenetRoute = findRouteByPath(
|
||||||
function findCurrentRoute(routes) {
|
parentPathArr[0] || path,
|
||||||
return routes.map(item => {
|
usePermissionStoreHook().wholeMenus
|
||||||
if (item.path === indexPath) {
|
);
|
||||||
// 切换左侧菜单 通知标签页
|
if (!parenetRoute?.children) return;
|
||||||
emitter.emit("changLayoutRoute", {
|
subMenuData.value = parenetRoute?.children;
|
||||||
indexPath,
|
}
|
||||||
parentPath
|
getSubMenuData(route.path);
|
||||||
});
|
|
||||||
} else {
|
|
||||||
if (item.children) findCurrentRoute(item.children);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
findCurrentRoute(algorithm.increaseIndexes(router));
|
|
||||||
};
|
|
||||||
|
|
||||||
onBeforeMount(() => {
|
onBeforeMount(() => {
|
||||||
emitter.on("logoChange", key => {
|
emitter.on("logoChange", key => {
|
||||||
showLogo.value = key;
|
showLogo.value = key;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
getSubMenuData(route.path);
|
||||||
|
menuSelect(route.path, routers);
|
||||||
|
}
|
||||||
|
);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div :class="['sidebar-container', showLogo ? 'has-logo' : '']">
|
<div :class="['sidebar-container', showLogo ? 'has-logo' : '']">
|
||||||
<Logo v-if="showLogo" :collapse="isCollapse" />
|
<Logo v-if="showLogo" :collapse="isCollapse" />
|
||||||
<el-scrollbar wrap-class="scrollbar-wrapper">
|
<el-scrollbar
|
||||||
|
wrap-class="scrollbar-wrapper"
|
||||||
|
:class="[device === 'mobile' ? 'mobile' : 'pc']"
|
||||||
|
>
|
||||||
<el-menu
|
<el-menu
|
||||||
:default-active="activeMenu"
|
|
||||||
:collapse="isCollapse"
|
|
||||||
unique-opened
|
|
||||||
router
|
router
|
||||||
:collapse-transition="false"
|
unique-opened
|
||||||
mode="vertical"
|
mode="vertical"
|
||||||
class="outer-most"
|
class="outer-most select-none"
|
||||||
@select="menuSelect"
|
:collapse="isCollapse"
|
||||||
|
:default-active="route.path"
|
||||||
|
:collapse-transition="false"
|
||||||
|
@select="indexPath => menuSelect(indexPath, routers)"
|
||||||
>
|
>
|
||||||
<sidebar-item
|
<sidebar-item
|
||||||
v-for="route in usePermissionStoreHook().wholeMenus"
|
v-for="routes in menuData"
|
||||||
:key="route.path"
|
:key="routes.path"
|
||||||
:item="route"
|
:item="routes"
|
||||||
class="outer-most"
|
:base-path="routes.path"
|
||||||
:base-path="route.path"
|
class="outer-most select-none"
|
||||||
/>
|
/>
|
||||||
</el-menu>
|
</el-menu>
|
||||||
</el-scrollbar>
|
</el-scrollbar>
|
||||||
|
<leftCollapse
|
||||||
|
v-if="device !== 'mobile'"
|
||||||
|
:is-active="pureApp.sidebar.opened"
|
||||||
|
@toggleClick="toggleSideBar"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -18,36 +18,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@-webkit-keyframes rotate {
|
|
||||||
from {
|
|
||||||
-webkit-transform: rotate(0deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
to {
|
|
||||||
-webkit-transform: rotate(360deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@-moz-keyframes rotate {
|
|
||||||
from {
|
|
||||||
-moz-transform: rotate(0deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
to {
|
|
||||||
-moz-transform: rotate(360deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@-o-keyframes rotate {
|
|
||||||
from {
|
|
||||||
-o-transform: rotate(0deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
to {
|
|
||||||
-o-transform: rotate(360deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@keyframes rotate {
|
@keyframes rotate {
|
||||||
from {
|
from {
|
||||||
transform: rotate(0deg);
|
transform: rotate(0deg);
|
||||||
@@ -73,14 +43,14 @@
|
|||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
color: var(--el-text-color-regular);
|
color: var(--el-text-color-primary);
|
||||||
background: #fff;
|
background: #fff;
|
||||||
position: relative;
|
position: relative;
|
||||||
box-shadow: 0 0 1px #888;
|
box-shadow: 0 0 1px #888;
|
||||||
|
|
||||||
.scroll-item {
|
.scroll-item {
|
||||||
border-radius: 3px 3px 0 0;
|
border-radius: 3px 3px 0 0;
|
||||||
padding: 0 6px 0 6px;
|
padding: 0 6px;
|
||||||
box-shadow: 0 0 1px #888;
|
box-shadow: 0 0 1px #888;
|
||||||
position: relative;
|
position: relative;
|
||||||
margin-right: 4px;
|
margin-right: 4px;
|
||||||
@@ -92,7 +62,7 @@
|
|||||||
|
|
||||||
.el-icon-close {
|
.el-icon-close {
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
color: #1890ff;
|
color: var(--el-color-primary);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 50%;
|
top: 50%;
|
||||||
@@ -122,8 +92,8 @@
|
|||||||
|
|
||||||
a {
|
a {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: #666;
|
color: var(--el-text-color-primary);
|
||||||
padding: 0 4px 0 4px;
|
padding: 0 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scroll-container {
|
.scroll-container {
|
||||||
@@ -174,7 +144,7 @@
|
|||||||
list-style-type: none;
|
list-style-type: none;
|
||||||
padding: 5px 0;
|
padding: 5px 0;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
color: #000000d9;
|
color: var(--el-text-color-primary);
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
@@ -190,7 +160,8 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background: #eee;
|
// background: var(--el-color-primary-light-9);
|
||||||
|
color: var(--el-color-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
svg {
|
svg {
|
||||||
@@ -202,8 +173,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.el-dropdown-menu {
|
.el-dropdown-menu {
|
||||||
padding: 0;
|
|
||||||
|
|
||||||
li {
|
li {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
@@ -236,7 +205,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.scroll-item.is-active {
|
.scroll-item.is-active {
|
||||||
background-color: #eaf4fe;
|
// background-color: var(--el-color-primary-light-9);
|
||||||
position: relative;
|
position: relative;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
|
|
||||||
@@ -249,17 +218,27 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
color: #1890ff;
|
color: var(--el-color-primary) !important;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.ri-arrow-left-s-line {
|
.arrow-left,
|
||||||
|
.arrow-right {
|
||||||
width: 40px;
|
width: 40px;
|
||||||
height: 38px;
|
height: 38px;
|
||||||
line-height: 38px;
|
color: var(--el-text-color-primary);
|
||||||
text-align: center;
|
position: relative;
|
||||||
font-size: 20px;
|
|
||||||
color: #00000073;
|
svg {
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, 50%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrow-left {
|
||||||
box-shadow: 5px 0 5px -6px #ccc;
|
box-shadow: 5px 0 5px -6px #ccc;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
@@ -267,15 +246,9 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.ri-arrow-right-s-line {
|
.arrow-right {
|
||||||
width: 40px;
|
|
||||||
height: 38px;
|
|
||||||
line-height: 38px;
|
|
||||||
text-align: center;
|
|
||||||
font-size: 20px;
|
|
||||||
border-right: 1px solid #ccc;
|
|
||||||
color: #00000073;
|
|
||||||
box-shadow: -5px 0 5px -6px #ccc;
|
box-shadow: -5px 0 5px -6px #ccc;
|
||||||
|
border-right: 1px solid #ccc;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
cursor: e-resize;
|
cursor: e-resize;
|
||||||
@@ -284,10 +257,10 @@
|
|||||||
|
|
||||||
/* 卡片模式下鼠标移入显示蓝色边框 */
|
/* 卡片模式下鼠标移入显示蓝色边框 */
|
||||||
.card-in {
|
.card-in {
|
||||||
color: #1890ff;
|
color: var(--el-color-primary);
|
||||||
|
|
||||||
a {
|
a {
|
||||||
color: #1890ff;
|
color: var(--el-color-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -308,7 +281,7 @@
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
left: 0;
|
left: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
background: #1890ff;
|
background: var(--el-color-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 灵动模式下鼠标移入显示蓝色进度条 */
|
/* 灵动模式下鼠标移入显示蓝色进度条 */
|
||||||
@@ -318,7 +291,7 @@
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
left: 0;
|
left: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
background: #1890ff;
|
background: var(--el-color-primary);
|
||||||
animation: scheduleInWidth 400ms ease-in;
|
animation: scheduleInWidth 400ms ease-in;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -329,14 +302,11 @@
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
left: 0;
|
left: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
background: #1890ff;
|
background: var(--el-color-primary);
|
||||||
animation: scheduleOutWidth 400ms ease-in;
|
animation: scheduleOutWidth 400ms ease-in;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 刷新按钮动画效果 */
|
/* 刷新按钮动画效果 */
|
||||||
.refresh-button {
|
.refresh-button {
|
||||||
-webkit-animation: rotate 600ms linear infinite;
|
|
||||||
-moz-animation: rotate 600ms linear infinite;
|
|
||||||
-o-animation: rotate 600ms linear infinite;
|
|
||||||
animation: rotate 600ms linear infinite;
|
animation: rotate 600ms linear infinite;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,143 +1,71 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
|
||||||
ref,
|
|
||||||
watch,
|
|
||||||
unref,
|
|
||||||
reactive,
|
|
||||||
nextTick,
|
|
||||||
computed,
|
|
||||||
ComputedRef,
|
|
||||||
CSSProperties,
|
|
||||||
onBeforeMount,
|
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
|
||||||
|
|
||||||
import close from "/@/assets/svg/close.svg?component";
|
|
||||||
import refresh from "/@/assets/svg/refresh.svg?component";
|
|
||||||
import closeAll from "/@/assets/svg/close_all.svg?component";
|
|
||||||
import closeLeft from "/@/assets/svg/close_left.svg?component";
|
|
||||||
import closeOther from "/@/assets/svg/close_other.svg?component";
|
|
||||||
import closeRight from "/@/assets/svg/close_right.svg?component";
|
|
||||||
|
|
||||||
import { emitter } from "/@/utils/mitt";
|
import { emitter } from "/@/utils/mitt";
|
||||||
import { $t as t } from "/@/plugins/i18n";
|
import { RouteConfigs } from "../../types";
|
||||||
import { isEqual, isEmpty } from "lodash-es";
|
import { useTags } from "../../hooks/useTag";
|
||||||
import { transformI18n } from "/@/plugins/i18n";
|
import { routerArrays } from "/@/layout/types";
|
||||||
import { storageLocal } from "/@/utils/storage";
|
import { isEqual, isEmpty } from "lodash-unified";
|
||||||
import { useRoute, useRouter } from "vue-router";
|
import { toggleClass, removeClass } from "@pureadmin/utils";
|
||||||
import { RouteConfigs, tagsViewsType } from "../../types";
|
import { useResizeObserver, useDebounceFn } from "@vueuse/core";
|
||||||
import { useSettingStoreHook } from "/@/store/modules/settings";
|
import { useSettingStoreHook } from "/@/store/modules/settings";
|
||||||
import { handleAliveRoute, delAliveRoutes } from "/@/router/utils";
|
import { handleAliveRoute, delAliveRoutes } from "/@/router/utils";
|
||||||
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
||||||
import { usePermissionStoreHook } from "/@/store/modules/permission";
|
import { ref, watch, unref, toRaw, nextTick, onBeforeMount } from "vue";
|
||||||
import { toggleClass, removeClass, hasClass } from "/@/utils/operate";
|
|
||||||
import { templateRef, useResizeObserver, useDebounceFn } from "@vueuse/core";
|
|
||||||
|
|
||||||
const route = useRoute();
|
const {
|
||||||
const router = useRouter();
|
route,
|
||||||
const translateX = ref<number>(0);
|
router,
|
||||||
const activeIndex = ref<number>(-1);
|
visible,
|
||||||
let refreshButton = "refresh-button";
|
showTags,
|
||||||
const instance = getCurrentInstance();
|
instance,
|
||||||
const pureSetting = useSettingStoreHook();
|
multiTags,
|
||||||
const showTags = ref(storageLocal.getItem("tagsVal") || false);
|
tagsViews,
|
||||||
const tabDom = templateRef<HTMLElement | null>("tabDom", null);
|
buttonTop,
|
||||||
const containerDom = templateRef<HTMLElement | null>("containerDom", null);
|
buttonLeft,
|
||||||
const scrollbarDom = templateRef<HTMLElement | null>("scrollbarDom", null);
|
showModel,
|
||||||
|
translateX,
|
||||||
|
activeIndex,
|
||||||
|
getTabStyle,
|
||||||
|
iconIsActive,
|
||||||
|
linkIsActive,
|
||||||
|
currentSelect,
|
||||||
|
scheduleIsActive,
|
||||||
|
getContextMenuStyle,
|
||||||
|
closeMenu,
|
||||||
|
onMounted,
|
||||||
|
onMouseenter,
|
||||||
|
onMouseleave,
|
||||||
|
transformI18n
|
||||||
|
} = useTags();
|
||||||
|
|
||||||
let multiTags: ComputedRef<Array<RouteConfigs>> = computed(() => {
|
const tabDom = ref();
|
||||||
return useMultiTagsStoreHook()?.multiTags;
|
const containerDom = ref();
|
||||||
});
|
const scrollbarDom = ref();
|
||||||
|
|
||||||
const linkIsActive = computed(() => {
|
|
||||||
return item => {
|
|
||||||
if (Object.keys(route.query).length === 0) {
|
|
||||||
if (route.path === item.path) {
|
|
||||||
return "is-active";
|
|
||||||
} else {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isEqual(route?.query, item?.query)) {
|
|
||||||
return "is-active";
|
|
||||||
} else {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const scheduleIsActive = computed(() => {
|
|
||||||
return item => {
|
|
||||||
if (Object.keys(route.query).length === 0) {
|
|
||||||
if (route.path === item.path) {
|
|
||||||
return "schedule-active";
|
|
||||||
} else {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isEqual(route?.query, item?.query)) {
|
|
||||||
return "schedule-active";
|
|
||||||
} else {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const iconIsActive = computed(() => {
|
|
||||||
return (item, index) => {
|
|
||||||
if (index === 0) return;
|
|
||||||
if (Object.keys(route.query).length === 0) {
|
|
||||||
if (route.path === item.path) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isEqual(route?.query, item?.query)) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const dynamicTagView = () => {
|
const dynamicTagView = () => {
|
||||||
const index = multiTags.value.findIndex(item => {
|
const index = multiTags.value.findIndex(item => {
|
||||||
return item.path === route.path;
|
if (item.query) {
|
||||||
|
return isEqual(route.query, item.query);
|
||||||
|
} else if (item.params) {
|
||||||
|
return isEqual(route.params, item.params);
|
||||||
|
} else {
|
||||||
|
return item.path === route.path;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
moveToView(index);
|
moveToView(index);
|
||||||
};
|
};
|
||||||
|
|
||||||
watch([route], () => {
|
|
||||||
activeIndex.value = -1;
|
|
||||||
dynamicTagView();
|
|
||||||
});
|
|
||||||
|
|
||||||
useResizeObserver(
|
|
||||||
scrollbarDom,
|
|
||||||
useDebounceFn(() => {
|
|
||||||
dynamicTagView();
|
|
||||||
}, 200)
|
|
||||||
);
|
|
||||||
|
|
||||||
const tabNavPadding = 10;
|
|
||||||
const moveToView = (index: number): void => {
|
const moveToView = (index: number): void => {
|
||||||
if (!instance.refs["dynamic" + index]) {
|
const tabNavPadding = 10;
|
||||||
return;
|
if (!instance.refs["dynamic" + index]) return;
|
||||||
}
|
|
||||||
const tabItemEl = instance.refs["dynamic" + index][0];
|
const tabItemEl = instance.refs["dynamic" + index][0];
|
||||||
const tabItemElOffsetLeft = (tabItemEl as HTMLElement).offsetLeft;
|
const tabItemElOffsetLeft = (tabItemEl as HTMLElement)?.offsetLeft;
|
||||||
const tabItemOffsetWidth = (tabItemEl as HTMLElement).offsetWidth;
|
const tabItemOffsetWidth = (tabItemEl as HTMLElement)?.offsetWidth;
|
||||||
// 标签页导航栏可视长度(不包含溢出部分)
|
// 标签页导航栏可视长度(不包含溢出部分)
|
||||||
const scrollbarDomWidth = scrollbarDom.value
|
const scrollbarDomWidth = scrollbarDom.value
|
||||||
? scrollbarDom.value.offsetWidth
|
? scrollbarDom.value?.offsetWidth
|
||||||
: 0;
|
: 0;
|
||||||
// 已有标签页总长度(包含溢出部分)
|
// 已有标签页总长度(包含溢出部分)
|
||||||
const tabDomWidth = tabDom.value ? tabDom.value.offsetWidth : 0;
|
const tabDomWidth = tabDom.value ? tabDom.value?.offsetWidth : 0;
|
||||||
|
|
||||||
if (tabDomWidth < scrollbarDomWidth || tabItemElOffsetLeft === 0) {
|
if (tabDomWidth < scrollbarDomWidth || tabItemElOffsetLeft === 0) {
|
||||||
translateX.value = 0;
|
translateX.value = 0;
|
||||||
} else if (tabItemElOffsetLeft < -translateX.value) {
|
} else if (tabItemElOffsetLeft < -translateX.value) {
|
||||||
@@ -186,68 +114,6 @@ const handleScroll = (offset: number): void => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const tagsViews = reactive<Array<tagsViewsType>>([
|
|
||||||
{
|
|
||||||
icon: refresh,
|
|
||||||
text: t("buttons.hsreload"),
|
|
||||||
divided: false,
|
|
||||||
disabled: false,
|
|
||||||
show: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: close,
|
|
||||||
text: t("buttons.hscloseCurrentTab"),
|
|
||||||
divided: false,
|
|
||||||
disabled: multiTags.value.length > 1 ? false : true,
|
|
||||||
show: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: closeLeft,
|
|
||||||
text: t("buttons.hscloseLeftTabs"),
|
|
||||||
divided: true,
|
|
||||||
disabled: multiTags.value.length > 1 ? false : true,
|
|
||||||
show: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: closeRight,
|
|
||||||
text: t("buttons.hscloseRightTabs"),
|
|
||||||
divided: false,
|
|
||||||
disabled: multiTags.value.length > 1 ? false : true,
|
|
||||||
show: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: closeOther,
|
|
||||||
text: t("buttons.hscloseOtherTabs"),
|
|
||||||
divided: true,
|
|
||||||
disabled: multiTags.value.length > 2 ? false : true,
|
|
||||||
show: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: closeAll,
|
|
||||||
text: t("buttons.hscloseAllTabs"),
|
|
||||||
divided: false,
|
|
||||||
disabled: multiTags.value.length > 1 ? false : true,
|
|
||||||
show: true
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
// 显示模式,默认灵动模式显示
|
|
||||||
const showModel = ref(
|
|
||||||
storageLocal.getItem("responsive-configure")?.showModel || "smart"
|
|
||||||
);
|
|
||||||
if (!showModel.value) {
|
|
||||||
const configure = storageLocal.getItem("responsive-configure");
|
|
||||||
configure.showModel = "card";
|
|
||||||
storageLocal.setItem("responsive-configure", configure);
|
|
||||||
}
|
|
||||||
|
|
||||||
let visible = ref(false);
|
|
||||||
let buttonLeft = ref(0);
|
|
||||||
let buttonTop = ref(0);
|
|
||||||
|
|
||||||
// 当前右键选中的路由信息
|
|
||||||
let currentSelect = ref({});
|
|
||||||
|
|
||||||
function dynamicRouteTag(value: string, parentPath: string): void {
|
function dynamicRouteTag(value: string, parentPath: string): void {
|
||||||
const hasValue = multiTags.value.some(item => {
|
const hasValue = multiTags.value.some(item => {
|
||||||
return item.path === value;
|
return item.path === value;
|
||||||
@@ -272,11 +138,12 @@ function dynamicRouteTag(value: string, parentPath: string): void {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
concatPath(router.options.routes, value, parentPath);
|
concatPath(router.options.routes as any, value, parentPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 重新加载
|
/** 刷新路由 */
|
||||||
function onFresh() {
|
function onFresh() {
|
||||||
|
const refreshButton = "refresh-button";
|
||||||
toggleClass(true, refreshButton, document.querySelector(".rotate"));
|
toggleClass(true, refreshButton, document.querySelector(".rotate"));
|
||||||
const { fullPath, query } = unref(route);
|
const { fullPath, query } = unref(route);
|
||||||
router.replace({
|
router.replace({
|
||||||
@@ -296,6 +163,10 @@ function deleteDynamicTag(obj: any, current: any, tag?: string) {
|
|||||||
if (item.path === obj.path) {
|
if (item.path === obj.path) {
|
||||||
return item.query === obj.query;
|
return item.query === obj.query;
|
||||||
}
|
}
|
||||||
|
} else if (item.params) {
|
||||||
|
if (item.path === obj.path) {
|
||||||
|
return item.params === obj.params;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
return item.path === obj.path;
|
return item.path === obj.path;
|
||||||
}
|
}
|
||||||
@@ -307,25 +178,12 @@ function deleteDynamicTag(obj: any, current: any, tag?: string) {
|
|||||||
other?: boolean
|
other?: boolean
|
||||||
): void => {
|
): void => {
|
||||||
if (other) {
|
if (other) {
|
||||||
useMultiTagsStoreHook().handleTags("equal", [
|
useMultiTagsStoreHook().handleTags("equal", [routerArrays[0], obj]);
|
||||||
{
|
|
||||||
path: "/welcome",
|
|
||||||
parentPath: "/",
|
|
||||||
meta: {
|
|
||||||
title: "menus.hshome",
|
|
||||||
i18n: true,
|
|
||||||
icon: "el-icon-s-home",
|
|
||||||
showLink: true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
obj
|
|
||||||
]);
|
|
||||||
} else {
|
} else {
|
||||||
// @ts-ignore
|
|
||||||
delAliveRouteList = useMultiTagsStoreHook().handleTags("splice", "", {
|
delAliveRouteList = useMultiTagsStoreHook().handleTags("splice", "", {
|
||||||
startIndex,
|
startIndex,
|
||||||
length
|
length
|
||||||
});
|
}) as any;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -347,24 +205,25 @@ function deleteDynamicTag(obj: any, current: any, tag?: string) {
|
|||||||
: handleAliveRoute(route.matched, "delete");
|
: handleAliveRoute(route.matched, "delete");
|
||||||
// 如果删除当前激活tag就自动切换到最后一个tag
|
// 如果删除当前激活tag就自动切换到最后一个tag
|
||||||
if (tag === "left") return;
|
if (tag === "left") return;
|
||||||
nextTick(() => {
|
if (newRoute[0]?.query) {
|
||||||
router.push({
|
router.push({ name: newRoute[0].name, query: newRoute[0].query });
|
||||||
path: newRoute[0].path,
|
} else if (newRoute[0]?.params) {
|
||||||
query: newRoute[0].query
|
router.push({ name: newRoute[0].name, params: newRoute[0].params });
|
||||||
});
|
} else {
|
||||||
});
|
router.push({ path: newRoute[0].path });
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// 删除缓存路由
|
// 删除缓存路由
|
||||||
tag ? delAliveRoutes(delAliveRouteList) : delAliveRoutes([obj]);
|
tag ? delAliveRoutes(delAliveRouteList) : delAliveRoutes([obj]);
|
||||||
if (!multiTags.value.length) return;
|
if (!multiTags.value.length) return;
|
||||||
let isHasActiveTag = multiTags.value.some(item => {
|
if (multiTags.value.some(item => item.path === route.path)) return;
|
||||||
return item.path === route.path;
|
if (newRoute[0]?.query) {
|
||||||
});
|
router.push({ name: newRoute[0].name, query: newRoute[0].query });
|
||||||
!isHasActiveTag &&
|
} else if (newRoute[0]?.params) {
|
||||||
router.push({
|
router.push({ name: newRoute[0].name, params: newRoute[0].params });
|
||||||
path: newRoute[0].path,
|
} else {
|
||||||
query: newRoute[0].query
|
router.push({ path: newRoute[0].path });
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -381,7 +240,8 @@ function onClickDrop(key, item, selectRoute?: RouteConfigs) {
|
|||||||
path: selectRoute.path,
|
path: selectRoute.path,
|
||||||
meta: selectRoute.meta,
|
meta: selectRoute.meta,
|
||||||
name: selectRoute.name,
|
name: selectRoute.name,
|
||||||
query: selectRoute.query
|
query: selectRoute?.query,
|
||||||
|
params: selectRoute?.params
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
selectTagRoute = { path: route.path, meta: route.meta };
|
selectTagRoute = { path: route.path, meta: route.meta };
|
||||||
@@ -390,7 +250,7 @@ function onClickDrop(key, item, selectRoute?: RouteConfigs) {
|
|||||||
// 当前路由信息
|
// 当前路由信息
|
||||||
switch (key) {
|
switch (key) {
|
||||||
case 0:
|
case 0:
|
||||||
// 重新加载
|
// 刷新路由
|
||||||
onFresh();
|
onFresh();
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
@@ -415,7 +275,6 @@ function onClickDrop(key, item, selectRoute?: RouteConfigs) {
|
|||||||
startIndex: 1,
|
startIndex: 1,
|
||||||
length: multiTags.value.length
|
length: multiTags.value.length
|
||||||
});
|
});
|
||||||
usePermissionStoreHook().clearAllCachePage();
|
|
||||||
router.push("/welcome");
|
router.push("/welcome");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -424,13 +283,14 @@ function onClickDrop(key, item, selectRoute?: RouteConfigs) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 触发右键中菜单的点击事件
|
function handleCommand(command: any) {
|
||||||
function selectTag(key, item) {
|
const { key, item } = command;
|
||||||
onClickDrop(key, item, currentSelect.value);
|
onClickDrop(key, item);
|
||||||
}
|
}
|
||||||
|
|
||||||
function closeMenu() {
|
/** 触发右键中菜单的点击事件 */
|
||||||
visible.value = false;
|
function selectTag(key, item) {
|
||||||
|
onClickDrop(key, item, currentSelect.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
function showMenus(value: boolean) {
|
function showMenus(value: boolean) {
|
||||||
@@ -445,7 +305,7 @@ function disabledMenus(value: boolean) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 检查当前右键的菜单两边是否存在别的菜单,如果左侧的菜单是首页,则不显示关闭左侧标签页,如果右侧没有菜单,则不显示关闭右侧标签页
|
/** 检查当前右键的菜单两边是否存在别的菜单,如果左侧的菜单是首页,则不显示关闭左侧标签页,如果右侧没有菜单,则不显示关闭右侧标签页 */
|
||||||
function showMenuModel(
|
function showMenuModel(
|
||||||
currentPath: string,
|
currentPath: string,
|
||||||
query: object = {},
|
query: object = {},
|
||||||
@@ -470,7 +330,6 @@ function showMenuModel(
|
|||||||
* currentIndex为1时,左侧的菜单是首页,则不显示关闭左侧标签页
|
* currentIndex为1时,左侧的菜单是首页,则不显示关闭左侧标签页
|
||||||
* 如果currentIndex等于routeLength-1,右侧没有菜单,则不显示关闭右侧标签页
|
* 如果currentIndex等于routeLength-1,右侧没有菜单,则不显示关闭右侧标签页
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (currentIndex === 1 && routeLength !== 2) {
|
if (currentIndex === 1 && routeLength !== 2) {
|
||||||
// 左侧的菜单是首页,右侧存在别的菜单
|
// 左侧的菜单是首页,右侧存在别的菜单
|
||||||
tagsViews[2].show = false;
|
tagsViews[2].show = false;
|
||||||
@@ -506,7 +365,7 @@ function openMenu(tag, e) {
|
|||||||
// 右键菜单为首页,只显示刷新
|
// 右键菜单为首页,只显示刷新
|
||||||
showMenus(false);
|
showMenus(false);
|
||||||
tagsViews[0].show = true;
|
tagsViews[0].show = true;
|
||||||
} else if (route.path !== tag.path) {
|
} else if (route.path !== tag.path && route.name !== tag.name) {
|
||||||
// 右键菜单不匹配当前路由,隐藏刷新
|
// 右键菜单不匹配当前路由,隐藏刷新
|
||||||
tagsViews[0].show = false;
|
tagsViews[0].show = false;
|
||||||
showMenuModel(tag.path, tag.query);
|
showMenuModel(tag.path, tag.query);
|
||||||
@@ -534,63 +393,36 @@ function openMenu(tag, e) {
|
|||||||
} else {
|
} else {
|
||||||
buttonLeft.value = left;
|
buttonLeft.value = left;
|
||||||
}
|
}
|
||||||
pureSetting.hiddenSideBar
|
useSettingStoreHook().hiddenSideBar
|
||||||
? (buttonTop.value = e.clientY)
|
? (buttonTop.value = e.clientY)
|
||||||
: (buttonTop.value = e.clientY - 40);
|
: (buttonTop.value = e.clientY - 40);
|
||||||
setTimeout(() => {
|
nextTick(() => {
|
||||||
visible.value = true;
|
visible.value = true;
|
||||||
}, 10);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 触发tags标签切换
|
|
||||||
function tagOnClick(item) {
|
|
||||||
router.push({
|
|
||||||
path: item?.path,
|
|
||||||
query: item?.query
|
|
||||||
});
|
});
|
||||||
showMenuModel(item?.path, item?.query);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 鼠标移入
|
/** 触发tags标签切换 */
|
||||||
function onMouseenter(index) {
|
function tagOnClick(item) {
|
||||||
if (index) activeIndex.value = index;
|
const { name, path } = item;
|
||||||
if (unref(showModel) === "smart") {
|
if (name) {
|
||||||
if (hasClass(instance.refs["schedule" + index][0], "schedule-active"))
|
if (item.query) {
|
||||||
return;
|
router.push({
|
||||||
toggleClass(true, "schedule-in", instance.refs["schedule" + index][0]);
|
name,
|
||||||
toggleClass(false, "schedule-out", instance.refs["schedule" + index][0]);
|
query: item.query
|
||||||
} else {
|
});
|
||||||
if (hasClass(instance.refs["dynamic" + index][0], "card-active")) return;
|
} else if (item.params) {
|
||||||
toggleClass(true, "card-in", instance.refs["dynamic" + index][0]);
|
router.push({
|
||||||
toggleClass(false, "card-out", instance.refs["dynamic" + index][0]);
|
name,
|
||||||
}
|
params: item.params
|
||||||
}
|
});
|
||||||
|
|
||||||
// 鼠标移出
|
|
||||||
function onMouseleave(index) {
|
|
||||||
activeIndex.value = -1;
|
|
||||||
if (unref(showModel) === "smart") {
|
|
||||||
if (hasClass(instance.refs["schedule" + index][0], "schedule-active"))
|
|
||||||
return;
|
|
||||||
toggleClass(false, "schedule-in", instance.refs["schedule" + index][0]);
|
|
||||||
toggleClass(true, "schedule-out", instance.refs["schedule" + index][0]);
|
|
||||||
} else {
|
|
||||||
if (hasClass(instance.refs["dynamic" + index][0], "card-active")) return;
|
|
||||||
toggleClass(false, "card-in", instance.refs["dynamic" + index][0]);
|
|
||||||
toggleClass(true, "card-out", instance.refs["dynamic" + index][0]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
watch(
|
|
||||||
() => visible.value,
|
|
||||||
val => {
|
|
||||||
if (val) {
|
|
||||||
document.body.addEventListener("click", closeMenu);
|
|
||||||
} else {
|
} else {
|
||||||
document.body.removeEventListener("click", closeMenu);
|
router.push({ name });
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
router.push({ path });
|
||||||
}
|
}
|
||||||
);
|
// showMenuModel(item?.path, item?.query);
|
||||||
|
}
|
||||||
|
|
||||||
onBeforeMount(() => {
|
onBeforeMount(() => {
|
||||||
if (!instance) return;
|
if (!instance) return;
|
||||||
@@ -599,9 +431,9 @@ onBeforeMount(() => {
|
|||||||
showMenuModel(route.fullPath);
|
showMenuModel(route.fullPath);
|
||||||
|
|
||||||
// 触发隐藏标签页
|
// 触发隐藏标签页
|
||||||
emitter.on("tagViewsChange", key => {
|
emitter.on("tagViewsChange", (key: any) => {
|
||||||
if (unref(showTags) === key) return;
|
if (unref(showTags as any) === key) return;
|
||||||
showTags.value = key;
|
(showTags as any).value = key;
|
||||||
});
|
});
|
||||||
|
|
||||||
// 改变标签风格
|
// 改变标签风格
|
||||||
@@ -618,20 +450,26 @@ onBeforeMount(() => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const getTabStyle = computed((): CSSProperties => {
|
watch([route], () => {
|
||||||
return {
|
activeIndex.value = -1;
|
||||||
transform: `translateX(${translateX.value}px)`
|
dynamicTagView();
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const getContextMenuStyle = computed((): CSSProperties => {
|
onMounted(() => {
|
||||||
return { left: buttonLeft.value + "px", top: buttonTop.value + "px" };
|
useResizeObserver(
|
||||||
|
scrollbarDom,
|
||||||
|
useDebounceFn(() => {
|
||||||
|
dynamicTagView();
|
||||||
|
}, 200)
|
||||||
|
);
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div ref="containerDom" class="tags-view" v-if="!showTags">
|
<div ref="containerDom" class="tags-view" v-if="!showTags">
|
||||||
<i class="ri-arrow-left-s-line" @click="handleScroll(200)"></i>
|
<div class="arrow-left">
|
||||||
|
<IconifyIconOffline icon="arrow-left-s-line" @click="handleScroll(200)" />
|
||||||
|
</div>
|
||||||
<div ref="scrollbarDom" class="scroll-container">
|
<div ref="scrollbarDom" class="scroll-container">
|
||||||
<div class="tab" ref="tabDom" :style="getTabStyle">
|
<div class="tab" ref="tabDom" :style="getTabStyle">
|
||||||
<div
|
<div
|
||||||
@@ -650,10 +488,13 @@ const getContextMenuStyle = computed((): CSSProperties => {
|
|||||||
@mouseleave.prevent="onMouseleave(index)"
|
@mouseleave.prevent="onMouseleave(index)"
|
||||||
@click="tagOnClick(item)"
|
@click="tagOnClick(item)"
|
||||||
>
|
>
|
||||||
<router-link :to="item.path"
|
<router-link
|
||||||
>{{ transformI18n(item.meta.title, item.meta.i18n) }}
|
:to="item.path"
|
||||||
|
class="dark:!text-text_color_primary dark:hover:!text-primary"
|
||||||
|
>
|
||||||
|
{{ transformI18n(item.meta.title) }}
|
||||||
</router-link>
|
</router-link>
|
||||||
<el-icon
|
<span
|
||||||
v-if="
|
v-if="
|
||||||
iconIsActive(item, index) ||
|
iconIsActive(item, index) ||
|
||||||
(index === activeIndex && index !== 0)
|
(index === activeIndex && index !== 0)
|
||||||
@@ -662,16 +503,21 @@ const getContextMenuStyle = computed((): CSSProperties => {
|
|||||||
@click.stop="deleteMenu(item)"
|
@click.stop="deleteMenu(item)"
|
||||||
>
|
>
|
||||||
<IconifyIconOffline icon="close-bold" />
|
<IconifyIconOffline icon="close-bold" />
|
||||||
</el-icon>
|
</span>
|
||||||
<div
|
<div
|
||||||
:ref="'schedule' + index"
|
:ref="'schedule' + index"
|
||||||
v-if="showModel !== 'card'"
|
v-if="showModel !== 'card'"
|
||||||
:class="[scheduleIsActive(item)]"
|
:class="[scheduleIsActive(item)]"
|
||||||
></div>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<i class="ri-arrow-right-s-line" @click="handleScroll(-200)"></i>
|
<span class="arrow-right">
|
||||||
|
<IconifyIconOffline
|
||||||
|
icon="arrow-right-s-line"
|
||||||
|
@click="handleScroll(-200)"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
<!-- 右键菜单按钮 -->
|
<!-- 右键菜单按钮 -->
|
||||||
<transition name="el-zoom-in-top">
|
<transition name="el-zoom-in-top">
|
||||||
<ul
|
<ul
|
||||||
@@ -686,8 +532,8 @@ const getContextMenuStyle = computed((): CSSProperties => {
|
|||||||
style="display: flex; align-items: center"
|
style="display: flex; align-items: center"
|
||||||
>
|
>
|
||||||
<li v-if="item.show" @click="selectTag(key, item)">
|
<li v-if="item.show" @click="selectTag(key, item)">
|
||||||
<component :is="item.icon" :key="key" />
|
<component :is="toRaw(item.icon)" :key="key" />
|
||||||
{{ $t(item.text) }}
|
{{ transformI18n(item.text) }}
|
||||||
</li>
|
</li>
|
||||||
</div>
|
</div>
|
||||||
</ul>
|
</ul>
|
||||||
@@ -695,37 +541,43 @@ const getContextMenuStyle = computed((): CSSProperties => {
|
|||||||
<!-- 右侧功能按钮 -->
|
<!-- 右侧功能按钮 -->
|
||||||
<ul class="right-button">
|
<ul class="right-button">
|
||||||
<li>
|
<li>
|
||||||
<el-icon
|
<span
|
||||||
:title="$t('buttons.hsrefreshRoute')"
|
:title="transformI18n('buttons.hsrefreshRoute')"
|
||||||
class="el-icon-refresh-right rotate"
|
class="el-icon-refresh-right rotate"
|
||||||
@click="onFresh"
|
@click="onFresh"
|
||||||
>
|
>
|
||||||
<IconifyIconOffline icon="refresh-right" />
|
<IconifyIconOffline icon="refresh-right" />
|
||||||
</el-icon>
|
</span>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<el-dropdown trigger="click" placement="bottom-end">
|
<el-dropdown
|
||||||
<el-icon>
|
trigger="click"
|
||||||
<IconifyIconOffline icon="arrow-down" />
|
placement="bottom-end"
|
||||||
</el-icon>
|
@command="handleCommand"
|
||||||
|
>
|
||||||
|
<IconifyIconOffline icon="arrow-down" class="dark:text-white" />
|
||||||
<template #dropdown>
|
<template #dropdown>
|
||||||
<el-dropdown-menu>
|
<el-dropdown-menu>
|
||||||
<el-dropdown-item
|
<el-dropdown-item
|
||||||
v-for="(item, key) in tagsViews"
|
v-for="(item, key) in tagsViews"
|
||||||
:key="key"
|
:key="key"
|
||||||
|
:command="{ key, item }"
|
||||||
:divided="item.divided"
|
:divided="item.divided"
|
||||||
:disabled="item.disabled"
|
:disabled="item.disabled"
|
||||||
@click="onClickDrop(key, item)"
|
|
||||||
>
|
>
|
||||||
<component :is="item.icon" :key="key" />
|
<component
|
||||||
{{ $t(item.text) }}
|
:is="toRaw(item.icon)"
|
||||||
|
:key="key"
|
||||||
|
style="margin-right: 6px"
|
||||||
|
/>
|
||||||
|
{{ transformI18n(item.text) }}
|
||||||
</el-dropdown-item>
|
</el-dropdown-item>
|
||||||
</el-dropdown-menu>
|
</el-dropdown-menu>
|
||||||
</template>
|
</template>
|
||||||
</el-dropdown>
|
</el-dropdown>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<slot></slot>
|
<slot />
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
74
src/layout/frameView.vue
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useRoute } from "vue-router";
|
||||||
|
import { ref, unref, onMounted, nextTick } from "vue";
|
||||||
|
|
||||||
|
defineOptions({
|
||||||
|
name: "FrameView"
|
||||||
|
});
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const loading = ref(true);
|
||||||
|
const currentRoute = useRoute();
|
||||||
|
const frameSrc = ref<string>("");
|
||||||
|
const frameRef = ref<HTMLElement | null>(null);
|
||||||
|
|
||||||
|
if (unref(currentRoute.meta)?.frameSrc) {
|
||||||
|
frameSrc.value = unref(currentRoute.meta)?.frameSrc as string;
|
||||||
|
}
|
||||||
|
unref(currentRoute.meta)?.frameLoading === false && hideLoading();
|
||||||
|
|
||||||
|
function hideLoading() {
|
||||||
|
loading.value = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function init() {
|
||||||
|
nextTick(() => {
|
||||||
|
const iframe = unref(frameRef);
|
||||||
|
if (!iframe) return;
|
||||||
|
const _frame = iframe as any;
|
||||||
|
if (_frame.attachEvent) {
|
||||||
|
_frame.attachEvent("onload", () => {
|
||||||
|
hideLoading();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
iframe.onload = () => {
|
||||||
|
hideLoading();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
init();
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="frame"
|
||||||
|
v-loading="loading"
|
||||||
|
:element-loading-text="t('status.hsLoad')"
|
||||||
|
>
|
||||||
|
<iframe :src="frameSrc" class="frame-iframe" ref="frameRef" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.frame {
|
||||||
|
height: calc(100vh - 88px);
|
||||||
|
z-index: 998;
|
||||||
|
|
||||||
|
.frame-iframe {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.main-content {
|
||||||
|
margin: 2px 0 0 !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
26
src/layout/hooks/useBoolean.ts
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
import { ref } from "vue";
|
||||||
|
|
||||||
|
export function useBoolean(initValue = false) {
|
||||||
|
const bool = ref(initValue);
|
||||||
|
|
||||||
|
function setBool(value: boolean) {
|
||||||
|
bool.value = value;
|
||||||
|
}
|
||||||
|
function setTrue() {
|
||||||
|
setBool(true);
|
||||||
|
}
|
||||||
|
function setFalse() {
|
||||||
|
setBool(false);
|
||||||
|
}
|
||||||
|
function toggle() {
|
||||||
|
setBool(!bool.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
bool,
|
||||||
|
setBool,
|
||||||
|
setTrue,
|
||||||
|
setFalse,
|
||||||
|
toggle
|
||||||
|
};
|
||||||
|
}
|
||||||
116
src/layout/hooks/useDataThemeChange.ts
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
import { ref } from "vue";
|
||||||
|
import { getConfig } from "/@/config";
|
||||||
|
import { find } from "lodash-unified";
|
||||||
|
import { useLayout } from "./useLayout";
|
||||||
|
import { themeColorsType } from "../types";
|
||||||
|
import { TinyColor } from "@ctrl/tinycolor";
|
||||||
|
import { useGlobal } from "@pureadmin/utils";
|
||||||
|
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
||||||
|
import {
|
||||||
|
darken,
|
||||||
|
lighten,
|
||||||
|
toggleTheme
|
||||||
|
} from "@pureadmin/theme/dist/browser-utils";
|
||||||
|
|
||||||
|
export function useDataThemeChange() {
|
||||||
|
const { layoutTheme, layout } = useLayout();
|
||||||
|
const themeColors = ref<Array<themeColorsType>>([
|
||||||
|
/* 道奇蓝(默认) */
|
||||||
|
{ color: "#1b2a47", themeColor: "default" },
|
||||||
|
/* 亮白色 */
|
||||||
|
{ color: "#ffffff", themeColor: "light" },
|
||||||
|
/* 猩红色 */
|
||||||
|
{ color: "#f5222d", themeColor: "dusk" },
|
||||||
|
/* 橙红色 */
|
||||||
|
{ color: "#fa541c", themeColor: "volcano" },
|
||||||
|
/* 金色 */
|
||||||
|
{ color: "#fadb14", themeColor: "yellow" },
|
||||||
|
/* 绿宝石 */
|
||||||
|
{ color: "#13c2c2", themeColor: "mingQing" },
|
||||||
|
/* 酸橙绿 */
|
||||||
|
{ color: "#52c41a", themeColor: "auroraGreen" },
|
||||||
|
/* 深粉色 */
|
||||||
|
{ color: "#eb2f96", themeColor: "pink" },
|
||||||
|
/* 深紫罗兰色 */
|
||||||
|
{ color: "#722ed1", themeColor: "saucePurple" }
|
||||||
|
]);
|
||||||
|
|
||||||
|
const { $storage } = useGlobal<GlobalPropertiesApi>();
|
||||||
|
const dataTheme = ref<boolean>($storage?.layout?.darkMode);
|
||||||
|
const body = document.documentElement as HTMLElement;
|
||||||
|
|
||||||
|
/** 设置导航主题色 */
|
||||||
|
function setLayoutThemeColor(theme = "default") {
|
||||||
|
layoutTheme.value.theme = theme;
|
||||||
|
toggleTheme({
|
||||||
|
scopeName: `layout-theme-${theme}`
|
||||||
|
});
|
||||||
|
$storage.layout = {
|
||||||
|
layout: layout.value,
|
||||||
|
theme,
|
||||||
|
darkMode: dataTheme.value,
|
||||||
|
sidebarStatus: $storage.layout?.sidebarStatus,
|
||||||
|
epThemeColor: $storage.layout?.epThemeColor
|
||||||
|
};
|
||||||
|
|
||||||
|
if (theme === "default" || theme === "light") {
|
||||||
|
setEpThemeColor(getConfig().EpThemeColor);
|
||||||
|
} else {
|
||||||
|
const colors = find(themeColors.value, { themeColor: theme });
|
||||||
|
setEpThemeColor(colors.color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @description 自动计算hover和active颜色
|
||||||
|
* @see {@link https://element-plus.org/zh-CN/component/button.html#%E8%87%AA%E5%AE%9A%E4%B9%89%E9%A2%9C%E8%89%B2}
|
||||||
|
*/
|
||||||
|
const shadeBgColor = (color: string): string => {
|
||||||
|
return new TinyColor(color).shade(10).toString();
|
||||||
|
};
|
||||||
|
|
||||||
|
/** 设置ep主题色 */
|
||||||
|
const setEpThemeColor = (color: string) => {
|
||||||
|
useEpThemeStoreHook().setEpThemeColor(color);
|
||||||
|
body.style.setProperty("--el-color-primary-active", shadeBgColor(color));
|
||||||
|
document.documentElement.style.setProperty("--el-color-primary", color);
|
||||||
|
for (let i = 1; i <= 9; i++) {
|
||||||
|
document.documentElement.style.setProperty(
|
||||||
|
`--el-color-primary-light-${i}`,
|
||||||
|
lighten(color, i / 10)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
for (let i = 1; i <= 2; i++) {
|
||||||
|
document.documentElement.style.setProperty(
|
||||||
|
`--el-color-primary-dark-${i}`,
|
||||||
|
darken(color, i / 10)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/** 日间、夜间主题切换 */
|
||||||
|
function dataThemeChange() {
|
||||||
|
/* 如果当前是light夜间主题,默认切换到default主题 */
|
||||||
|
if (useEpThemeStoreHook().epTheme === "light" && dataTheme.value) {
|
||||||
|
setLayoutThemeColor("default");
|
||||||
|
} else {
|
||||||
|
setLayoutThemeColor(useEpThemeStoreHook().epTheme);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dataTheme.value) {
|
||||||
|
document.documentElement.classList.add("dark");
|
||||||
|
} else {
|
||||||
|
document.documentElement.classList.remove("dark");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
body,
|
||||||
|
dataTheme,
|
||||||
|
layoutTheme,
|
||||||
|
themeColors,
|
||||||
|
dataThemeChange,
|
||||||
|
setEpThemeColor,
|
||||||
|
setLayoutThemeColor
|
||||||
|
};
|
||||||
|
}
|
||||||
60
src/layout/hooks/useLayout.ts
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import { computed } from "vue";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { routerArrays } from "../types";
|
||||||
|
import { useGlobal } from "@pureadmin/utils";
|
||||||
|
import { useMultiTagsStore } from "/@/store/modules/multiTags";
|
||||||
|
|
||||||
|
export function useLayout() {
|
||||||
|
const { $storage, $config } = useGlobal<GlobalPropertiesApi>();
|
||||||
|
|
||||||
|
const initStorage = () => {
|
||||||
|
/** 路由 */
|
||||||
|
if (
|
||||||
|
useMultiTagsStore().multiTagsCache &&
|
||||||
|
(!$storage.tags || $storage.tags.length === 0)
|
||||||
|
) {
|
||||||
|
$storage.tags = routerArrays;
|
||||||
|
}
|
||||||
|
/** 国际化 */
|
||||||
|
if (!$storage.locale) {
|
||||||
|
$storage.locale = { locale: $config?.Locale ?? "zh" };
|
||||||
|
useI18n().locale.value = $config?.Locale ?? "zh";
|
||||||
|
}
|
||||||
|
/** 导航 */
|
||||||
|
if (!$storage.layout) {
|
||||||
|
$storage.layout = {
|
||||||
|
layout: $config?.Layout ?? "vertical",
|
||||||
|
theme: $config?.Theme ?? "default",
|
||||||
|
darkMode: $config?.DarkMode ?? false,
|
||||||
|
sidebarStatus: $config?.SidebarStatus ?? true,
|
||||||
|
epThemeColor: $config?.EpThemeColor ?? "#409EFF"
|
||||||
|
};
|
||||||
|
}
|
||||||
|
/** 灰色模式、色弱模式、隐藏标签页 */
|
||||||
|
if (!$storage.configure) {
|
||||||
|
$storage.configure = {
|
||||||
|
grey: $config?.Grey ?? false,
|
||||||
|
weak: $config?.Weak ?? false,
|
||||||
|
hideTabs: $config?.HideTabs ?? false,
|
||||||
|
showLogo: $config?.ShowLogo ?? true,
|
||||||
|
showModel: $config?.ShowModel ?? "smart",
|
||||||
|
multiTagsCache: $config?.MultiTagsCache ?? false
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/** 清空缓存后从serverConfig.json读取默认配置并赋值到storage中 */
|
||||||
|
const layout = computed(() => {
|
||||||
|
return $storage?.layout.layout;
|
||||||
|
});
|
||||||
|
|
||||||
|
const layoutTheme = computed(() => {
|
||||||
|
return $storage.layout;
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
layout,
|
||||||
|
layoutTheme,
|
||||||
|
initStorage
|
||||||
|
};
|
||||||
|
}
|
||||||
159
src/layout/hooks/useNav.ts
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
import { computed } from "vue";
|
||||||
|
import { router } from "/@/router";
|
||||||
|
import { getConfig } from "/@/config";
|
||||||
|
import { useRouter } from "vue-router";
|
||||||
|
import { emitter } from "/@/utils/mitt";
|
||||||
|
import { routeMetaType } from "../types";
|
||||||
|
import type { StorageConfigs } from "/#/index";
|
||||||
|
import { routerArrays } from "/@/layout/types";
|
||||||
|
import { transformI18n } from "/@/plugins/i18n";
|
||||||
|
import { useAppStoreHook } from "/@/store/modules/app";
|
||||||
|
import { remainingPaths, resetRouter } from "/@/router";
|
||||||
|
import { storageSession, useGlobal } from "@pureadmin/utils";
|
||||||
|
import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
|
||||||
|
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
||||||
|
|
||||||
|
const errorInfo = "当前路由配置不正确,请检查配置";
|
||||||
|
|
||||||
|
export function useNav() {
|
||||||
|
const pureApp = useAppStoreHook();
|
||||||
|
const routers = useRouter().options.routes;
|
||||||
|
/** 用户名 */
|
||||||
|
const username: string =
|
||||||
|
storageSession.getItem<StorageConfigs>("info")?.username;
|
||||||
|
|
||||||
|
/** 设置国际化选中后的样式 */
|
||||||
|
const getDropdownItemStyle = computed(() => {
|
||||||
|
return (locale, t) => {
|
||||||
|
return {
|
||||||
|
background: locale === t ? useEpThemeStoreHook().epThemeColor : "",
|
||||||
|
color: locale === t ? "#f4f4f5" : "#000"
|
||||||
|
};
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const getDropdownItemClass = computed(() => {
|
||||||
|
return (locale, t) => {
|
||||||
|
return locale === t ? "" : "dark:hover:!text-primary";
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const avatarsStyle = computed(() => {
|
||||||
|
return username ? { marginRight: "10px" } : "";
|
||||||
|
});
|
||||||
|
|
||||||
|
const isCollapse = computed(() => {
|
||||||
|
return !pureApp.getSidebarStatus;
|
||||||
|
});
|
||||||
|
|
||||||
|
const device = computed(() => {
|
||||||
|
return pureApp.getDevice;
|
||||||
|
});
|
||||||
|
|
||||||
|
const { $storage, $config } = useGlobal<GlobalPropertiesApi>();
|
||||||
|
const layout = computed(() => {
|
||||||
|
return $storage?.layout?.layout;
|
||||||
|
});
|
||||||
|
|
||||||
|
const title = computed(() => {
|
||||||
|
return $config.Title;
|
||||||
|
});
|
||||||
|
|
||||||
|
/** 动态title */
|
||||||
|
function changeTitle(meta: routeMetaType) {
|
||||||
|
const Title = getConfig().Title;
|
||||||
|
if (Title) document.title = `${transformI18n(meta.title)} | ${Title}`;
|
||||||
|
else document.title = transformI18n(meta.title);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 退出登录 */
|
||||||
|
function logout() {
|
||||||
|
useMultiTagsStoreHook().handleTags("equal", [...routerArrays]);
|
||||||
|
storageSession.removeItem("info");
|
||||||
|
router.push("/login");
|
||||||
|
resetRouter();
|
||||||
|
}
|
||||||
|
|
||||||
|
function backHome() {
|
||||||
|
router.push("/welcome");
|
||||||
|
}
|
||||||
|
|
||||||
|
function onPanel() {
|
||||||
|
emitter.emit("openPanel");
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleSideBar() {
|
||||||
|
pureApp.toggleSideBar();
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleResize(menuRef) {
|
||||||
|
menuRef?.handleResize();
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolvePath(route) {
|
||||||
|
if (!route.children) return console.error(errorInfo);
|
||||||
|
const httpReg = /^http(s?):\/\//;
|
||||||
|
const routeChildPath = route.children[0]?.path;
|
||||||
|
if (httpReg.test(routeChildPath)) {
|
||||||
|
return route.path + "/" + routeChildPath;
|
||||||
|
} else {
|
||||||
|
return routeChildPath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function menuSelect(indexPath: string, routers): void {
|
||||||
|
if (isRemaining(indexPath)) return;
|
||||||
|
let parentPath = "";
|
||||||
|
const parentPathIndex = indexPath.lastIndexOf("/");
|
||||||
|
if (parentPathIndex > 0) {
|
||||||
|
parentPath = indexPath.slice(0, parentPathIndex);
|
||||||
|
}
|
||||||
|
/** 找到当前路由的信息 */
|
||||||
|
function findCurrentRoute(indexPath: string, routes) {
|
||||||
|
if (!routes) return console.error(errorInfo);
|
||||||
|
return routes.map(item => {
|
||||||
|
if (item.path === indexPath) {
|
||||||
|
if (item.redirect) {
|
||||||
|
findCurrentRoute(item.redirect, item.children);
|
||||||
|
} else {
|
||||||
|
/** 切换左侧菜单 通知标签页 */
|
||||||
|
emitter.emit("changLayoutRoute", {
|
||||||
|
indexPath,
|
||||||
|
parentPath
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (item.children) findCurrentRoute(indexPath, item.children);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
findCurrentRoute(indexPath, routers);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 判断路径是否参与菜单 */
|
||||||
|
function isRemaining(path: string): boolean {
|
||||||
|
return remainingPaths.includes(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
title,
|
||||||
|
device,
|
||||||
|
layout,
|
||||||
|
logout,
|
||||||
|
routers,
|
||||||
|
$storage,
|
||||||
|
backHome,
|
||||||
|
onPanel,
|
||||||
|
changeTitle,
|
||||||
|
toggleSideBar,
|
||||||
|
menuSelect,
|
||||||
|
handleResize,
|
||||||
|
resolvePath,
|
||||||
|
isCollapse,
|
||||||
|
pureApp,
|
||||||
|
username,
|
||||||
|
avatarsStyle,
|
||||||
|
getDropdownItemStyle,
|
||||||
|
getDropdownItemClass
|
||||||
|
};
|
||||||
|
}
|
||||||
218
src/layout/hooks/useTag.ts
Normal file
@@ -0,0 +1,218 @@
|
|||||||
|
import {
|
||||||
|
ref,
|
||||||
|
unref,
|
||||||
|
watch,
|
||||||
|
computed,
|
||||||
|
reactive,
|
||||||
|
onMounted,
|
||||||
|
CSSProperties,
|
||||||
|
getCurrentInstance
|
||||||
|
} from "vue";
|
||||||
|
import { tagsViewsType } from "../types";
|
||||||
|
import { isEqual } from "lodash-unified";
|
||||||
|
import type { StorageConfigs } from "/#/index";
|
||||||
|
import { useEventListener } from "@vueuse/core";
|
||||||
|
import { useRoute, useRouter } from "vue-router";
|
||||||
|
import { transformI18n, $t } from "/@/plugins/i18n";
|
||||||
|
import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
|
||||||
|
import { storageLocal, toggleClass, hasClass } from "@pureadmin/utils";
|
||||||
|
|
||||||
|
import close from "/@/assets/svg/close.svg?component";
|
||||||
|
import refresh from "/@/assets/svg/refresh.svg?component";
|
||||||
|
import closeAll from "/@/assets/svg/close_all.svg?component";
|
||||||
|
import closeLeft from "/@/assets/svg/close_left.svg?component";
|
||||||
|
import closeOther from "/@/assets/svg/close_other.svg?component";
|
||||||
|
import closeRight from "/@/assets/svg/close_right.svg?component";
|
||||||
|
|
||||||
|
export function useTags() {
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const instance = getCurrentInstance();
|
||||||
|
|
||||||
|
const buttonTop = ref(0);
|
||||||
|
const buttonLeft = ref(0);
|
||||||
|
const translateX = ref(0);
|
||||||
|
const visible = ref(false);
|
||||||
|
const activeIndex = ref(-1);
|
||||||
|
// 当前右键选中的路由信息
|
||||||
|
const currentSelect = ref({});
|
||||||
|
|
||||||
|
/** 显示模式,默认灵动模式 */
|
||||||
|
const showModel = ref(
|
||||||
|
storageLocal.getItem<StorageConfigs>("responsive-configure")?.showModel ||
|
||||||
|
"smart"
|
||||||
|
);
|
||||||
|
/** 是否隐藏标签页,默认显示 */
|
||||||
|
const showTags =
|
||||||
|
ref(
|
||||||
|
storageLocal.getItem<StorageConfigs>("responsive-configure").hideTabs
|
||||||
|
) ?? ref("false");
|
||||||
|
const multiTags: any = computed(() => {
|
||||||
|
return useMultiTagsStoreHook().multiTags;
|
||||||
|
});
|
||||||
|
|
||||||
|
const tagsViews = reactive<Array<tagsViewsType>>([
|
||||||
|
{
|
||||||
|
icon: refresh,
|
||||||
|
text: $t("buttons.hsreload"),
|
||||||
|
divided: false,
|
||||||
|
disabled: false,
|
||||||
|
show: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: close,
|
||||||
|
text: $t("buttons.hscloseCurrentTab"),
|
||||||
|
divided: false,
|
||||||
|
disabled: multiTags.value.length > 1 ? false : true,
|
||||||
|
show: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: closeLeft,
|
||||||
|
text: $t("buttons.hscloseLeftTabs"),
|
||||||
|
divided: true,
|
||||||
|
disabled: multiTags.value.length > 1 ? false : true,
|
||||||
|
show: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: closeRight,
|
||||||
|
text: $t("buttons.hscloseRightTabs"),
|
||||||
|
divided: false,
|
||||||
|
disabled: multiTags.value.length > 1 ? false : true,
|
||||||
|
show: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: closeOther,
|
||||||
|
text: $t("buttons.hscloseOtherTabs"),
|
||||||
|
divided: true,
|
||||||
|
disabled: multiTags.value.length > 2 ? false : true,
|
||||||
|
show: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: closeAll,
|
||||||
|
text: $t("buttons.hscloseAllTabs"),
|
||||||
|
divided: false,
|
||||||
|
disabled: multiTags.value.length > 1 ? false : true,
|
||||||
|
show: true
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
|
function conditionHandle(item, previous, next) {
|
||||||
|
if (
|
||||||
|
Object.keys(route.query).length === 0 &&
|
||||||
|
Object.keys(route.params).length === 0
|
||||||
|
) {
|
||||||
|
return route.path === item.path ? previous : next;
|
||||||
|
} else if (Object.keys(route.query).length > 0) {
|
||||||
|
return isEqual(route.query, item.query) ? previous : next;
|
||||||
|
} else {
|
||||||
|
return isEqual(route.params, item.params) ? previous : next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const iconIsActive = computed(() => {
|
||||||
|
return (item, index) => {
|
||||||
|
if (index === 0) return;
|
||||||
|
return conditionHandle(item, true, false);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const linkIsActive = computed(() => {
|
||||||
|
return item => {
|
||||||
|
return conditionHandle(item, "is-active", "");
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const scheduleIsActive = computed(() => {
|
||||||
|
return item => {
|
||||||
|
return conditionHandle(item, "schedule-active", "");
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const getTabStyle = computed((): CSSProperties => {
|
||||||
|
return {
|
||||||
|
transform: `translateX(${translateX.value}px)`
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const getContextMenuStyle = computed((): CSSProperties => {
|
||||||
|
return { left: buttonLeft.value + "px", top: buttonTop.value + "px" };
|
||||||
|
});
|
||||||
|
|
||||||
|
const closeMenu = () => {
|
||||||
|
visible.value = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** 鼠标移入添加激活样式 */
|
||||||
|
function onMouseenter(index) {
|
||||||
|
if (index) activeIndex.value = index;
|
||||||
|
if (unref(showModel) === "smart") {
|
||||||
|
if (hasClass(instance.refs["schedule" + index][0], "schedule-active"))
|
||||||
|
return;
|
||||||
|
toggleClass(true, "schedule-in", instance.refs["schedule" + index][0]);
|
||||||
|
toggleClass(false, "schedule-out", instance.refs["schedule" + index][0]);
|
||||||
|
} else {
|
||||||
|
if (hasClass(instance.refs["dynamic" + index][0], "card-active")) return;
|
||||||
|
toggleClass(true, "card-in", instance.refs["dynamic" + index][0]);
|
||||||
|
toggleClass(false, "card-out", instance.refs["dynamic" + index][0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 鼠标移出恢复默认样式 */
|
||||||
|
function onMouseleave(index) {
|
||||||
|
activeIndex.value = -1;
|
||||||
|
if (unref(showModel) === "smart") {
|
||||||
|
if (hasClass(instance.refs["schedule" + index][0], "schedule-active"))
|
||||||
|
return;
|
||||||
|
toggleClass(false, "schedule-in", instance.refs["schedule" + index][0]);
|
||||||
|
toggleClass(true, "schedule-out", instance.refs["schedule" + index][0]);
|
||||||
|
} else {
|
||||||
|
if (hasClass(instance.refs["dynamic" + index][0], "card-active")) return;
|
||||||
|
toggleClass(false, "card-in", instance.refs["dynamic" + index][0]);
|
||||||
|
toggleClass(true, "card-out", instance.refs["dynamic" + index][0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
if (!showModel.value) {
|
||||||
|
const configure = storageLocal.getItem<StorageConfigs>(
|
||||||
|
"responsive-configure"
|
||||||
|
);
|
||||||
|
configure.showModel = "card";
|
||||||
|
storageLocal.setItem("responsive-configure", configure);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => visible.value,
|
||||||
|
() => {
|
||||||
|
useEventListener(document, "click", closeMenu);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
route,
|
||||||
|
router,
|
||||||
|
visible,
|
||||||
|
showTags,
|
||||||
|
instance,
|
||||||
|
multiTags,
|
||||||
|
showModel,
|
||||||
|
tagsViews,
|
||||||
|
buttonTop,
|
||||||
|
buttonLeft,
|
||||||
|
translateX,
|
||||||
|
activeIndex,
|
||||||
|
getTabStyle,
|
||||||
|
iconIsActive,
|
||||||
|
linkIsActive,
|
||||||
|
currentSelect,
|
||||||
|
scheduleIsActive,
|
||||||
|
getContextMenuStyle,
|
||||||
|
$t,
|
||||||
|
closeMenu,
|
||||||
|
onMounted,
|
||||||
|
onMouseenter,
|
||||||
|
onMouseleave,
|
||||||
|
transformI18n
|
||||||
|
};
|
||||||
|
}
|
||||||
37
src/layout/hooks/useTranslationLang.ts
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import { useNav } from "./useNav";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useRoute } from "vue-router";
|
||||||
|
import { watch, type Ref } from "vue";
|
||||||
|
|
||||||
|
export function useTranslationLang(ref?: Ref) {
|
||||||
|
const { $storage, changeTitle, handleResize } = useNav();
|
||||||
|
const { locale, t } = useI18n();
|
||||||
|
const route = useRoute();
|
||||||
|
|
||||||
|
function translationCh() {
|
||||||
|
$storage.locale = { locale: "zh" };
|
||||||
|
locale.value = "zh";
|
||||||
|
ref && handleResize(ref.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
function translationEn() {
|
||||||
|
$storage.locale = { locale: "en" };
|
||||||
|
locale.value = "en";
|
||||||
|
ref && handleResize(ref.value);
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => locale.value,
|
||||||
|
() => {
|
||||||
|
changeTitle(route.meta);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
t,
|
||||||
|
route,
|
||||||
|
locale,
|
||||||
|
translationCh,
|
||||||
|
translationEn
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,20 +1,11 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {
|
|
||||||
h,
|
|
||||||
reactive,
|
|
||||||
computed,
|
|
||||||
onMounted,
|
|
||||||
defineComponent,
|
|
||||||
getCurrentInstance
|
|
||||||
} from "vue";
|
|
||||||
import { setType } from "./types";
|
import { setType } from "./types";
|
||||||
import { useI18n } from "vue-i18n";
|
|
||||||
import { routerArrays } from "./types";
|
|
||||||
import { emitter } from "/@/utils/mitt";
|
import { emitter } from "/@/utils/mitt";
|
||||||
|
import { useLayout } from "./hooks/useLayout";
|
||||||
import { useAppStoreHook } from "/@/store/modules/app";
|
import { useAppStoreHook } from "/@/store/modules/app";
|
||||||
import { deviceDetection } from "/@/utils/deviceDetection";
|
|
||||||
import { useMultiTagsStore } from "/@/store/modules/multiTags";
|
|
||||||
import { useSettingStoreHook } from "/@/store/modules/settings";
|
import { useSettingStoreHook } from "/@/store/modules/settings";
|
||||||
|
import { deviceDetection, useDark, useGlobal } from "@pureadmin/utils";
|
||||||
|
import { h, reactive, computed, onMounted, defineComponent } from "vue";
|
||||||
|
|
||||||
import backTop from "/@/assets/svg/back_top.svg?component";
|
import backTop from "/@/assets/svg/back_top.svg?component";
|
||||||
import fullScreen from "/@/assets/svg/full_screen.svg?component";
|
import fullScreen from "/@/assets/svg/full_screen.svg?component";
|
||||||
@@ -27,51 +18,11 @@ import setting from "./components/setting/index.vue";
|
|||||||
import Vertical from "./components/sidebar/vertical.vue";
|
import Vertical from "./components/sidebar/vertical.vue";
|
||||||
import Horizontal from "./components/sidebar/horizontal.vue";
|
import Horizontal from "./components/sidebar/horizontal.vue";
|
||||||
|
|
||||||
|
const { isDark } = useDark();
|
||||||
|
const { layout } = useLayout();
|
||||||
const isMobile = deviceDetection();
|
const isMobile = deviceDetection();
|
||||||
const pureSetting = useSettingStoreHook();
|
const pureSetting = useSettingStoreHook();
|
||||||
const instance = getCurrentInstance().appContext.app.config.globalProperties;
|
const { $storage } = useGlobal<GlobalPropertiesApi>();
|
||||||
|
|
||||||
// 清空缓存后从serverConfig.json读取默认配置并赋值到storage中
|
|
||||||
const layout = computed(() => {
|
|
||||||
// 路由
|
|
||||||
if (
|
|
||||||
useMultiTagsStore().multiTagsCache &&
|
|
||||||
(!instance.$storage.tags || instance.$storage.tags.length === 0)
|
|
||||||
) {
|
|
||||||
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
|
||||||
instance.$storage.tags = routerArrays;
|
|
||||||
}
|
|
||||||
// 国际化
|
|
||||||
if (!instance.$storage.locale) {
|
|
||||||
// eslint-disable-next-line
|
|
||||||
instance.$storage.locale = { locale: instance.$config?.Locale ?? "zh" };
|
|
||||||
useI18n().locale.value = instance.$config?.Locale ?? "zh";
|
|
||||||
}
|
|
||||||
// 导航
|
|
||||||
if (!instance.$storage.layout) {
|
|
||||||
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
|
||||||
instance.$storage.layout = {
|
|
||||||
layout: instance.$config?.Layout ?? "vertical",
|
|
||||||
theme: instance.$config?.Theme ?? "default",
|
|
||||||
darkMode: instance.$config?.DarkMode ?? false,
|
|
||||||
sidebarStatus: instance.$config?.SidebarStatus ?? true,
|
|
||||||
epThemeColor: instance.$config?.EpThemeColor ?? "#409EFF"
|
|
||||||
};
|
|
||||||
}
|
|
||||||
// 灰色模式、色弱模式、隐藏标签页
|
|
||||||
if (!instance.$storage.configure) {
|
|
||||||
// eslint-disable-next-line
|
|
||||||
instance.$storage.configure = {
|
|
||||||
grey: instance.$config?.Grey ?? false,
|
|
||||||
weak: instance.$config?.Weak ?? false,
|
|
||||||
hideTabs: instance.$config?.HideTabs ?? false,
|
|
||||||
showLogo: instance.$config?.ShowLogo ?? true,
|
|
||||||
showModel: instance.$config?.ShowModel ?? "smart",
|
|
||||||
multiTagsCache: instance.$config?.MultiTagsCache ?? false
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return instance.$storage?.layout.layout;
|
|
||||||
});
|
|
||||||
|
|
||||||
const set: setType = reactive({
|
const set: setType = reactive({
|
||||||
sidebar: computed(() => {
|
sidebar: computed(() => {
|
||||||
@@ -96,18 +47,18 @@ const set: setType = reactive({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
hideTabs: computed(() => {
|
hideTabs: computed(() => {
|
||||||
return instance.$storage?.configure.hideTabs;
|
return $storage?.configure.hideTabs;
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
function setTheme(layoutModel: string) {
|
function setTheme(layoutModel: string) {
|
||||||
window.document.body.setAttribute("layout", layoutModel);
|
window.document.body.setAttribute("layout", layoutModel);
|
||||||
instance.$storage.layout = {
|
$storage.layout = {
|
||||||
layout: `${layoutModel}`,
|
layout: `${layoutModel}`,
|
||||||
theme: instance.$storage.layout?.theme,
|
theme: $storage.layout?.theme,
|
||||||
darkMode: instance.$storage.layout?.darkMode,
|
darkMode: $storage.layout?.darkMode,
|
||||||
sidebarStatus: instance.$storage.layout?.sidebarStatus,
|
sidebarStatus: $storage.layout?.sidebarStatus,
|
||||||
epThemeColor: instance.$storage.layout?.epThemeColor
|
epThemeColor: $storage.layout?.epThemeColor
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -123,7 +74,7 @@ let isAutoCloseSidebar = true;
|
|||||||
emitter.on("resize", ({ detail }) => {
|
emitter.on("resize", ({ detail }) => {
|
||||||
if (isMobile) return;
|
if (isMobile) return;
|
||||||
let { width } = detail;
|
let { width } = detail;
|
||||||
width <= 670 ? setTheme("vertical") : setTheme(useAppStoreHook().layout);
|
width <= 760 ? setTheme("vertical") : setTheme(useAppStoreHook().layout);
|
||||||
/** width app-wrapper类容器宽度
|
/** width app-wrapper类容器宽度
|
||||||
* 0 < width <= 760 隐藏侧边栏
|
* 0 < width <= 760 隐藏侧边栏
|
||||||
* 760 < width <= 990 折叠侧边栏
|
* 760 < width <= 990 折叠侧边栏
|
||||||
@@ -138,7 +89,7 @@ emitter.on("resize", ({ detail }) => {
|
|||||||
isAutoCloseSidebar = false;
|
isAutoCloseSidebar = false;
|
||||||
}
|
}
|
||||||
} else if (width > 990) {
|
} else if (width > 990) {
|
||||||
if (!set.sidebar.isClickHamburger) {
|
if (!set.sidebar.isClickCollapse) {
|
||||||
toggle("desktop", true);
|
toggle("desktop", true);
|
||||||
isAutoCloseSidebar = true;
|
isAutoCloseSidebar = true;
|
||||||
}
|
}
|
||||||
@@ -165,13 +116,16 @@ const layoutHeader = defineComponent({
|
|||||||
class: { "fixed-header": set.fixedHeader },
|
class: { "fixed-header": set.fixedHeader },
|
||||||
style: [
|
style: [
|
||||||
set.hideTabs && layout.value.includes("horizontal")
|
set.hideTabs && layout.value.includes("horizontal")
|
||||||
? "box-shadow: 0 1px 4px rgb(0 21 41 / 8%);"
|
? isDark.value
|
||||||
|
? "box-shadow: 0 1px 4px #0d0d0d"
|
||||||
|
: "box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08)"
|
||||||
: ""
|
: ""
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
default: () => [
|
default: () => [
|
||||||
!pureSetting.hiddenSideBar && layout.value.includes("vertical")
|
!pureSetting.hiddenSideBar &&
|
||||||
|
(layout.value.includes("vertical") || layout.value.includes("mix"))
|
||||||
? h(navbar)
|
? h(navbar)
|
||||||
: h("div"),
|
: h("div"),
|
||||||
!pureSetting.hiddenSideBar && layout.value.includes("horizontal")
|
!pureSetting.hiddenSideBar && layout.value.includes("horizontal")
|
||||||
@@ -184,10 +138,14 @@ const layoutHeader = defineComponent({
|
|||||||
default: () => [
|
default: () => [
|
||||||
h(
|
h(
|
||||||
"span",
|
"span",
|
||||||
{ onClick: onFullScreen },
|
{
|
||||||
|
onClick: onFullScreen
|
||||||
|
},
|
||||||
{
|
{
|
||||||
default: () => [
|
default: () => [
|
||||||
!pureSetting.hiddenSideBar ? h(fullScreen) : h(exitScreen)
|
!pureSetting.hiddenSideBar
|
||||||
|
? h(fullScreen, { class: "dark:text-white" })
|
||||||
|
: h(exitScreen, { class: "dark:text-white" })
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@@ -213,7 +171,10 @@ const layoutHeader = defineComponent({
|
|||||||
@click="useAppStoreHook().toggleSideBar()"
|
@click="useAppStoreHook().toggleSideBar()"
|
||||||
/>
|
/>
|
||||||
<Vertical
|
<Vertical
|
||||||
v-show="!pureSetting.hiddenSideBar && layout.includes('vertical')"
|
v-show="
|
||||||
|
!pureSetting.hiddenSideBar &&
|
||||||
|
(layout.includes('vertical') || layout.includes('mix'))
|
||||||
|
"
|
||||||
/>
|
/>
|
||||||
<div
|
<div
|
||||||
:class="[
|
:class="[
|
||||||
@@ -230,7 +191,8 @@ const layoutHeader = defineComponent({
|
|||||||
<el-backtop
|
<el-backtop
|
||||||
title="回到顶部"
|
title="回到顶部"
|
||||||
target=".main-container .el-scrollbar__wrap"
|
target=".main-container .el-scrollbar__wrap"
|
||||||
><backTop />
|
>
|
||||||
|
<backTop />
|
||||||
</el-backtop>
|
</el-backtop>
|
||||||
<layout-header />
|
<layout-header />
|
||||||
<!-- 主体内容 -->
|
<!-- 主体内容 -->
|
||||||
|
|||||||
@@ -2,6 +2,10 @@
|
|||||||
import { unref } from "vue";
|
import { unref } from "vue";
|
||||||
import { useRouter } from "vue-router";
|
import { useRouter } from "vue-router";
|
||||||
|
|
||||||
|
defineOptions({
|
||||||
|
name: "Redirect"
|
||||||
|
});
|
||||||
|
|
||||||
const { currentRoute, replace } = useRouter();
|
const { currentRoute, replace } = useRouter();
|
||||||
|
|
||||||
const { params, query } = unref(currentRoute);
|
const { params, query } = unref(currentRoute);
|
||||||
@@ -16,5 +20,5 @@ replace({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div></div>
|
<div />
|
||||||
</template>
|
</template>
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
<template>
|
|
||||||
<router-view>
|
|
||||||
<template #default="{ Component, route }">
|
|
||||||
<transition appear name="fade-transform" mode="out-in">
|
|
||||||
<component :is="Component" :key="route.fullPath" />
|
|
||||||
</transition>
|
|
||||||
</template>
|
|
||||||
</router-view>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
export default {
|
|
||||||
name: "layoutParentView"
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 酸橙绿 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #0b1e15;
|
|
||||||
$menuHover: #60ac80;
|
|
||||||
$subMenuBg: #000;
|
|
||||||
$subMenuActiveBg: #60ac80;
|
|
||||||
$navTextColor: #7a80b4;
|
|
||||||
$menuText: #7a80b4;
|
|
||||||
$sidebarLogo: #112f21;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #60ac80;
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
/**
|
|
||||||
* 道奇蓝(默认)
|
|
||||||
* 此scss变量文件作为multipleScopeVars去编译时,会自动移除!default以达到变量提升
|
|
||||||
* 同时此scss变量文件作为默认主题变量文件,被其他.scss通过 @import 时,必需 !default
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* 菜单选中后字体样式 */
|
|
||||||
$subMenuActiveText: #fff !default;
|
|
||||||
|
|
||||||
/* 菜单背景 */
|
|
||||||
$menuBg: #001529 !default;
|
|
||||||
|
|
||||||
/* 鼠标覆盖到菜单时的背景 */
|
|
||||||
$menuHover: #4091f7 !default;
|
|
||||||
|
|
||||||
/* 子菜单背景 */
|
|
||||||
$subMenuBg: #0f0303 !default;
|
|
||||||
|
|
||||||
/* 有无子集的激活菜单背景 */
|
|
||||||
$subMenuActiveBg: #4091f7 !default;
|
|
||||||
$navTextColor: #fff !default;
|
|
||||||
$menuText: rgba(254, 254, 254, 0.65) !default;
|
|
||||||
|
|
||||||
/* logo背景颜色 */
|
|
||||||
$sidebarLogo: #002140 !default;
|
|
||||||
|
|
||||||
/* 鼠标覆盖到菜单时的字体颜色 */
|
|
||||||
$menuTitleHover: #fff !default;
|
|
||||||
$menuActiveBefore: #4091f7 !default;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 猩红色 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #2a0608;
|
|
||||||
$menuHover: #e13c39;
|
|
||||||
$subMenuBg: #000;
|
|
||||||
$subMenuActiveBg: #e13c39;
|
|
||||||
$navTextColor: red;
|
|
||||||
$menuText: rgba(254, 254, 254, 0.651);
|
|
||||||
$sidebarLogo: #42090c;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #e13c39;
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
/* 动态改变element-plus主题色 */
|
|
||||||
import rgbHex from "rgb-hex";
|
|
||||||
import color from "css-color-function";
|
|
||||||
import { TinyColor } from "@ctrl/tinycolor";
|
|
||||||
import epCss from "element-plus/dist/index.css";
|
|
||||||
|
|
||||||
// 色值表
|
|
||||||
const formula = {
|
|
||||||
"shade-1": "color(primary shade(10%))",
|
|
||||||
"light-1": "color(primary tint(10%))",
|
|
||||||
"light-2": "color(primary tint(20%))",
|
|
||||||
"light-3": "color(primary tint(30%))",
|
|
||||||
"light-4": "color(primary tint(40%))",
|
|
||||||
"light-5": "color(primary tint(50%))",
|
|
||||||
"light-6": "color(primary tint(60%))",
|
|
||||||
"light-7": "color(primary tint(70%))",
|
|
||||||
"light-8": "color(primary tint(80%))",
|
|
||||||
"light-9": "color(primary tint(90%))"
|
|
||||||
};
|
|
||||||
|
|
||||||
// 把生成的样式表写入到style中
|
|
||||||
export const writeNewStyle = (newStyle: string): void => {
|
|
||||||
const style = window.document.createElement("style");
|
|
||||||
style.innerText = newStyle;
|
|
||||||
window.document.head.appendChild(style);
|
|
||||||
};
|
|
||||||
|
|
||||||
// 根据主题色,生成最新的样式表
|
|
||||||
export const createNewStyle = (primaryStyle: string): string => {
|
|
||||||
// 根据主色生成色值表
|
|
||||||
const colors = createColors(primaryStyle);
|
|
||||||
// 在当前ep的默认样式表中标记需要替换的色值
|
|
||||||
let cssText = getStyleTemplate(epCss);
|
|
||||||
// 遍历生成的色值表,在 默认样式表 进行全局替换
|
|
||||||
Object.keys(colors).forEach(key => {
|
|
||||||
cssText = cssText.replace(
|
|
||||||
new RegExp("(:|\\s+)" + key, "g"),
|
|
||||||
"$1" + colors[key]
|
|
||||||
);
|
|
||||||
});
|
|
||||||
return cssText;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createColors = (primary: string) => {
|
|
||||||
if (!primary) return;
|
|
||||||
const colors = {
|
|
||||||
primary
|
|
||||||
};
|
|
||||||
Object.keys(formula).forEach(key => {
|
|
||||||
const value = formula[key].replace(/primary/, primary);
|
|
||||||
colors[key] = "#" + rgbHex(color.convert(value));
|
|
||||||
});
|
|
||||||
return colors;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getStyleTemplate = (data: string): string => {
|
|
||||||
const colorMap = {
|
|
||||||
"#3a8ee6": "shade-1",
|
|
||||||
"#409eff": "primary",
|
|
||||||
"#53a8ff": "light-1",
|
|
||||||
"#66b1ff": "light-2",
|
|
||||||
"#79bbff": "light-3",
|
|
||||||
"#8cc5ff": "light-4",
|
|
||||||
"#a0cfff": "light-5",
|
|
||||||
"#b3d8ff": "light-6",
|
|
||||||
"#c6e2ff": "light-7",
|
|
||||||
"#d9ecff": "light-8",
|
|
||||||
"#ecf5ff": "light-9"
|
|
||||||
};
|
|
||||||
Object.keys(colorMap).forEach(key => {
|
|
||||||
const value = colorMap[key];
|
|
||||||
data = data.replace(new RegExp(key, "ig"), value);
|
|
||||||
});
|
|
||||||
return data;
|
|
||||||
};
|
|
||||||
|
|
||||||
// 自动计算hover和active颜色 https://element-plus.gitee.io/zh-CN/component/button.html#%E8%87%AA%E5%AE%9A%E4%B9%89%E9%A2%9C%E8%89%B2-%E6%B5%8B%E8%AF%95%E7%89%88
|
|
||||||
export const shadeBgColor = (color: string): string => {
|
|
||||||
return new TinyColor(color).shade(10).toString();
|
|
||||||
};
|
|
||||||
159
src/layout/theme/index.ts
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
/**
|
||||||
|
* @description ⚠️:此文件仅供主题插件使用,请不要在此文件中导出别的工具函数(仅在页面加载前运行)
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { EpThemeColor } from "../../../public/serverConfig.json";
|
||||||
|
|
||||||
|
type MultipleScopeVarsItem = {
|
||||||
|
scopeName: string;
|
||||||
|
varsContent: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** 将vxe默认主题色和ep默认主题色保持一致 */
|
||||||
|
const vxeColor = EpThemeColor;
|
||||||
|
/** 预设主题色 */
|
||||||
|
const themeColors = {
|
||||||
|
default: {
|
||||||
|
vxeColor,
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#001529",
|
||||||
|
menuHover: "#4091f7",
|
||||||
|
subMenuBg: "#0f0303",
|
||||||
|
subMenuActiveBg: "#4091f7",
|
||||||
|
navTextColor: "#fff",
|
||||||
|
menuText: "rgb(254 254 254 / 65%)",
|
||||||
|
sidebarLogo: "#002140",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#4091f7"
|
||||||
|
},
|
||||||
|
light: {
|
||||||
|
vxeColor,
|
||||||
|
subMenuActiveText: "#409eff",
|
||||||
|
menuBg: "#fff",
|
||||||
|
menuHover: "#e0ebf6",
|
||||||
|
subMenuBg: "#fff",
|
||||||
|
subMenuActiveBg: "#e0ebf6",
|
||||||
|
navTextColor: "#7a80b4",
|
||||||
|
menuText: "#7a80b4",
|
||||||
|
sidebarLogo: "#fff",
|
||||||
|
menuTitleHover: "#000",
|
||||||
|
menuActiveBefore: "#4091f7"
|
||||||
|
},
|
||||||
|
dusk: {
|
||||||
|
vxeColor: "#f5222d",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#2a0608",
|
||||||
|
menuHover: "#e13c39",
|
||||||
|
subMenuBg: "#000",
|
||||||
|
subMenuActiveBg: "#e13c39",
|
||||||
|
navTextColor: "#red",
|
||||||
|
menuText: "rgb(254 254 254 / 65.1%)",
|
||||||
|
sidebarLogo: "#42090c",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#e13c39"
|
||||||
|
},
|
||||||
|
volcano: {
|
||||||
|
vxeColor: "#fa541c",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#2b0e05",
|
||||||
|
menuHover: "#e85f33",
|
||||||
|
subMenuBg: "#0f0603",
|
||||||
|
subMenuActiveBg: "#e85f33",
|
||||||
|
navTextColor: "#fff",
|
||||||
|
menuText: "rgb(254 254 254 / 65%)",
|
||||||
|
sidebarLogo: "#441708",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#e85f33"
|
||||||
|
},
|
||||||
|
yellow: {
|
||||||
|
vxeColor: "#fadb14",
|
||||||
|
subMenuActiveText: "#d25f00",
|
||||||
|
menuBg: "#2b2503",
|
||||||
|
menuHover: "#f6da4d",
|
||||||
|
subMenuBg: "#0f0603",
|
||||||
|
subMenuActiveBg: "#f6da4d",
|
||||||
|
navTextColor: "#fff",
|
||||||
|
menuText: "rgb(254 254 254 / 65%)",
|
||||||
|
sidebarLogo: "#443b05",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#f6da4d"
|
||||||
|
},
|
||||||
|
mingQing: {
|
||||||
|
vxeColor: "#13c2c2",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#032121",
|
||||||
|
menuHover: "#59bfc1",
|
||||||
|
subMenuBg: "#000",
|
||||||
|
subMenuActiveBg: "#59bfc1",
|
||||||
|
navTextColor: "#7a80b4",
|
||||||
|
menuText: "#7a80b4",
|
||||||
|
sidebarLogo: "#053434",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#59bfc1"
|
||||||
|
},
|
||||||
|
auroraGreen: {
|
||||||
|
vxeColor: "#52c41a",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#0b1e15",
|
||||||
|
menuHover: "#60ac80",
|
||||||
|
subMenuBg: "#000",
|
||||||
|
subMenuActiveBg: "#60ac80",
|
||||||
|
navTextColor: "#7a80b4",
|
||||||
|
menuText: "#7a80b4",
|
||||||
|
sidebarLogo: "#112f21",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#60ac80"
|
||||||
|
},
|
||||||
|
pink: {
|
||||||
|
vxeColor: "#eb2f96",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#28081a",
|
||||||
|
menuHover: "#d84493",
|
||||||
|
subMenuBg: "#000",
|
||||||
|
subMenuActiveBg: "#d84493",
|
||||||
|
navTextColor: "#7a80b4",
|
||||||
|
menuText: "#7a80b4",
|
||||||
|
sidebarLogo: "#3f0d29",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#d84493"
|
||||||
|
},
|
||||||
|
saucePurple: {
|
||||||
|
vxeColor: "#722ed1",
|
||||||
|
subMenuActiveText: "#fff",
|
||||||
|
menuBg: "#130824",
|
||||||
|
menuHover: "#693ac9",
|
||||||
|
subMenuBg: "#000",
|
||||||
|
subMenuActiveBg: "#693ac9",
|
||||||
|
navTextColor: "#7a80b4",
|
||||||
|
menuText: "#7a80b4",
|
||||||
|
sidebarLogo: "#1f0c38",
|
||||||
|
menuTitleHover: "#fff",
|
||||||
|
menuActiveBefore: "#693ac9"
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @description 将预设主题色处理成主题插件所需格式
|
||||||
|
*/
|
||||||
|
export const genScssMultipleScopeVars = (): MultipleScopeVarsItem[] => {
|
||||||
|
const result = [] as MultipleScopeVarsItem[];
|
||||||
|
Object.keys(themeColors).forEach(key => {
|
||||||
|
result.push({
|
||||||
|
scopeName: `layout-theme-${key}`,
|
||||||
|
varsContent: `
|
||||||
|
$vxe-primary-color: ${themeColors[key].vxeColor} !default;
|
||||||
|
$subMenuActiveText: ${themeColors[key].subMenuActiveText} !default;
|
||||||
|
$menuBg: ${themeColors[key].menuBg} !default;
|
||||||
|
$menuHover: ${themeColors[key].menuHover} !default;
|
||||||
|
$subMenuBg: ${themeColors[key].subMenuBg} !default;
|
||||||
|
$subMenuActiveBg: ${themeColors[key].subMenuActiveBg} !default;
|
||||||
|
$navTextColor: ${themeColors[key].navTextColor} !default;
|
||||||
|
$menuText: ${themeColors[key].menuText} !default;
|
||||||
|
$sidebarLogo: ${themeColors[key].sidebarLogo} !default;
|
||||||
|
$menuTitleHover: ${themeColors[key].menuTitleHover} !default;
|
||||||
|
$menuActiveBefore: ${themeColors[key].menuActiveBefore} !default;
|
||||||
|
`
|
||||||
|
} as MultipleScopeVarsItem);
|
||||||
|
});
|
||||||
|
return result;
|
||||||
|
};
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 亮白色 */
|
|
||||||
$subMenuActiveText: #409eff;
|
|
||||||
$menuBg: #fff;
|
|
||||||
$menuHover: #e0ebf6;
|
|
||||||
$subMenuBg: #fff;
|
|
||||||
$subMenuActiveBg: #e0ebf6;
|
|
||||||
$navTextColor: #7a80b4;
|
|
||||||
$menuText: #7a80b4;
|
|
||||||
$sidebarLogo: #fff;
|
|
||||||
$menuTitleHover: #000;
|
|
||||||
$menuActiveBefore: #4091f7;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 绿宝石 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #032121;
|
|
||||||
$menuHover: #59bfc1;
|
|
||||||
$subMenuBg: #000;
|
|
||||||
$subMenuActiveBg: #59bfc1;
|
|
||||||
$navTextColor: #7a80b4;
|
|
||||||
$menuText: #7a80b4;
|
|
||||||
$sidebarLogo: #053434;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #59bfc1;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 深粉色 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #28081a;
|
|
||||||
$menuHover: #d84493;
|
|
||||||
$subMenuBg: #000;
|
|
||||||
$subMenuActiveBg: #d84493;
|
|
||||||
$navTextColor: #7a80b4;
|
|
||||||
$menuText: #7a80b4;
|
|
||||||
$sidebarLogo: #3f0d29;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #d84493;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 深紫罗兰色 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #130824;
|
|
||||||
$menuHover: #693ac9;
|
|
||||||
$subMenuBg: #000;
|
|
||||||
$subMenuActiveBg: #693ac9;
|
|
||||||
$navTextColor: #7a80b4;
|
|
||||||
$menuText: #7a80b4;
|
|
||||||
$sidebarLogo: #1f0c38;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #693ac9;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 橙红色 */
|
|
||||||
$subMenuActiveText: #fff;
|
|
||||||
$menuBg: #2b0e05;
|
|
||||||
$menuHover: #e85f33;
|
|
||||||
$subMenuBg: #0f0603;
|
|
||||||
$subMenuActiveBg: #e85f33;
|
|
||||||
$navTextColor: #fff;
|
|
||||||
$menuText: rgba(254, 254, 254, 0.65);
|
|
||||||
$sidebarLogo: #441708;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #e85f33;
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
/* 金色 */
|
|
||||||
$subMenuActiveText: #d25f00;
|
|
||||||
$menuBg: #2b2503;
|
|
||||||
$menuHover: #f6da4d;
|
|
||||||
$subMenuBg: #0f0603;
|
|
||||||
$subMenuActiveBg: #f6da4d;
|
|
||||||
$navTextColor: #fff;
|
|
||||||
$menuText: rgba(254, 254, 254, 0.65);
|
|
||||||
$sidebarLogo: #443b05;
|
|
||||||
$menuTitleHover: #fff;
|
|
||||||
$menuActiveBefore: #f6da4d;
|
|
||||||
@@ -5,25 +5,25 @@ export const routerArrays: Array<RouteConfigs> = [
|
|||||||
parentPath: "/",
|
parentPath: "/",
|
||||||
meta: {
|
meta: {
|
||||||
title: "menus.hshome",
|
title: "menus.hshome",
|
||||||
i18n: true,
|
icon: "home-filled"
|
||||||
icon: "home-filled",
|
|
||||||
showLink: true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export type routeMetaType = {
|
||||||
|
title?: string;
|
||||||
|
icon?: string;
|
||||||
|
showLink?: boolean;
|
||||||
|
savedPosition?: boolean;
|
||||||
|
authority?: Array<string>;
|
||||||
|
};
|
||||||
|
|
||||||
export type RouteConfigs = {
|
export type RouteConfigs = {
|
||||||
path?: string;
|
path?: string;
|
||||||
parentPath?: string;
|
parentPath?: string;
|
||||||
query?: object;
|
query?: object;
|
||||||
meta?: {
|
params?: object;
|
||||||
title?: string;
|
meta?: routeMetaType;
|
||||||
i18n?: boolean;
|
|
||||||
icon?: string;
|
|
||||||
showLink?: boolean;
|
|
||||||
savedPosition?: boolean;
|
|
||||||
authority?: Array<string>;
|
|
||||||
};
|
|
||||||
children?: RouteConfigs[];
|
children?: RouteConfigs[];
|
||||||
name?: string;
|
name?: string;
|
||||||
};
|
};
|
||||||
@@ -44,7 +44,7 @@ export interface setType {
|
|||||||
sidebar: {
|
sidebar: {
|
||||||
opened: boolean;
|
opened: boolean;
|
||||||
withoutAnimation: boolean;
|
withoutAnimation: boolean;
|
||||||
isClickHamburger: boolean;
|
isClickCollapse: boolean;
|
||||||
};
|
};
|
||||||
device: string;
|
device: string;
|
||||||
fixedHeader: boolean;
|
fixedHeader: boolean;
|
||||||
@@ -65,17 +65,19 @@ export type childrenType = {
|
|||||||
meta?: {
|
meta?: {
|
||||||
icon?: string;
|
icon?: string;
|
||||||
title?: string;
|
title?: string;
|
||||||
i18n?: boolean;
|
showParent?: boolean;
|
||||||
extraIcon?: {
|
extraIcon?: {
|
||||||
svg?: boolean;
|
svg?: boolean;
|
||||||
name?: string;
|
name?: string;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
showTooltip?: boolean;
|
showTooltip?: boolean;
|
||||||
|
parentId?: number;
|
||||||
|
pathList?: number[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type themeColorsType = {
|
export type themeColorsType = {
|
||||||
rgb: string;
|
color: string;
|
||||||
themeColor: string;
|
themeColor: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
40
src/main.ts
@@ -1,18 +1,27 @@
|
|||||||
import App from "./App.vue";
|
import App from "./App.vue";
|
||||||
import router from "./router";
|
import router from "./router";
|
||||||
import { setupStore } from "/@/store";
|
import { setupStore } from "/@/store";
|
||||||
|
import ElementPlus from "element-plus";
|
||||||
import { getServerConfig } from "./config";
|
import { getServerConfig } from "./config";
|
||||||
import { createApp, Directive } from "vue";
|
import { createApp, Directive } from "vue";
|
||||||
import { usI18n } from "../src/plugins/i18n";
|
import { useI18n } from "../src/plugins/i18n";
|
||||||
import { MotionPlugin } from "@vueuse/motion";
|
import { MotionPlugin } from "@vueuse/motion";
|
||||||
import { useFontawesome } from "../src/plugins/fontawesome";
|
// import { useEcharts } from "/@/plugins/echarts";
|
||||||
import { useElementPlus } from "../src/plugins/element-plus";
|
// import { useTable } from "../src/plugins/vxe-table";
|
||||||
import { injectResponsiveStorage } from "/@/utils/storage/responsive";
|
import { injectResponsiveStorage } from "/@/utils/responsive";
|
||||||
|
|
||||||
|
// import Table from "@pureadmin/table";
|
||||||
|
// import PureDescriptions from "@pureadmin/descriptions";
|
||||||
|
|
||||||
import "animate.css";
|
import "animate.css";
|
||||||
import "virtual:windi.css";
|
// 引入重置样式
|
||||||
|
import "./style/reset.scss";
|
||||||
// 导入公共样式
|
// 导入公共样式
|
||||||
import "./style/index.scss";
|
import "./style/index.scss";
|
||||||
|
import "element-plus/dist/index.css";
|
||||||
|
import "@pureadmin/components/dist/index.css";
|
||||||
|
import "@pureadmin/components/dist/theme.css";
|
||||||
|
import "@pureadmin/components/dist/dark.scss";
|
||||||
// 导入字体图标
|
// 导入字体图标
|
||||||
import "./assets/iconfont/iconfont.js";
|
import "./assets/iconfont/iconfont.js";
|
||||||
import "./assets/iconfont/iconfont.css";
|
import "./assets/iconfont/iconfont.css";
|
||||||
@@ -26,19 +35,24 @@ Object.keys(directives).forEach(key => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// 全局注册`@iconify/vue`图标库
|
// 全局注册`@iconify/vue`图标库
|
||||||
import { IconifyIconOffline, IconifyIconOnline } from "./components/ReIcon";
|
import {
|
||||||
|
IconifyIconOffline,
|
||||||
|
IconifyIconOnline,
|
||||||
|
FontIcon
|
||||||
|
} from "./components/ReIcon";
|
||||||
app.component("IconifyIconOffline", IconifyIconOffline);
|
app.component("IconifyIconOffline", IconifyIconOffline);
|
||||||
app.component("IconifyIconOnline", IconifyIconOnline);
|
app.component("IconifyIconOnline", IconifyIconOnline);
|
||||||
|
app.component("FontIcon", FontIcon);
|
||||||
|
|
||||||
getServerConfig(app).then(async config => {
|
getServerConfig(app).then(async config => {
|
||||||
|
app.use(router);
|
||||||
|
await router.isReady();
|
||||||
injectResponsiveStorage(app, config);
|
injectResponsiveStorage(app, config);
|
||||||
setupStore(app);
|
setupStore(app);
|
||||||
app
|
app.use(MotionPlugin).use(useI18n).use(ElementPlus);
|
||||||
.use(router)
|
// .use(useEcharts);
|
||||||
.use(MotionPlugin)
|
// .use(Table);
|
||||||
.use(useElementPlus)
|
// .use(PureDescriptions);
|
||||||
.use(usI18n)
|
// .use(useTable);
|
||||||
.use(useFontawesome);
|
|
||||||
await router.isReady();
|
|
||||||
app.mount("#app");
|
app.mount("#app");
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
import { createProdMockServer } from "vite-plugin-mock/es/createProdMockServer";
|
import { createProdMockServer } from "vite-plugin-mock/es/createProdMockServer";
|
||||||
import asyncRoutesMock from "../mock/asyncRoutes";
|
|
||||||
|
|
||||||
export const mockModules = [...asyncRoutesMock];
|
const modules: Record<string, any> = import.meta.glob("../mock/*.ts", {
|
||||||
|
eager: true
|
||||||
|
});
|
||||||
|
const mockModules = [];
|
||||||
|
|
||||||
|
Object.keys(modules).forEach(key => {
|
||||||
|
mockModules.push(...modules[key].default);
|
||||||
|
});
|
||||||
|
|
||||||
export function setupProdMockServer() {
|
export function setupProdMockServer() {
|
||||||
createProdMockServer(mockModules);
|
createProdMockServer(mockModules);
|
||||||
|
|||||||