初始化提交
This commit is contained in:
8
web/.env
Normal file
8
web/.env
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# port 端口号
|
||||||
|
VITE_PORT = 8888
|
||||||
|
|
||||||
|
# open 运行 npm run dev 时自动打开浏览器
|
||||||
|
VITE_OPEN = false
|
||||||
|
|
||||||
|
# public path 配置线上环境路径(打包)、本地通过 http-server 访问时,请置空即可
|
||||||
|
VITE_PUBLIC_PATH = /vue-next-admin-preview/
|
||||||
5
web/.env.development
Normal file
5
web/.env.development
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# 本地环境
|
||||||
|
ENV = 'development'
|
||||||
|
|
||||||
|
# 本地环境接口地址
|
||||||
|
VITE_API_URL = 'http://localhost:8888/'
|
||||||
5
web/.env.production
Normal file
5
web/.env.production
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# 线上环境
|
||||||
|
ENV = 'production'
|
||||||
|
|
||||||
|
# 线上环境接口地址
|
||||||
|
VITE_API_URL = 'https://lyt-top.gitee.io/vue-next-admin-preview/'
|
||||||
18
web/.eslintignore
Normal file
18
web/.eslintignore
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
|
||||||
|
*.sh
|
||||||
|
node_modules
|
||||||
|
lib
|
||||||
|
*.md
|
||||||
|
*.scss
|
||||||
|
*.woff
|
||||||
|
*.ttf
|
||||||
|
.vscode
|
||||||
|
.idea
|
||||||
|
dist
|
||||||
|
mock
|
||||||
|
public
|
||||||
|
bin
|
||||||
|
build
|
||||||
|
config
|
||||||
|
index.html
|
||||||
|
src/assets
|
||||||
63
web/.eslintrc.js
Normal file
63
web/.eslintrc.js
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
module.exports = {
|
||||||
|
root: true,
|
||||||
|
env: {
|
||||||
|
browser: true,
|
||||||
|
es2021: true,
|
||||||
|
node: true,
|
||||||
|
},
|
||||||
|
parser: 'vue-eslint-parser',
|
||||||
|
parserOptions: {
|
||||||
|
ecmaVersion: 12,
|
||||||
|
parser: '@typescript-eslint/parser',
|
||||||
|
sourceType: 'module',
|
||||||
|
},
|
||||||
|
extends: ['plugin:vue/vue3-essential', 'plugin:vue/essential', 'eslint:recommended'],
|
||||||
|
plugins: ['vue', '@typescript-eslint'],
|
||||||
|
rules: {
|
||||||
|
// http://eslint.cn/docs/rules/
|
||||||
|
// https://eslint.vuejs.org/rules/
|
||||||
|
'@typescript-eslint/ban-ts-ignore': 'off',
|
||||||
|
'@typescript-eslint/explicit-function-return-type': 'off',
|
||||||
|
'@typescript-eslint/no-explicit-any': 'off',
|
||||||
|
'@typescript-eslint/no-var-requires': 'off',
|
||||||
|
'@typescript-eslint/no-empty-function': 'off',
|
||||||
|
'@typescript-eslint/no-use-before-define': 'off',
|
||||||
|
'@typescript-eslint/ban-ts-comment': 'off',
|
||||||
|
'@typescript-eslint/ban-types': 'off',
|
||||||
|
'@typescript-eslint/no-non-null-assertion': 'off',
|
||||||
|
'@typescript-eslint/explicit-module-boundary-types': 'off',
|
||||||
|
'vue/custom-event-name-casing': 'off',
|
||||||
|
'vue/attributes-order': 'off',
|
||||||
|
'vue/one-component-per-file': 'off',
|
||||||
|
'vue/html-closing-bracket-newline': 'off',
|
||||||
|
'vue/max-attributes-per-line': 'off',
|
||||||
|
'vue/multiline-html-element-content-newline': 'off',
|
||||||
|
'vue/singleline-html-element-content-newline': 'off',
|
||||||
|
'vue/attribute-hyphenation': 'off',
|
||||||
|
'vue/html-self-closing': 'off',
|
||||||
|
'vue/no-multiple-template-root': 'off',
|
||||||
|
'vue/require-default-prop': 'off',
|
||||||
|
'vue/no-v-model-argument': 'off',
|
||||||
|
'vue/no-arrow-functions-in-watch': 'off',
|
||||||
|
'vue/no-template-key': 'off',
|
||||||
|
'vue/no-v-html': 'off',
|
||||||
|
'vue/comment-directive': 'off',
|
||||||
|
'vue/no-parsing-error': 'off',
|
||||||
|
'vue/no-deprecated-v-on-native-modifier': 'off',
|
||||||
|
'vue/multi-word-component-names': 'off',
|
||||||
|
'no-useless-escape': 'off',
|
||||||
|
'no-sparse-arrays': 'off',
|
||||||
|
'no-prototype-builtins': 'off',
|
||||||
|
'no-constant-condition': 'off',
|
||||||
|
'no-use-before-define': 'off',
|
||||||
|
'no-restricted-globals': 'off',
|
||||||
|
'no-restricted-syntax': 'off',
|
||||||
|
'generator-star-spacing': 'off',
|
||||||
|
'no-unreachable': 'off',
|
||||||
|
'no-multiple-template-root': 'off',
|
||||||
|
'no-unused-vars': 'error',
|
||||||
|
'no-v-model-argument': 'off',
|
||||||
|
'no-case-declarations': 'off',
|
||||||
|
'no-console': 'error',
|
||||||
|
},
|
||||||
|
};
|
||||||
23
web/.gitignore
vendored
Normal file
23
web/.gitignore
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
.DS_Store
|
||||||
|
node_modules
|
||||||
|
/dist
|
||||||
|
|
||||||
|
|
||||||
|
# local env files
|
||||||
|
.env.local
|
||||||
|
.env.*.local
|
||||||
|
|
||||||
|
# Log files
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
|
||||||
|
# Editor directories and files
|
||||||
|
.idea
|
||||||
|
.vscode
|
||||||
|
*.suo
|
||||||
|
*.ntvs*
|
||||||
|
*.njsproj
|
||||||
|
*.sln
|
||||||
|
*.sw?
|
||||||
39
web/.prettierrc.js
Normal file
39
web/.prettierrc.js
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
module.exports = {
|
||||||
|
// 一行最多多少个字符
|
||||||
|
printWidth: 150,
|
||||||
|
// 指定每个缩进级别的空格数
|
||||||
|
tabWidth: 2,
|
||||||
|
// 使用制表符而不是空格缩进行
|
||||||
|
useTabs: true,
|
||||||
|
// 在语句末尾打印分号
|
||||||
|
semi: true,
|
||||||
|
// 使用单引号而不是双引号
|
||||||
|
singleQuote: true,
|
||||||
|
// 更改引用对象属性的时间 可选值"<as-needed|consistent|preserve>"
|
||||||
|
quoteProps: 'as-needed',
|
||||||
|
// 在JSX中使用单引号而不是双引号
|
||||||
|
jsxSingleQuote: false,
|
||||||
|
// 多行时尽可能打印尾随逗号。(例如,单行数组永远不会出现逗号结尾。) 可选值"<none|es5|all>",默认none
|
||||||
|
trailingComma: 'es5',
|
||||||
|
// 在对象文字中的括号之间打印空格
|
||||||
|
bracketSpacing: true,
|
||||||
|
// jsx 标签的反尖括号需要换行
|
||||||
|
jsxBracketSameLine: false,
|
||||||
|
// 在单独的箭头函数参数周围包括括号 always:(x) => x \ avoid:x => x
|
||||||
|
arrowParens: 'always',
|
||||||
|
// 这两个选项可用于格式化以给定字符偏移量(分别包括和不包括)开始和结束的代码
|
||||||
|
rangeStart: 0,
|
||||||
|
rangeEnd: Infinity,
|
||||||
|
// 指定要使用的解析器,不需要写文件开头的 @prettier
|
||||||
|
requirePragma: false,
|
||||||
|
// 不需要自动在文件开头插入 @prettier
|
||||||
|
insertPragma: false,
|
||||||
|
// 使用默认的折行标准 always\never\preserve
|
||||||
|
proseWrap: 'preserve',
|
||||||
|
// 指定HTML文件的全局空格敏感度 css\strict\ignore
|
||||||
|
htmlWhitespaceSensitivity: 'css',
|
||||||
|
// Vue文件脚本和样式标签缩进
|
||||||
|
vueIndentScriptAndStyle: false,
|
||||||
|
// 换行符使用 lf 结尾是 可选值"<auto|lf|crlf|cr>"
|
||||||
|
endOfLine: 'lf',
|
||||||
|
};
|
||||||
367
web/CHANGELOG.md
Normal file
367
web/CHANGELOG.md
Normal file
@@ -0,0 +1,367 @@
|
|||||||
|
# <a href="https://gitee.com/lyt-top/vue-next-admin" target="_blank">vue-next-admin 更新日志</a>
|
||||||
|
|
||||||
|
🎉🎉🔥 `vue-next-admin` 基于 vue3.x 、Typescript、vite、Element plus 等,适配手机、平板、pc 的后台开源免费模板库(vue2.x 请切换 vue-prev-admin 分支)
|
||||||
|
|
||||||
|
## 2.2.0
|
||||||
|
|
||||||
|
`2022.07.10`
|
||||||
|
|
||||||
|
⚡⚡⚡ [/sec/stores/userInfo.ts](https://gitee.com/lyt-top/vue-next-admin/blob/master/src/stores/userInfo.ts) 下添加了 `getApiUserInfo` 接口模拟数据 `setTimeout` 为 3 秒
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🐞 修复 [主界面重新授权按钮点击卡死不跳转登录界面#I5C3JS](https://gitee.com/lyt-top/vue-next-admin/issues/I5C3JS),感谢[@Hero-Typ](https://gitee.com/tian_yu_peng)
|
||||||
|
- 🐞 修复 编译警告[#I5CVSB](https://gitee.com/lyt-top/vue-next-admin/issues/I5CVSB),全局替换成 `:deep(attr)`,感谢[@Linvas](https://gitee.com/linvas)。参考文档:[vue3 sfc-style](https://v3.cn.vuejs.org/api/sfc-style.html#style-scoped)。`node_modules\print-js\dist\print.js` 需 `print-js` 作者适配或去除 `package.json` 中的 `"print-js": "^1.6.0"`
|
||||||
|
- 🐞 修复 [vue-next-admin-template-js 版本前端控制路由:userInfo.js 请求用户信息接口报错,加载不到路由 可以写个定时器模拟一下接口 一样的报错#I5F1HP](https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP),感谢[@白开水](https://gitee.com/libin951223)
|
||||||
|
|
||||||
|
## 2.1.1
|
||||||
|
|
||||||
|
`2022.05.27`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 深色模式下,`<el-button text></el-button>` 时,`:active` 样式
|
||||||
|
- 🎯 优化 [页面缓存在刷新之后失效 #I58U75](https://gitee.com/lyt-top/vue-next-admin/issues/I58U75)),感谢[@ls0428](https://gitee.com/ls0428)
|
||||||
|
- 🎯 优化 [SvgIcon 对下载的 Svg 图像设置颜色无效 #I59ND0](https://gitee.com/lyt-top/vue-next-admin/issues/I59ND0)),感谢[@elus_z](https://gitee.com/elus_z)
|
||||||
|
- 🎯 优化 `/src/utils/toolsValidate.ts` 工具类
|
||||||
|
- 🐞 修复 [布局切换,TagsView 显示的 tab 会多一个出来 #I58WGM](https://gitee.com/lyt-top/vue-next-admin/issues/I58WGM),感谢[@lg_boy](https://gitee.com/lg_boy)
|
||||||
|
- 🐞 修复 [如果设置顶部面包屑导航开启图标 isBreadcrumbIcon=true 后,样式有点问题 如果不开启就是正常的 #I58VB8](https://gitee.com/lyt-top/vue-next-admin/issues/I58VB8)
|
||||||
|
- 🐞 修复 地址栏路由地址输入错误时,返回首页后,再次输入路由地址错误时,不跳转 404 问题
|
||||||
|
- 🐞 修复 [2.1.0 版本的图标选择组件多次点击后功能失效 #I590TH](https://gitee.com/lyt-top/vue-next-admin/issues/I590TH),感谢[@quber](https://gitee.com/quber)
|
||||||
|
|
||||||
|
## 2.1.0
|
||||||
|
|
||||||
|
`2022.04.18`
|
||||||
|
|
||||||
|
⚡⚡⚡ 此版本为破环性更新,优化内容如下:(谨慎更新!谨慎更新!!谨慎更新!!!)。因为 `vuex` 替换成 `pinia`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 部分界面图片不显示问题(更换 gitee 在线图片地址源)
|
||||||
|
- 🎯 优化 各界面方法引入与逻辑之间添加一行空行,方便区分内容
|
||||||
|
- 🎯 优化 图标选择器 [#I4YAHB](https://gitee.com/lyt-top/vue-next-admin/issues/I4YAHB),感谢[@真有你的](https://gitee.com/sunliusen)
|
||||||
|
- 🎯 优化 图标选择器 icon type 类型为 all 时,类型 ali、ele、awe 回显问题
|
||||||
|
- 🎯 优化 去掉开发环境 i18n 控制台警告,页面代码:[i18n/index.ts](https://gitee.com/lyt-top/vue-next-admin/blob/master/src/i18n/index.ts)
|
||||||
|
- 🎯 优化 `NextLoading.start()` 方法,防止第一次进入界面时出现短暂空白
|
||||||
|
- 🎯 优化 地址栏有参数退出登录,再次登录不跳之前界面问题 `src/layout/navBars/breadcrumb/user.vue`
|
||||||
|
- 🎯 优化 `SvgIcon` 组件,防止 `开启 Tagsview 图标` 时,`tagsView 右键菜单关闭` 报错问题,工作流不可连线、全屏时关闭按钮消失问题
|
||||||
|
- 🎯 优化 [如果 url 中有中文等特殊字符,第一次切换该 tab 时 keep-alive 失效#I55JS7](https://gitee.com/lyt-top/vue-next-admin/issues/I55JS7),感谢[yuyong1566](https://gitee.com/yuyong1566)
|
||||||
|
- 🎯 优化 [wangEditor](https://www.wangeditor.com/) 更新到 v5,[vue3 版本线上示例中 wangeditor 富文本编辑器 demo 实例,无法换行#I5565B](https://gitee.com/lyt-top/vue-next-admin/issues/I5565B),感谢@[jenchih](https://gitee.com/jenchih)
|
||||||
|
- 🎯 优化 [在关闭 tagview 时,高度刷新时会会变化,出现滚动条](https://gitee.com/lyt-top/vue-next-admin/issues/I55FHM),感谢[张松](https://gitee.com/zs310071113)
|
||||||
|
- 🎯 优化 [路由参数](https://lyt-top.gitee.io/vue-next-admin-preview/#/params/common)演示
|
||||||
|
- 🎉 新增 [vuex](https://vuex.vuejs.org/) 替换成 [pinia](https://pinia.vuejs.org/getting-started.html)
|
||||||
|
- 🎉 新增 tagsView 支持自定义 tagsView 名称(文章详情时有用),前往体验:[路由参数/普通路由](https://lyt-top.gitee.io/vue-next-admin-preview/#/params/common)。新增 tagsView 支持自定义名称国际化,感谢[@q7but](https://gitee.com/q7but)、[!22 add 添加自定义 tagVIewName 拓展,支持国际化](https://gitee.com/lyt-top/vue-next-admin/pulls/22/files)、感谢[@tony_tong_xin](https://gitee.com/tony_tong_xin)
|
||||||
|
- 🐞 修复 适配 `"element-plus": "^2.1.9",2.2.0` 版本
|
||||||
|
- 🐞 修复 [导航栏横向布局后,一级菜单显示问题#I4Z3M3](https://gitee.com/lyt-top/vue-next-admin/issues/I4Z3M3)
|
||||||
|
- 🐞 修复 横向布局三级及以上导航菜单高亮、导航高度不统一问题
|
||||||
|
- 🐞 修复 分栏模式下,选中的菜单是 primary 样式,鼠标移入字也变成 primary 色了,感谢群友@孤夜-流殇
|
||||||
|
- 🐞 修复 [vuex 里面改了颜色 但是不生效 #I4WFMA](https://gitee.com/lyt-top/vue-next-admin/issues/I4WFMA)
|
||||||
|
- 🐞 修复 全局主题 primary 清空颜色后报错,[#I4X0LG](https://gitee.com/lyt-top/vue-next-admin/issues/I4X0LG),感谢[面向 BUG 编程](https://gitee.com/fhtfy)
|
||||||
|
- 🐞 修复 [.eslintrc.js 文件 rules 标签名错误 #I53IPK](https://gitee.com/lyt-top/vue-next-admin/issues/I53IPK),感谢[yuyong1566](https://gitee.com/yuyong1566)
|
||||||
|
- 🐞 修复 `开启 Tagsview 图标` 时,`tagsView 右键菜单关闭` 报错问题
|
||||||
|
- 🐞 修复 `router.push` 路径找不到时报错问题,`404、401 界面` 已移入到 `main` 主布局里(之前全屏)
|
||||||
|
- 🐞 修复 [全局修改组件大小失效了](https://gitee.com/lyt-top/vue-next-admin/issues/I551RP),感谢[lg_boy](https://gitee.com/lg_boy)
|
||||||
|
- 🐞 修复 [修改一下配置时,需要每次都清理 `window.localStorage` 浏览器永久缓存,配置才会生效,问题解决#I567R1](https://gitee.com/lyt-top/vue-next-admin/issues/I567R1),感谢[@lanbao123](https://gitee.com/lanbao123)
|
||||||
|
- 🐞 修复 [标记为需要缓存的 tab 页后,再次从左侧菜单打开,还是显示被缓存的页面内容#I4UY3G](https://gitee.com/lyt-top/vue-next-admin/issues/I4UY3G),感谢@axcc1234、特别感谢群友@华仔
|
||||||
|
- 🌈 重构 路由(`/src/router/index.ts`)解决 No match found for location with path "xxx"(前端控制,后端控制未解决) 问题
|
||||||
|
|
||||||
|
## 2.0.2
|
||||||
|
|
||||||
|
`2022.03.04`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 Alert 提示添加边框
|
||||||
|
- 🎯 优化 功能 / 数字滚动 演示界面
|
||||||
|
- 🐞 修复 全局主题按钮颜色 :active 问题
|
||||||
|
- 🐞 修复 Dropdown 下拉菜单样式问题
|
||||||
|
- 🐞 修复 SvgIcon 图标组件动态切换时报警告问题,[SvgIcon 改变 name 时可能导致图像不显示](https://gitee.com/lyt-top/vue-next-admin/issues/I4VGE0),感谢@axcc1234
|
||||||
|
|
||||||
|
## 2.0.1
|
||||||
|
|
||||||
|
`2022.02.25`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 svgIcon 图标组件
|
||||||
|
- 🎯 优化 vite.config.ts 打包,感谢群友@YourObjec
|
||||||
|
- 🐞 修复 tagViews 开启图标不显示问题(风格 5),感谢群友@坏人
|
||||||
|
- 🐞 修复 [Element Plus 1.2.0-beta.6 以后的版本 el-table 在移动端无法左右滑动](https://gitee.com/lyt-top/vue-next-admin/issues/I4UPTP),感谢@YGDada
|
||||||
|
|
||||||
|
## 2.0.0
|
||||||
|
|
||||||
|
`2022.02.21`
|
||||||
|
|
||||||
|
⚡⚡⚡ 此版本为破环性更新,优化内容如下:(谨慎更新!谨慎更新!!谨慎更新!!!)。演示界面建议直接覆盖文件。如需使用之前版本,请前往[gitee 发行版](https://gitee.com/lyt-top/vue-next-admin/releases) 进行对应版本下载。基础版会基于 `master` 分支进行修改
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🌟 更新 登录页、首页
|
||||||
|
- 💔 移除 vue-web-screen-shot
|
||||||
|
- 💔 移除 城市多级联动,完整 json 数据请去 [vue-next-admin-images/menu](https://gitee.com/lyt-top/vue-next-admin-images/tree/master/menu) 仓库查看
|
||||||
|
- 💔 移除 功能/echartsTree 树图
|
||||||
|
- 💔 移除 其它设置/Tagsview 风格 2、Tagsview 风格 3
|
||||||
|
- 💔 移除 功能/验证器
|
||||||
|
- 🚧 调整 src/api 编写方式
|
||||||
|
- 🚧 调整 自定义封装公用组件演示,更好的维护
|
||||||
|
- 🎉 新增 Volar 支持,vs code 配置参考 [Vue Language Features (Volar)](https://lyt-top.gitee.io/vue-next-admin-doc-preview/home/vscode/)
|
||||||
|
- 🎉 新增 `SvgIcon` 支持本地 svg 图标使用
|
||||||
|
- 🎉 新增 表单表格验证演示
|
||||||
|
- 🎯 优化 全局主题(移除 success、info、warning、danger)
|
||||||
|
- 🎯 优化 工作流(开源)
|
||||||
|
- 🎯 优化 element plus svg 图标,`elementXXX` 改成 `ele-XXX`
|
||||||
|
- 🌈 重构 深色模式
|
||||||
|
- 🌹 合并 [处理 parent 的 h100 由于外层有 min-height 导致失效的问题](https://gitee.com/lyt-top/vue-next-admin/pulls/20),感谢@MaxNull、@21030442-mao
|
||||||
|
- 🐞 修复 element plus 升级 `^1.3.0-beta.5` 后 组件 size 大小问题(大改:涉及布局、演示界面)
|
||||||
|
- 🐞 修复 vs code 使用 Vue Language Features (Volar) 插件 代码报红问题(可以把公用的 ts 类型定义封装起来公用)
|
||||||
|
|
||||||
|
## 1.2.2
|
||||||
|
|
||||||
|
`2021.12.21`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 iframes 滚动条问题
|
||||||
|
- 🎯 优化 部署后每次都要强制刷新清浏览器缓存问题
|
||||||
|
- 🎉 新增 工具类百分比验证演示
|
||||||
|
- 🐞 修复 [tag-view 标签右键会超出浏览器 #I4KN78](https://gitee.com/lyt-top/vue-next-admin/issues/I4KN78)
|
||||||
|
|
||||||
|
## 1.2.1
|
||||||
|
|
||||||
|
`2021.12.12`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 cropper 裁剪时卡顿问题 [#I4M2VQ](https://gitee.com/lyt-top/vue-next-admin/issues/I4M2VQ)
|
||||||
|
- 🎯 优化 Wangeditor 富文本编辑器的问题 [#I4LPC1](https://gitee.com/lyt-top/vue-next-admin/issues/I4LPC1)、[#I4LM7I](https://gitee.com/lyt-top/vue-next-admin/issues/I4LM7I)
|
||||||
|
- 🐞 修复 浏览器标题问题
|
||||||
|
- 🐞 修复 element plus svg 图标引入
|
||||||
|
- 🐞 修复 工作流不可以拖线连接问题
|
||||||
|
|
||||||
|
## 1.2.0
|
||||||
|
|
||||||
|
`2021.11.28`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 深色模式
|
||||||
|
- 🎯 优化 `/@/utils` 文件夹,合并删除单一内容
|
||||||
|
- 🎯 优化 系统设置:菜单管理(新增、修改)、角色管理(新增菜单权限)、用户管理、部门管理、字典管理
|
||||||
|
- 🎯 优化 登录界面逻辑、权限管理逻辑
|
||||||
|
- 🎯 优化 同步 [vue-next-admin-images](https://gitee.com/lyt-top/vue-next-admin-images/tree/master/menu) 后端控制菜单模拟数据
|
||||||
|
- 🎉 新增 适配 Font Icon 向 SVG Icon 迁移(改动大,"element-plus": "^1.2.0-beta.4" 谨慎更新)
|
||||||
|
- 🐞 修复 热更新问题,感谢@甜蜜蜜
|
||||||
|
- 🐞 修复 页面/element 字体图标演示
|
||||||
|
- 🐞 修复 功能/图标选择器演示,新增高级功能 [issues #I4GJXQ](https://gitee.com/lyt-top/vue-next-admin/issues/I4GJXQ)
|
||||||
|
|
||||||
|
## 1.1.2
|
||||||
|
|
||||||
|
`2021.10.17`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🐞 修复 开启全屏时,刷新界面被还原成未全屏的状态
|
||||||
|
- 🎯 优化 tagsView 右键菜单关闭逻辑
|
||||||
|
- 🎯 优化 wangeditor 富文本编辑器(增加双向绑定)
|
||||||
|
- 🎉 新增 工作流(暂不开源)
|
||||||
|
- 🎉 新增 基础版 ts(不带国际化),切换 `vue-next-admin-template` 分支
|
||||||
|
|
||||||
|
## 1.1.1
|
||||||
|
|
||||||
|
`2021.09.25`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本(`"element-plus": "^1.1.0-beta.13"` 版本运行错误,`^1.1.0-beta.16`修复横向菜单卡死问题)
|
||||||
|
- 🐞 修复 Dialog 弹窗位置错误、Drawer 抽屉内边距、el-menu 菜单收起时背景色问题
|
||||||
|
- 🎯 优化 锁屏界面自动锁屏(s/秒)必须设置至少 1 秒
|
||||||
|
- 🎉 新增 分栏布局,鼠标移入当前项时,显示当前项菜单内容
|
||||||
|
- 🎉 新增 工作流(未完成)
|
||||||
|
|
||||||
|
## 1.1.0
|
||||||
|
|
||||||
|
`2021.09.10`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 小屏模式下登录页二维码遮挡标题问题
|
||||||
|
- 🎉 新增 图片验证器
|
||||||
|
- 🎉 新增 动态复杂表单
|
||||||
|
- 🎉 新增 工作流(未完成)
|
||||||
|
- 🎉 新增 深色主题(伪深色,样式变动大,谨慎更新)
|
||||||
|
|
||||||
|
## 1.0.18
|
||||||
|
|
||||||
|
`2021.08.29`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 权限组件去掉顶级 div(`/src/components/auth`)
|
||||||
|
- 🎉 新增 布局配置添加恢复默认按钮
|
||||||
|
- 🐞 修复 升级 <a href="https://element-plus.gitee.io/#/zh-CN/component/changelog" target="_blank">element plus 1.1.0-beta.7</a>后项目无法启动、el-menu 菜单
|
||||||
|
- 🐞 修复 表格固定列时的层级、设置了相对定位时,遮挡左侧导航菜单问题
|
||||||
|
|
||||||
|
## 1.0.17
|
||||||
|
|
||||||
|
`2021.08.22`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 去除设置布局切换,重置主题样式(initSetLayoutChange),切换布局需手动设置样式,设置的样式自动同步各布局
|
||||||
|
- 🎯 优化 Dropdown 下拉菜单用户账号靠边时换行问题
|
||||||
|
- 🎯 优化 左侧导航菜单,共用菜单树,防止 `布局配置` 设置 `菜单 / 顶栏` 时,样式丢失等问题
|
||||||
|
- 🐞 修复 固定 header 后没有回到顶部的 bug,拉取项目后运行不起来的 bug。<a href="https://gitee.com/lyt-top/vue-next-admin/pulls/14" target="_blank">!14</a>,感谢<a href="https://gitee.com/wjs0509" target="_blank">@wjs0509</a>
|
||||||
|
- 🐞 修复 tagView 右键全屏后,浏览器窗口大小发生任何变化都会导致左边菜单显示出来,并且可点击打开对应页面。<a href="https://gitee.com/lyt-top/vue-next-admin/issues/I46E6T" target="_blank">I46E6T</a>
|
||||||
|
- 🐞 修复 默认设置 `菜单 / 顶栏` 样式不生效问题(/@/src/store/modules/themeConfig.ts)
|
||||||
|
|
||||||
|
## 1.0.16
|
||||||
|
|
||||||
|
`2021.08.14`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 菜单高亮(详情且详情设置了 meta.isHide 时,顶级菜单高亮),感谢群友@YourObject
|
||||||
|
- 🎯 优化 详情路径写法:如父级(/pages/filtering),那么详情为(/pages/filtering/details?id=1)。这样写可实现(详情时,父级菜单高亮),否则写成(/pages/filteringDetails?id=1)顶级菜单将不会高亮。可参考:`页面/过滤筛选组件`,点击当前图片进行测试
|
||||||
|
- 🎯 优化 tagsView 右键菜单全屏时,打开的界面高度问题
|
||||||
|
- 🎯 优化 图表批量 resize 问题
|
||||||
|
- 🐞 修复 菜单收起时(设置全局主题:primary 且有二级菜单时),文字高亮颜色不对
|
||||||
|
- 🐞 修复 国际化 <a href="https://gitee.com/lyt-top/vue-next-admin/issues/I43NPE" target="_blank">#I43NPE</a>。可参考:`页面/过滤筛选组件`,点击顶部语言切换,进行底部分页国际化查看
|
||||||
|
|
||||||
|
## 1.0.15
|
||||||
|
|
||||||
|
`2021.08.06`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 tagsView 右键菜单点击时的字段名(id 已修改成 contextMenuClickId)与路由中返回的 id 名冲突问题,感谢群友@伯牙已遇钟子期
|
||||||
|
- 🎉 新增 多个 form 表单验证界面演示
|
||||||
|
|
||||||
|
## 1.0.14
|
||||||
|
|
||||||
|
`2021.07.29`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本(vue、vuex、vue-router),出现问题,请手动降级。版本查看:<a href="https://www.npmjs.com/" target="_blank">vnpm</a>
|
||||||
|
- 🎯 优化 数据可视化图表演示加载卡顿问题、优化有图表的演示界面
|
||||||
|
- 🎯 优化 路由参数演示界面
|
||||||
|
- 🎯 优化 tagsView 操作演示界面,由于存在相同路由多标签,必须要传全部参数值(query 或者 params)
|
||||||
|
- 🎉 新增 开启 TagsView 共用,开启时:(多个路由菜单共用一个详情组件(参数为后点击的覆盖前面点击的),tagsView 中只会出现一个(不支持同时出现多个 tagsView 标签))。关闭时:(多个路由菜单共用一个详情组件,参数不同,会同时出现多个 tagsView 标签)
|
||||||
|
- 🐞 修复 tagsView 共用(单标签)时,右键菜单功能点击,参数不对的问题(第 2n+个参数未覆盖第一个参数值)
|
||||||
|
- 🐞 修复 多 tagsView 标签(参数不同)、单个 tagsView 标签公用(参数不同)所带来的刷新功能、横向自动滚动等问题
|
||||||
|
- 🐞 修复 处理全屏若干问题,<a href="https://gitee.com/lyt-top/vue-next-admin/pulls/12" target="_blank">pr!12</a>,感谢群友@另一个前端
|
||||||
|
|
||||||
|
## 1.0.13
|
||||||
|
|
||||||
|
`2021.07.25`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 数据可视化演示界面(/visualizingDemo1、/visualizingDemo2)
|
||||||
|
- 🎉 新增 登录页扫码登录
|
||||||
|
|
||||||
|
## 1.0.12
|
||||||
|
|
||||||
|
`2021.07.16`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 数据可视化演示空界面(待完善)
|
||||||
|
- 🎯 优化 tagsView 动态路由(xxx/:id/:name)时的右键菜单刷新、关闭其它时参数丢失问题(2021.07.15 优化)
|
||||||
|
- 🐞 修复 路由带参数时,复制路径到登录页,跳转后参数消失的问题
|
||||||
|
- 🐞 修复 设置多个外链,点击后,页面内容停留在上一个内容(内容未改变)、国际化处理、打开新窗口 sessionStorage 共享等
|
||||||
|
|
||||||
|
## 1.0.11
|
||||||
|
|
||||||
|
`2021.07.14`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 路由参数、图片懒加载界面演示
|
||||||
|
- ⚠️ 警告 Form 表单 `binding value must be a string or number`,解决:加上 `label-position="top"` 不报警告(等待官方修复)
|
||||||
|
- 🎯 优化 锁屏界面动画效果、首页图表显示
|
||||||
|
- 🎯 优化 tagsView 右键菜单 `关闭` 功能逻辑
|
||||||
|
- 🐞 修复 开启 TagsView 拖拽报错及小于 `1000px` 时自动设置禁止拖拽(<a href="https://gitee.com/lyt-top/vue-next-admin/issues/I3ZRRI" target="_blank">#I3ZRRI</a>)
|
||||||
|
- 🐞 修复 `iframe 内嵌、外链` 高度问题,使用 computed 进行计算
|
||||||
|
- 🐞 修复 默认布局开启 `侧边栏 Logo` 与关闭 `菜单水平折叠`,切换到横向布局时,菜单看不见的问题
|
||||||
|
- 🐞 修复 切换不同布局时,再去开启 `经典布局分割菜单` 功能不生效问题
|
||||||
|
- 🐞 修复 浏览器窗口标题中/英文切换不实时生效的问题
|
||||||
|
- 🐞 修复 切换布局时,某些功能不可以使用。部分界面不需要取消事件监听(proxy.mittBus.off('xxx'))
|
||||||
|
- 🐞 修复 动态路由带参数,router-link 跳转问题(<a href="hhttps://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G" target="_blank">#I3YX6G</a>)
|
||||||
|
- 🐞 修复 横向菜单有二级菜单时,点击子级菜单不高亮问题
|
||||||
|
- 🐞 修复 功能 tagsView 操作演示不生效
|
||||||
|
|
||||||
|
## 1.0.10
|
||||||
|
|
||||||
|
`2021.07.07`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本(字体图标无问题)
|
||||||
|
- 🎯 优化 内嵌 iframe、外链,解决 tagsView 刷新问题
|
||||||
|
|
||||||
|
## 1.0.9
|
||||||
|
|
||||||
|
`2021.07.02`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎯 优化 图标选择器设置宽度、v-model 等问题
|
||||||
|
- 🎯 优化 滚动通知栏在手机上的体验
|
||||||
|
- 🎯 优化 系统管理/新增菜单(编辑菜单),使用 `图标选择器` 进行模拟
|
||||||
|
- 🎯 优化 字体图标(自动载入) 逻辑
|
||||||
|
- 🐞 修复 screenfull 全屏时,按键盘 esc 键图标不改变问题,感谢群友@伯牙已遇钟子期
|
||||||
|
|
||||||
|
## 1.0.8
|
||||||
|
|
||||||
|
`2021.06.29`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 表单中英文切换演示
|
||||||
|
- 🎯 优化 登录页查看密码 icon 图标
|
||||||
|
- 🎯 优化 图标选择器
|
||||||
|
- 🎯 优化 拖动指令
|
||||||
|
- 🐞 修复 form 表单在页面小于 576px 时的排版问题
|
||||||
|
|
||||||
|
## 1.0.7
|
||||||
|
|
||||||
|
`2021.06.24`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 拖动指令及其演示界面
|
||||||
|
- 🎯 优化 锁屏界面,解锁提示
|
||||||
|
- 🎯 优化 登录页在手机上显示的效果
|
||||||
|
|
||||||
|
## 1.0.6
|
||||||
|
|
||||||
|
`2021.06.23`
|
||||||
|
|
||||||
|
- 🎯 优化 去掉内嵌 iframe 内边距(padding)
|
||||||
|
- 🎯 优化 城市多级联动组件
|
||||||
|
- 🎯 优化 Tree 树形控件改成表格组件
|
||||||
|
- 🐞 修复 Cascader 级联选择器高度问题
|
||||||
|
|
||||||
|
## 1.0.5
|
||||||
|
|
||||||
|
`2021.06.22`
|
||||||
|
|
||||||
|
- 🌟 更新 vite 降级为@vite2.3.7,降级方法 `cnpm install vite@2.3.7`,防止 element plus 字体图标消失
|
||||||
|
- 🐞 修复 开启后端控制路由(isRequestRoutes = true)时,内嵌 iframe、外链不可使用的问题
|
||||||
|
|
||||||
|
## 1.0.4
|
||||||
|
|
||||||
|
`2021.06.19`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本("vite": "^2.3.7")热更新无问题
|
||||||
|
- 🎉 新增 深克隆工具,方便开发,感谢<a href="https://gitee.com/kangert" target="_blank">@kangert</a>(<a href="https://gitee.com/lyt-top/vue-next-admin/pulls/6" target="_blank">#6</a>)
|
||||||
|
- 🎯 优化 vuex 模块自动导入。感谢<a href="https://gitee.com/kangert" target="_blank">@kangert</a>(<a href="https://gitee.com/lyt-top/vue-next-admin/pulls/4" target="_blank">#4</a>),感谢群友@web 小学生-第五君
|
||||||
|
- 🎯 优化 类型定义提高编码体验,修复不能将类型“string | undefined”分配给类型“string”的问题。感谢<a href="https://gitee.com/kangert" target="_blank">@kangert</a>(<a href="https://gitee.com/lyt-top/vue-next-admin/pulls/5" target="_blank">#5</a>)
|
||||||
|
- 🎯 优化 `layout` 文件夹移动到与 `views` 文件夹同级(改动较大,`/@/views/layout` 变成 `/@/layout`)
|
||||||
|
- 🎯 优化 页面有 `console.log` 时 `eslint` 不生效问题
|
||||||
|
- 🎯 优化 页面、ts 中 `any` 类型问题(改动较大)
|
||||||
|
- 🎯 优化 登录页在手机上显示的效果
|
||||||
|
- 🎯 优化 多行注释信息,鼠标放到方法名即可查看,更加直观的知道方法参数等。引入方法时需去掉以 `.ts` 结尾的后缀(改动较大)
|
||||||
|
- 🎯 优化 移除 `utils/storage.ts` 下的旧写法(改动较大)
|
||||||
|
- 🎯 优化 拆分 `router` 下内容,路由、前端、后端控制分开写,方便理解
|
||||||
|
- 🐞 修复 鼠标移入顶部用户信息栏 `开/关全屏` 文字反向问题
|
||||||
|
- 🐞 修复 热更新时,NextLoading(界面 loading) 不消失问题 `window.nextLoading === undefined`
|
||||||
|
- 🐞 修复 vuex 中不可以使用 `/@/api/xxx` 下的接口调用问题
|
||||||
|
|
||||||
|
## 1.0.3
|
||||||
|
|
||||||
|
`2021.06.02`
|
||||||
|
|
||||||
|
- ❄️ 删除 G6 思维导图界面
|
||||||
|
- 🌟 更新 手动更新 vue、vue-router、vuex 到最近最多人使用的版本,出现不可预测的问题请降低版本。版本查看:<a href="https://www.npmjs.com/package/vue" target="_blank">vue 版本查看</a>
|
||||||
|
- 🐞 修复 开启后端控制路由 `isRequestRoutes` 在非首页刷新页面后,回到首页的问题,感谢群友@伯牙已遇钟子期
|
||||||
|
|
||||||
|
## 1.0.2
|
||||||
|
|
||||||
|
`2021.06.01`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🐞 修复 菜单搜索中文不可以搜索的问题,感谢群友@逍遥天意
|
||||||
|
|
||||||
|
## 1.0.1
|
||||||
|
|
||||||
|
`2021.05.31`
|
||||||
|
|
||||||
|
- 🎉 新增 更新日志文件 `CHANGELOG.md`,以后每次更新都会在这里显示对应内容
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🐞 修复 分栏、经典布局路由设置 `meta.isHide` 为 `true` 时报错问题,感谢群友@29、@芭芭拉
|
||||||
|
- 🐞 修复 经典布局点击 `tagsView` 左侧菜单数据不变问题
|
||||||
21
web/LICENSE
Normal file
21
web/LICENSE
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2021 lyt-Top
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
145
web/README.md
Normal file
145
web/README.md
Normal file
@@ -0,0 +1,145 @@
|
|||||||
|
<div align="center">
|
||||||
|
<img src="https://img-blog.csdnimg.cn/9efd5420327a46b7bd6d93524a97229d.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAbHl0LXRvcA==,size_14,color_FFFFFF,t_70,g_se,x_16">
|
||||||
|
<p align="center">
|
||||||
|
<a href="https://v3.vuejs.org/" target="_blank">
|
||||||
|
<img src="https://img.shields.io/badge/vue.js-vue3.x-green" alt="vue">
|
||||||
|
</a>
|
||||||
|
<a href="https://element-plus.gitee.io/#/zh-CN/component/changelog" target="_blank">
|
||||||
|
<img src="https://img.shields.io/badge/element--plus-%3E1.0.0-blue" alt="element plus">
|
||||||
|
</a>
|
||||||
|
<a href="https://www.tslang.cn/" target="_blank">
|
||||||
|
<img src="https://img.shields.io/badge/typescript-%3E4.0.0-blue" alt="typescript">
|
||||||
|
</a>
|
||||||
|
<a href="https://vitejs.dev/" target="_blank">
|
||||||
|
<img src="https://img.shields.io/badge/vite-%3E2.0.0-yellow" alt="vite">
|
||||||
|
</a>
|
||||||
|
<a href="https://gitee.com/lyt-top/vue-next-admin/blob/master/LICENSE" target="_blank">
|
||||||
|
<img src="https://img.shields.io/badge/license-MIT-success" alt="license">
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
<p> </p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
#### 🌈 介绍
|
||||||
|
|
||||||
|
基于 vue3.x + CompositionAPI + typescript + vite + element plus + vue-router-next + next.vuex,适配手机、平板、pc 的后台开源免费模板,希望减少工作量,帮助大家实现快速开发。
|
||||||
|
|
||||||
|
#### ⛱️ 线上预览
|
||||||
|
|
||||||
|
- vue3.x 版本预览(vue-next-admin)<a href="https://lyt-top.gitee.io/vue-next-admin-preview/#/login" target="_blank">https://lyt-top.gitee.io/vue-next-admin-preview/#/login</a>
|
||||||
|
- vue2.x 版本预览(vue-prev-admin)<a href="https://lyt-top.gitee.io/vue-prev-admin-preview/#/login" target="_blank">https://lyt-top.gitee.io/vue-prev-admin-preview/#/login</a>
|
||||||
|
|
||||||
|
#### 💒 代码仓库
|
||||||
|
|
||||||
|
- vue3.x 版本 <a href="https://gitee.com/lyt-top/vue-next-admin" target="_blank">https://gitee.com/lyt-top/vue-next-admin</a>
|
||||||
|
- vue2.x 版本 <a href="https://gitee.com/lyt-top/vue-next-admin/tree/vue-prev-admin" target="_blank">https://gitee.com/lyt-top/vue-next-admin/tree/vue-prev-admin</a>
|
||||||
|
|
||||||
|
#### 🚧 安装 cnpm、yarn
|
||||||
|
|
||||||
|
- 复制代码(桌面 cmd 运行) `npm install -g cnpm --registry=https://registry.npm.taobao.org`
|
||||||
|
- 复制代码(桌面 cmd 运行) `npm install -g yarn`
|
||||||
|
|
||||||
|
#### 🏭 环境支持
|
||||||
|
|
||||||
|
| Edge | Firefox | Chrome | Safari |
|
||||||
|
| --------- | ------------ | ----------- | ----------- |
|
||||||
|
| Edge ≥ 79 | Firefox ≥ 78 | Chrome ≥ 64 | Safari ≥ 12 |
|
||||||
|
|
||||||
|
> 由于 Vue3 不再支持 IE11,故而 ElementPlus 也不支持 IE11 及之前版本。
|
||||||
|
|
||||||
|
#### ⚡ 使用说明
|
||||||
|
|
||||||
|
建议使用 cnpm,因为 yarn 有时会报错。<a href="http://nodejs.cn/" target="_blank">node 版本 > 12xx.xx.x</a>
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# 克隆项目
|
||||||
|
git clone https://gitee.com/lyt-top/vue-next-admin.git
|
||||||
|
|
||||||
|
# 进入项目
|
||||||
|
cd vue-next-admin
|
||||||
|
|
||||||
|
# 安装依赖
|
||||||
|
cnpm install
|
||||||
|
|
||||||
|
# 运行项目
|
||||||
|
cnpm run dev
|
||||||
|
|
||||||
|
# 打包发布
|
||||||
|
cnpm run build
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 💯 学习交流加 QQ 群
|
||||||
|
|
||||||
|
- 若加群了没同意(一般秒过),那就是群满了(500 人群),请换一个群试试。群会定期清理半年(6 个月)未发言的群友,资源有限,请谅解。建议勿加多群,可能会误伤!
|
||||||
|
- 查看开发文档:<a href="https://lyt-top.gitee.io/vue-next-admin-doc-preview" target="_blank">vue-next-admin-doc</a>
|
||||||
|
- 群号码:
|
||||||
|
1 群:<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=RdUY97Vx0T0vZ_1OOu-X1yFNkWgDwbjC&jump_from=webapi">665452019</a>
|
||||||
|
2 群:<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=zVfy3gNy7pNWVK3kMduDzwU369PZg2fw&jump_from=webapi">766356862</a>
|
||||||
|
3 群:<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=02EWb5P2JkP-8iwzaDadgFdxA0HSHPpn&jump_from=webapi">795345435</a>
|
||||||
|
4 群:<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=0gTFO04WwkeZZ6R4lju6gucbeXHK-wNd&jump_from=webapi">736626228</a>
|
||||||
|
|
||||||
|
<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=RdUY97Vx0T0vZ_1OOu-X1yFNkWgDwbjC&jump_from=webapi">
|
||||||
|
<img src="https://img-blog.csdnimg.cn/35e00f12a3fe4820892ec630ca72f15f.png" width="220" height="220" alt="vue-next-admin 讨论群1" title="vue-next-admin 讨论群1"/>
|
||||||
|
</a>
|
||||||
|
<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=zVfy3gNy7pNWVK3kMduDzwU369PZg2fw&jump_from=webapi">
|
||||||
|
<img src="https://img-blog.csdnimg.cn/5f1b548abd9f434eb41edde31d1c1fa9.png" width="220" height="220" alt="vue-next-admin 讨论群2" title="vue-next-admin 讨论群2"/>
|
||||||
|
</a>
|
||||||
|
<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=02EWb5P2JkP-8iwzaDadgFdxA0HSHPpn&jump_from=webapi">
|
||||||
|
<img src="https://img-blog.csdnimg.cn/70c8a012dd304246bddeac2184c4ab3a.png" width="220" height="220" alt="vue-next-admin 讨论群3" title="vue-next-admin 讨论群3"/>
|
||||||
|
</a>
|
||||||
|
<a target="_blank" href="https://qm.qq.com/cgi-bin/qm/qr?k=0gTFO04WwkeZZ6R4lju6gucbeXHK-wNd&jump_from=webapi">
|
||||||
|
<img src="https://img-blog.csdnimg.cn/e5c9704eed1342bc9d9e74b37203402d.png" width="220" height="220" alt="vue-next-admin 讨论群4" title="vue-next-admin 讨论群4"/>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
#### 💒 集成后端
|
||||||
|
|
||||||
|
- <a target="_blank" href="https://github.com/PandaGoAdmin/PandaX">@熊猫 PandaGoAdmin</a>
|
||||||
|
- <a target="_blank" href="https://toscode.gitee.com/GionConnection/gopro_free">@甜蜜蜜 GoPro 平台</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/GionConnection/niupi-free">@甜蜜蜜 NiuPi 平台</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/tiger1103/gfast/tree/os-v3/">@游子 GFast-V3</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/diygw/diygw-ui-php/">@diygw.com gw-ui-php</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/zsvg/vboot-net">@zsvg vboot-net</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/zsvg/vboot-java">@zsvg vboot-java</a>
|
||||||
|
- <a target="_blank" href="https://gitee.com/wonderful-code/buildadmin">@青红造了个白 buildadmin</a>
|
||||||
|
- <a target="_blank" href="https://github.com/xiaodingding/iotfast">@Goodwell iotfast(一个开源的物联网平台)</a>
|
||||||
|
|
||||||
|
#### ❤️ 鸣谢列表
|
||||||
|
|
||||||
|
- <a href="https://github.com/vuejs/vue" target="_blank">vue</a>
|
||||||
|
- <a href="https://github.com/vuejs/vue-next" target="_blank">vue-next</a>
|
||||||
|
- <a href="https://github.com/ElemeFE/element" target="_blank">element-ui</a>
|
||||||
|
- <a href="https://github.com/element-plus/element-plus" target="_blank">element-plus</a>
|
||||||
|
- <a href="https://github.com/vuejs/vue-router-next" target="_blank">vue-router-next</a>
|
||||||
|
- <a href="https://github.com/vuejs/vuex" target="_blank">vuex</a>
|
||||||
|
- <a href="https://github.com/apache/echarts" target="_blank">echarts</a>
|
||||||
|
- <a href="https://github.com/axios/axios" target="_blank">axios</a>
|
||||||
|
- <a href="https://github.com/zenorocha/clipboard.js" target="_blank">clipboard</a>
|
||||||
|
- <a href="https://github.com/inorganik/countUp.js" target="_blank">countUp</a>
|
||||||
|
- <a href="https://github.com/developit/mitt" target="_blank">mitt</a>
|
||||||
|
- <a href="https://github.com/rstacruz/nprogress" target="_blank">nprogress</a>
|
||||||
|
- <a href="https://github.com/sindresorhus/screenfull.js" target="_blank">screenfull</a>
|
||||||
|
- <a href="https://github.com/SortableJS/Sortable" target="_blank">sortablejs</a>
|
||||||
|
- <a href="https://github.com/sass/sass" target="_blank">sass</a>
|
||||||
|
- <a href="https://github.com/microsoft/TypeScript" target="_blank">typescript</a>
|
||||||
|
- <a href="https://github.com/vitejs/vite" target="_blank">vite</a>
|
||||||
|
- <a href="https://github.com/wangeditor-team/wangEditor" target="_blank">wangeditor</a>
|
||||||
|
- <a href="https://github.com/fengyuanchen/cropperjs" target="_blank">cropperjs</a>
|
||||||
|
- <a href="https://github.com/davidshimjs/qrcodejs" target="_blank">qrcodejs</a>
|
||||||
|
- <a href="https://github.com/crabbly/Print.js" target="_blank">print-js</a>
|
||||||
|
- <a href="https://github.com/jbaysolutions/vue-grid-layout" target="_blank">vue-grid-layout</a>
|
||||||
|
- <a href="https://github.com/antoniandre/splitpanes" target="_blank">splitpanes</a>
|
||||||
|
- <a href="https://github.com/jsplumb/jsplumb" target="_blank">jsplumb</a>
|
||||||
|
|
||||||
|
#### 💕 特别感谢
|
||||||
|
|
||||||
|
特别感谢老哥们的建议、指导与帮忙。谢谢!
|
||||||
|
|
||||||
|
- <a href="https://gitee.com/click33/sa-plus" target="_blank">@省长</a>
|
||||||
|
- <a href="https://gitee.com/jskz/Jskz-SpringCloud" target="_blank">@唐参</a>
|
||||||
|
- <a href="https://gitee.com/chuange" target="_blank">@川歌</a>
|
||||||
|
- @华仔
|
||||||
|
|
||||||
|
#### 💌 支持作者
|
||||||
|
|
||||||
|
如果觉得框架不错,或者已经在使用了,希望你可以去 <a target="_blank" href="https://github.com/lyt-Top/vue-next-admin">Github</a> 或者
|
||||||
|
<a target="_blank" href="https://gitee.com/lyt-top/vue-next-admin">Gitee</a> 帮我点个 ⭐ Star,这将是对我极大的鼓励与支持。
|
||||||
32
web/index.html
Normal file
32
web/index.html
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="zh-CN">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<meta
|
||||||
|
name="keywords"
|
||||||
|
content="vue-next-admin,vue-prev-admin,vue-admin-wonderful,后台管理系统一站式平台模板,希望可以帮你完成快速开发。vue2.x,vue2.0,vue2,vue3,vue3.x,vue3.0,CompositionAPI,typescript,element plus,element,plus,admin,wonderful,wonderful-next,vue-next-admin,vite,vite-admin,快速,高效,后台模板,后台系统,管理系统"
|
||||||
|
/>
|
||||||
|
<meta
|
||||||
|
name="description"
|
||||||
|
content="vue-next-admin,基于 vue3 + CompositionAPI + typescript + vite + element plus,适配手机、平板、pc 的后台开源免费管理系统模板!vue-prev-admin,基于 vue2 + element ui,适配手机、平板、pc 的后台开源免费管理系统模板!"
|
||||||
|
/>
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
<title>vue-next-admin</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="app"></div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
var _hmt = _hmt || [];
|
||||||
|
(function () {
|
||||||
|
var hm = document.createElement('script');
|
||||||
|
hm.src = 'https://hm.baidu.com/hm.js?d9c8b87d10717013641458b300c552e4';
|
||||||
|
var s = document.getElementsByTagName('script')[0];
|
||||||
|
s.parentNode.insertBefore(hm, s);
|
||||||
|
})();
|
||||||
|
</script>
|
||||||
|
<script type="module" src="/src/main.ts"></script>
|
||||||
|
<script type="text/javascript" src="https://api.map.baidu.com/api?v=3.0&ak=wsijQt8sLXrCW71YesmispvYHitfG9gv&s=1"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
7770
web/package-lock.json
generated
Normal file
7770
web/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
83
web/package.json
Normal file
83
web/package.json
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
{
|
||||||
|
"name": "vue-next-admin",
|
||||||
|
"version": "2.2.0",
|
||||||
|
"description": "vue3 vite next admin template",
|
||||||
|
"author": "lyt_20201208",
|
||||||
|
"license": "MIT",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite --force",
|
||||||
|
"build": "vite build",
|
||||||
|
"lint-fix": "eslint --fix --ext .js --ext .jsx --ext .vue src/"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@element-plus/icons-vue": "^2.0.6",
|
||||||
|
"@fast-crud/fast-crud": "^1.4.2",
|
||||||
|
"@fast-crud/ui-element": "^1.4.2",
|
||||||
|
"@wangeditor/editor": "^5.1.11",
|
||||||
|
"axios": "^0.27.2",
|
||||||
|
"countup.js": "^2.3.2",
|
||||||
|
"cropperjs": "^1.5.12",
|
||||||
|
"echarts": "^5.3.3",
|
||||||
|
"echarts-gl": "^2.0.9",
|
||||||
|
"echarts-wordcloud": "^2.0.0",
|
||||||
|
"element-plus": "^2.2.9",
|
||||||
|
"js-cookie": "^3.0.1",
|
||||||
|
"jsplumb": "^2.15.6",
|
||||||
|
"mitt": "^3.0.0",
|
||||||
|
"nprogress": "^0.2.0",
|
||||||
|
"pinia": "^2.0.16",
|
||||||
|
"print-js": "^1.6.0",
|
||||||
|
"qrcodejs2-fixes": "^0.0.2",
|
||||||
|
"screenfull": "^6.0.2",
|
||||||
|
"sortablejs": "^1.15.0",
|
||||||
|
"splitpanes": "^3.1.1",
|
||||||
|
"vue": "^3.2.37",
|
||||||
|
"vue-clipboard3": "^2.0.0",
|
||||||
|
"vue-grid-layout": "^3.0.0-beta1",
|
||||||
|
"vue-i18n": "^9.1.10",
|
||||||
|
"vue-router": "^4.1.2"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/node": "^18.0.6",
|
||||||
|
"@types/nprogress": "^0.2.0",
|
||||||
|
"@types/sortablejs": "^1.13.0",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^5.30.7",
|
||||||
|
"@typescript-eslint/parser": "^5.30.7",
|
||||||
|
"@vitejs/plugin-vue": "^2.3.3",
|
||||||
|
"@vue/compiler-sfc": "^3.2.37",
|
||||||
|
"dotenv": "^16.0.1",
|
||||||
|
"eslint": "^8.20.0",
|
||||||
|
"eslint-plugin-vue": "^9.2.0",
|
||||||
|
"prettier": "^2.7.1",
|
||||||
|
"sass": "^1.53.0",
|
||||||
|
"sass-loader": "^13.0.2",
|
||||||
|
"typescript": "^4.7.4",
|
||||||
|
"vite": "^2.9.14",
|
||||||
|
"vue-eslint-parser": "^9.0.3"
|
||||||
|
},
|
||||||
|
"browserslist": [
|
||||||
|
"> 1%",
|
||||||
|
"last 2 versions",
|
||||||
|
"not dead"
|
||||||
|
],
|
||||||
|
"bugs": {
|
||||||
|
"url": "https://gitee.com/lyt-top/vue-next-admin/issues"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12.0.0",
|
||||||
|
"npm": ">= 6.0.0"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"vue",
|
||||||
|
"vue3",
|
||||||
|
"vuejs/vue-next",
|
||||||
|
"element-ui",
|
||||||
|
"element-plus",
|
||||||
|
"vue-next-admin",
|
||||||
|
"next-admin"
|
||||||
|
],
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://gitee.com/lyt-top/vue-next-admin.git"
|
||||||
|
}
|
||||||
|
}
|
||||||
4
web/plugins.d.ts
vendored
Normal file
4
web/plugins.d.ts
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
declare module 'vue-grid-layout';
|
||||||
|
declare module 'qrcodejs2-fixes';
|
||||||
|
declare module 'splitpanes';
|
||||||
|
declare module 'js-cookie';
|
||||||
BIN
web/public/favicon.ico
Normal file
BIN
web/public/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.1 KiB |
13
web/shim.d.ts
vendored
Normal file
13
web/shim.d.ts
vendored
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
/* eslint-disable */
|
||||||
|
|
||||||
|
// 声明文件,*.vue 后缀的文件交给 vue 模块来处理
|
||||||
|
declare module '*.vue' {
|
||||||
|
import type { DefineComponent } from 'vue';
|
||||||
|
const component: DefineComponent<{}, {}, any>;
|
||||||
|
export default component;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 声明文件,定义全局变量。其它 app.config.globalProperties.xxx,使用 getCurrentInstance() 来获取
|
||||||
|
interface Window {
|
||||||
|
nextLoading: boolean;
|
||||||
|
}
|
||||||
6
web/source.d.ts
vendored
Normal file
6
web/source.d.ts
vendored
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
declare module '*.json';
|
||||||
|
declare module '*.png';
|
||||||
|
declare module '*.jpg';
|
||||||
|
declare module '*.scss';
|
||||||
|
declare module '*.ts';
|
||||||
|
declare module '*.js';
|
||||||
96
web/src/App.vue
Normal file
96
web/src/App.vue
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
<template>
|
||||||
|
<el-config-provider :size="getGlobalComponentSize" :locale="i18nLocale">
|
||||||
|
<router-view v-show="themeConfig.lockScreenTime > 1" />
|
||||||
|
<LockScreen v-if="themeConfig.isLockScreen" />
|
||||||
|
<Setings ref="setingsRef" v-show="themeConfig.lockScreenTime > 1" />
|
||||||
|
<CloseFull v-if="!themeConfig.isLockScreen" />
|
||||||
|
</el-config-provider>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, ref, getCurrentInstance, onBeforeMount, onMounted, onUnmounted, nextTick, defineComponent, watch, reactive, toRefs } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import { Local, Session } from '/@/utils/storage';
|
||||||
|
import setIntroduction from '/@/utils/setIconfont';
|
||||||
|
import LockScreen from '/@/layout/lockScreen/index.vue';
|
||||||
|
import Setings from '/@/layout/navBars/breadcrumb/setings.vue';
|
||||||
|
import CloseFull from '/@/layout/navBars/breadcrumb/closeFull.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'app',
|
||||||
|
components: { LockScreen, Setings, CloseFull },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const setingsRef = ref();
|
||||||
|
const route = useRoute();
|
||||||
|
const stores = useTagsViewRoutes();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const state = reactive({
|
||||||
|
i18nLocale: null,
|
||||||
|
});
|
||||||
|
// 获取全局组件大小
|
||||||
|
const getGlobalComponentSize = computed(() => {
|
||||||
|
return other.globalComponentSize();
|
||||||
|
});
|
||||||
|
// 布局配置弹窗打开
|
||||||
|
const openSetingsDrawer = () => {
|
||||||
|
setingsRef.value.openDrawer();
|
||||||
|
};
|
||||||
|
// 设置初始化,防止刷新时恢复默认
|
||||||
|
onBeforeMount(() => {
|
||||||
|
// 设置批量第三方 icon 图标
|
||||||
|
setIntroduction.cssCdn();
|
||||||
|
// 设置批量第三方 js
|
||||||
|
setIntroduction.jsCdn();
|
||||||
|
});
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
nextTick(() => {
|
||||||
|
// 监听布局配置弹窗点击打开
|
||||||
|
proxy.mittBus.on('openSetingsDrawer', () => {
|
||||||
|
openSetingsDrawer();
|
||||||
|
});
|
||||||
|
// 设置 i18n,App.vue 中的 el-config-provider
|
||||||
|
proxy.mittBus.on('getI18nConfig', (locale: string) => {
|
||||||
|
(state.i18nLocale as string | null) = locale;
|
||||||
|
});
|
||||||
|
// 获取缓存中的布局配置
|
||||||
|
if (Local.get('themeConfig')) {
|
||||||
|
storesThemeConfig.setThemeConfig(Local.get('themeConfig'));
|
||||||
|
document.documentElement.style.cssText = Local.get('themeConfigStyle');
|
||||||
|
}
|
||||||
|
// 获取缓存中的全屏配置
|
||||||
|
if (Session.get('isTagsViewCurrenFull')) {
|
||||||
|
stores.setCurrenFullscreen(Session.get('isTagsViewCurrenFull'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面销毁时,关闭监听布局配置/i18n监听
|
||||||
|
onUnmounted(() => {
|
||||||
|
proxy.mittBus.off('openSetingsDrawer', () => {});
|
||||||
|
proxy.mittBus.off('getI18nConfig', () => {});
|
||||||
|
});
|
||||||
|
// 监听路由的变化,设置网站标题
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
other.useTitle();
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
themeConfig,
|
||||||
|
setingsRef,
|
||||||
|
getGlobalComponentSize,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
25
web/src/api/login/index.ts
Normal file
25
web/src/api/login/index.ts
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import request from '/@/utils/request';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 登录api接口集合
|
||||||
|
* @method signIn 用户登录
|
||||||
|
* @method signOut 用户退出登录
|
||||||
|
*/
|
||||||
|
export function useLoginApi() {
|
||||||
|
return {
|
||||||
|
signIn: (params: object) => {
|
||||||
|
return request({
|
||||||
|
url: '/user/signIn',
|
||||||
|
method: 'post',
|
||||||
|
data: params,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
signOut: (params: object) => {
|
||||||
|
return request({
|
||||||
|
url: '/user/signOut',
|
||||||
|
method: 'post',
|
||||||
|
data: params,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
26
web/src/api/menu/index.ts
Normal file
26
web/src/api/menu/index.ts
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
import request from '/@/utils/request';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端控制菜单模拟json,路径在 https://gitee.com/lyt-top/vue-next-admin-images/tree/master/menu
|
||||||
|
* 后端控制路由,isRequestRoutes 为 true,则开启后端控制路由
|
||||||
|
* @method getMenuAdmin 获取后端动态路由菜单(admin)
|
||||||
|
* @method getMenuTest 获取后端动态路由菜单(test)
|
||||||
|
*/
|
||||||
|
export function useMenuApi() {
|
||||||
|
return {
|
||||||
|
getMenuAdmin: (params?: object) => {
|
||||||
|
return request({
|
||||||
|
url: '/gitee/lyt-top/vue-next-admin-images/raw/master/menu/adminMenu.json',
|
||||||
|
method: 'get',
|
||||||
|
params,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
getMenuTest: (params?: object) => {
|
||||||
|
return request({
|
||||||
|
url: '/gitee/lyt-top/vue-next-admin-images/raw/master/menu/testMenu.json',
|
||||||
|
method: 'get',
|
||||||
|
params,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
1
web/src/assets/login-icon-two.svg
Normal file
1
web/src/assets/login-icon-two.svg
Normal file
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 26 KiB |
9
web/src/assets/logo-mini.svg
Normal file
9
web/src/assets/logo-mini.svg
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 64 50" width="64" height="50">
|
||||||
|
<defs>
|
||||||
|
<image width="64" height="50" id="img1" href=""/>
|
||||||
|
</defs>
|
||||||
|
<style>
|
||||||
|
tspan { white-space:pre }
|
||||||
|
</style>
|
||||||
|
<use id="Background" href="#img1" x="0" y="0" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.2 KiB |
30
web/src/components/auth/auth.vue
Normal file
30
web/src/components/auth/auth.vue
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<template>
|
||||||
|
<slot v-if="getUserAuthBtnList" />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'auth',
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
const stores = useUserInfo();
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
// 获取 vuex 中的用户权限
|
||||||
|
const getUserAuthBtnList = computed(() => {
|
||||||
|
return userInfos.value.authBtnList.some((v: string) => v === props.value);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
getUserAuthBtnList,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
31
web/src/components/auth/authAll.vue
Normal file
31
web/src/components/auth/authAll.vue
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
<template>
|
||||||
|
<slot v-if="getUserAuthBtnList" />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { judementSameArr } from '/@/utils/arrayOperation';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'authAll',
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
const stores = useUserInfo();
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
// 获取 pinia 中的用户权限
|
||||||
|
const getUserAuthBtnList = computed(() => {
|
||||||
|
return judementSameArr(props.value, userInfos.value.authBtnList);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
getUserAuthBtnList,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
36
web/src/components/auth/auths.vue
Normal file
36
web/src/components/auth/auths.vue
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<template>
|
||||||
|
<slot v-if="getUserAuthBtnList" />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'auths',
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
const stores = useUserInfo();
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
// 获取 vuex 中的用户权限
|
||||||
|
const getUserAuthBtnList = computed(() => {
|
||||||
|
let flag = false;
|
||||||
|
userInfos.value.authBtnList.map((val: string) => {
|
||||||
|
props.value.map((v) => {
|
||||||
|
if (val === v) flag = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return flag;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
getUserAuthBtnList,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
149
web/src/components/cropper/index.vue
Normal file
149
web/src/components/cropper/index.vue
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<el-dialog title="更换头像" v-model="isShowDialog" width="769px">
|
||||||
|
<div class="cropper-warp">
|
||||||
|
<div class="cropper-warp-left">
|
||||||
|
<img :src="cropperImg" class="cropper-warp-left-img" />
|
||||||
|
</div>
|
||||||
|
<div class="cropper-warp-right">
|
||||||
|
<div class="cropper-warp-right-title">预览</div>
|
||||||
|
<div class="cropper-warp-right-item">
|
||||||
|
<div class="cropper-warp-right-value">
|
||||||
|
<img :src="cropperImgBase64" class="cropper-warp-right-value-img" />
|
||||||
|
</div>
|
||||||
|
<div class="cropper-warp-right-label">100 x 100</div>
|
||||||
|
</div>
|
||||||
|
<div class="cropper-warp-right-item">
|
||||||
|
<div class="cropper-warp-right-value">
|
||||||
|
<img :src="cropperImgBase64" class="cropper-warp-right-value-img cropper-size" />
|
||||||
|
</div>
|
||||||
|
<div class="cropper-warp-right-label">50 x 50</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<template #footer>
|
||||||
|
<span class="dialog-footer">
|
||||||
|
<el-button @click="onCancel" size="default">取 消</el-button>
|
||||||
|
<el-button type="primary" @click="onSubmit" size="default">更 换</el-button>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { reactive, toRefs, nextTick, defineComponent } from 'vue';
|
||||||
|
import Cropper from 'cropperjs';
|
||||||
|
import 'cropperjs/dist/cropper.css';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'cropperIndex',
|
||||||
|
setup() {
|
||||||
|
const state = reactive({
|
||||||
|
isShowDialog: false,
|
||||||
|
cropperImg: '',
|
||||||
|
cropperImgBase64: '',
|
||||||
|
cropper: null,
|
||||||
|
});
|
||||||
|
// 打开弹窗
|
||||||
|
const openDialog = (imgs: any) => {
|
||||||
|
state.cropperImg = imgs;
|
||||||
|
state.isShowDialog = true;
|
||||||
|
nextTick(() => {
|
||||||
|
initCropper();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 关闭弹窗
|
||||||
|
const closeDialog = () => {
|
||||||
|
state.isShowDialog = false;
|
||||||
|
};
|
||||||
|
// 取消
|
||||||
|
const onCancel = () => {
|
||||||
|
closeDialog();
|
||||||
|
};
|
||||||
|
// 更换
|
||||||
|
const onSubmit = () => {
|
||||||
|
// state.cropperImgBase64 = state.cropper.getCroppedCanvas().toDataURL('image/jpeg');
|
||||||
|
};
|
||||||
|
// 初始化cropperjs图片裁剪
|
||||||
|
const initCropper = () => {
|
||||||
|
const letImg: any = document.querySelector('.cropper-warp-left-img');
|
||||||
|
(<any>state.cropper) = new Cropper(letImg, {
|
||||||
|
viewMode: 1,
|
||||||
|
dragMode: 'none',
|
||||||
|
initialAspectRatio: 1,
|
||||||
|
aspectRatio: 1,
|
||||||
|
preview: '.before',
|
||||||
|
background: false,
|
||||||
|
autoCropArea: 0.6,
|
||||||
|
zoomOnWheel: false,
|
||||||
|
crop: () => {
|
||||||
|
state.cropperImgBase64 = (<any>state.cropper).getCroppedCanvas().toDataURL('image/jpeg');
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
openDialog,
|
||||||
|
closeDialog,
|
||||||
|
onCancel,
|
||||||
|
onSubmit,
|
||||||
|
initCropper,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.cropper-warp {
|
||||||
|
display: flex;
|
||||||
|
.cropper-warp-left {
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
height: 350px;
|
||||||
|
flex: 1;
|
||||||
|
border: 1px solid var(--el-border-color);
|
||||||
|
background: var(--el-color-white);
|
||||||
|
overflow: hidden;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
cursor: move;
|
||||||
|
border-radius: var(--el-border-radius-base);
|
||||||
|
.cropper-warp-left-img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cropper-warp-right {
|
||||||
|
width: 150px;
|
||||||
|
height: 350px;
|
||||||
|
.cropper-warp-right-title {
|
||||||
|
text-align: center;
|
||||||
|
height: 20px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
.cropper-warp-right-item {
|
||||||
|
margin: 15px 0;
|
||||||
|
.cropper-warp-right-value {
|
||||||
|
display: flex;
|
||||||
|
.cropper-warp-right-value-img {
|
||||||
|
width: 100px;
|
||||||
|
height: 100px;
|
||||||
|
border-radius: var(--el-border-radius-circle);
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
.cropper-size {
|
||||||
|
width: 50px;
|
||||||
|
height: 50px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.cropper-warp-right-label {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
height: 30px;
|
||||||
|
line-height: 30px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
115
web/src/components/editor/index.vue
Normal file
115
web/src/components/editor/index.vue
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
<template>
|
||||||
|
<div class="editor-container">
|
||||||
|
<div ref="editorToolbar"></div>
|
||||||
|
<div ref="editorContent" :style="{ height }"></div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, onMounted, watch, defineComponent } from 'vue';
|
||||||
|
import { createEditor, createToolbar, IEditorConfig, IToolbarConfig, IDomEditor } from '@wangeditor/editor';
|
||||||
|
import '@wangeditor/editor/dist/css/style.css';
|
||||||
|
import { toolbarKeys } from './toolbar';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface WangeditorState {
|
||||||
|
editorToolbar: HTMLDivElement | null;
|
||||||
|
editorContent: HTMLDivElement | null;
|
||||||
|
editor: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'wngEditor',
|
||||||
|
props: {
|
||||||
|
// 节点 id
|
||||||
|
id: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'wangeditor',
|
||||||
|
},
|
||||||
|
// 是否禁用
|
||||||
|
isDisable: {
|
||||||
|
type: Boolean,
|
||||||
|
default: () => false,
|
||||||
|
},
|
||||||
|
// 内容框默认 placeholder
|
||||||
|
placeholder: {
|
||||||
|
type: String,
|
||||||
|
default: () => '请输入内容',
|
||||||
|
},
|
||||||
|
// 双向绑定:双向绑定值,字段名为固定,改了之后将不生效
|
||||||
|
// 参考:https://v3.cn.vuejs.org/guide/migration/v-model.html#%E8%BF%81%E7%A7%BB%E7%AD%96%E7%95%A5
|
||||||
|
modelValue: String,
|
||||||
|
// https://www.wangeditor.com/v5/getting-started.html#mode-%E6%A8%A1%E5%BC%8F
|
||||||
|
// 模式,可选 <default|simple>,默认 default
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'default',
|
||||||
|
},
|
||||||
|
// 高度
|
||||||
|
height: {
|
||||||
|
type: String,
|
||||||
|
default: () => '310px',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props, { emit }) {
|
||||||
|
const state = reactive<WangeditorState>({
|
||||||
|
editorToolbar: null,
|
||||||
|
editor: null,
|
||||||
|
editorContent: null,
|
||||||
|
});
|
||||||
|
// 富文本配置
|
||||||
|
const wangeditorConfig = () => {
|
||||||
|
const editorConfig: Partial<IEditorConfig> = { MENU_CONF: {} };
|
||||||
|
props.isDisable ? (editorConfig.readOnly = true) : (editorConfig.readOnly = false);
|
||||||
|
editorConfig.placeholder = props.placeholder;
|
||||||
|
editorConfig.onChange = (editor: IDomEditor) => {
|
||||||
|
// console.log('content', editor.children);
|
||||||
|
// console.log('html', editor.getHtml());
|
||||||
|
emit('update:modelValue', editor.getHtml());
|
||||||
|
};
|
||||||
|
(<any>editorConfig).MENU_CONF['uploadImage'] = {
|
||||||
|
base64LimitSize: 10 * 1024 * 1024,
|
||||||
|
};
|
||||||
|
return editorConfig;
|
||||||
|
};
|
||||||
|
//
|
||||||
|
const toolbarConfig = () => {
|
||||||
|
const toolbarConfig: Partial<IToolbarConfig> = {};
|
||||||
|
toolbarConfig.toolbarKeys = toolbarKeys;
|
||||||
|
return toolbarConfig;
|
||||||
|
};
|
||||||
|
// 初始化富文本
|
||||||
|
// https://www.wangeditor.com/
|
||||||
|
const initWangeditor = () => {
|
||||||
|
state.editor = createEditor({
|
||||||
|
html: props.modelValue,
|
||||||
|
selector: state.editorContent!,
|
||||||
|
config: wangeditorConfig(),
|
||||||
|
mode: props.mode,
|
||||||
|
});
|
||||||
|
createToolbar({
|
||||||
|
editor: state.editor,
|
||||||
|
selector: state.editorToolbar!,
|
||||||
|
mode: props.mode,
|
||||||
|
config: toolbarConfig(),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initWangeditor();
|
||||||
|
});
|
||||||
|
// 监听双向绑定值的改变
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I4LM7I
|
||||||
|
watch(
|
||||||
|
() => props.modelValue,
|
||||||
|
(value) => {
|
||||||
|
state.editor.clear();
|
||||||
|
state.editor.dangerouslyInsertHtml(value);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
60
web/src/components/editor/toolbar.ts
Normal file
60
web/src/components/editor/toolbar.ts
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
/**
|
||||||
|
* 工具栏配置
|
||||||
|
*/
|
||||||
|
export const toolbarKeys = [
|
||||||
|
'headerSelect',
|
||||||
|
'blockquote',
|
||||||
|
'|',
|
||||||
|
'bold',
|
||||||
|
'underline',
|
||||||
|
'italic',
|
||||||
|
{
|
||||||
|
key: 'group-more-style',
|
||||||
|
title: '更多',
|
||||||
|
iconSvg:
|
||||||
|
'<svg viewBox="0 0 1024 1024"><path d="M204.8 505.6m-76.8 0a76.8 76.8 0 1 0 153.6 0 76.8 76.8 0 1 0-153.6 0Z"></path><path d="M505.6 505.6m-76.8 0a76.8 76.8 0 1 0 153.6 0 76.8 76.8 0 1 0-153.6 0Z"></path><path d="M806.4 505.6m-76.8 0a76.8 76.8 0 1 0 153.6 0 76.8 76.8 0 1 0-153.6 0Z"></path></svg>',
|
||||||
|
menuKeys: ['through', 'code', 'sup', 'sub', 'clearStyle'],
|
||||||
|
},
|
||||||
|
'color',
|
||||||
|
'bgColor',
|
||||||
|
'|',
|
||||||
|
'fontSize',
|
||||||
|
'fontFamily',
|
||||||
|
'lineHeight',
|
||||||
|
'|',
|
||||||
|
'bulletedList',
|
||||||
|
'numberedList',
|
||||||
|
'todo',
|
||||||
|
{
|
||||||
|
key: 'group-justify',
|
||||||
|
title: '对齐',
|
||||||
|
iconSvg:
|
||||||
|
'<svg viewBox="0 0 1024 1024"><path d="M768 793.6v102.4H51.2v-102.4h716.8z m204.8-230.4v102.4H51.2v-102.4h921.6z m-204.8-230.4v102.4H51.2v-102.4h716.8zM972.8 102.4v102.4H51.2V102.4h921.6z"></path></svg>',
|
||||||
|
menuKeys: ['justifyLeft', 'justifyRight', 'justifyCenter', 'justifyJustify'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'group-indent',
|
||||||
|
title: '缩进',
|
||||||
|
iconSvg:
|
||||||
|
'<svg viewBox="0 0 1024 1024"><path d="M0 64h1024v128H0z m384 192h640v128H384z m0 192h640v128H384z m0 192h640v128H384zM0 832h1024v128H0z m0-128V320l256 192z"></path></svg>',
|
||||||
|
menuKeys: ['indent', 'delIndent'],
|
||||||
|
},
|
||||||
|
'|',
|
||||||
|
'emotion',
|
||||||
|
'insertLink',
|
||||||
|
{
|
||||||
|
key: 'group-image',
|
||||||
|
title: '图片',
|
||||||
|
iconSvg:
|
||||||
|
'<svg viewBox="0 0 1024 1024"><path d="M959.877 128l0.123 0.123v767.775l-0.123 0.122H64.102l-0.122-0.122V128.123l0.122-0.123h895.775zM960 64H64C28.795 64 0 92.795 0 128v768c0 35.205 28.795 64 64 64h896c35.205 0 64-28.795 64-64V128c0-35.205-28.795-64-64-64zM832 288.01c0 53.023-42.988 96.01-96.01 96.01s-96.01-42.987-96.01-96.01S682.967 192 735.99 192 832 234.988 832 288.01zM896 832H128V704l224.01-384 256 320h64l224.01-192z"></path></svg>',
|
||||||
|
menuKeys: ['uploadImage'],
|
||||||
|
},
|
||||||
|
'insertTable',
|
||||||
|
'codeBlock',
|
||||||
|
'divider',
|
||||||
|
'|',
|
||||||
|
'undo',
|
||||||
|
'redo',
|
||||||
|
'|',
|
||||||
|
'fullScreen',
|
||||||
|
];
|
||||||
252
web/src/components/iconSelector/index.vue
Normal file
252
web/src/components/iconSelector/index.vue
Normal file
@@ -0,0 +1,252 @@
|
|||||||
|
<template>
|
||||||
|
<div class="icon-selector w100 h100">
|
||||||
|
<el-popover
|
||||||
|
placement="bottom"
|
||||||
|
:width="fontIconWidth"
|
||||||
|
trigger="click"
|
||||||
|
transition="el-zoom-in-top"
|
||||||
|
popper-class="icon-selector-popper"
|
||||||
|
@show="onPopoverShow"
|
||||||
|
>
|
||||||
|
<template #reference>
|
||||||
|
<el-input
|
||||||
|
v-model="fontIconSearch"
|
||||||
|
:placeholder="fontIconPlaceholder"
|
||||||
|
:clearable="clearable"
|
||||||
|
:disabled="disabled"
|
||||||
|
:size="size"
|
||||||
|
ref="inputWidthRef"
|
||||||
|
@clear="onClearFontIcon"
|
||||||
|
@focus="onIconFocus"
|
||||||
|
@blur="onIconBlur"
|
||||||
|
>
|
||||||
|
<template #prepend>
|
||||||
|
<SvgIcon
|
||||||
|
:name="fontIconPrefix === '' ? prepend : fontIconPrefix"
|
||||||
|
class="font14"
|
||||||
|
v-if="fontIconPrefix === '' ? prepend?.indexOf('ele-') > -1 : fontIconPrefix?.indexOf('ele-') > -1"
|
||||||
|
/>
|
||||||
|
<i v-else :class="fontIconPrefix === '' ? prepend : fontIconPrefix" class="font14"></i>
|
||||||
|
</template>
|
||||||
|
</el-input>
|
||||||
|
</template>
|
||||||
|
<template #default>
|
||||||
|
<div class="icon-selector-warp">
|
||||||
|
<div class="icon-selector-warp-title flex">
|
||||||
|
<div class="flex-auto">{{ title }}</div>
|
||||||
|
<div class="icon-selector-warp-title-tab" v-if="type === 'all'">
|
||||||
|
<span :class="{ 'span-active': fontIconType === 'ali' }" @click="onIconChange('ali')" class="ml10" title="iconfont 图标">ali</span>
|
||||||
|
<span :class="{ 'span-active': fontIconType === 'ele' }" @click="onIconChange('ele')" class="ml10" title="elementPlus 图标">ele</span>
|
||||||
|
<span :class="{ 'span-active': fontIconType === 'awe' }" @click="onIconChange('awe')" class="ml10" title="fontawesome 图标">awe</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="icon-selector-warp-row">
|
||||||
|
<el-scrollbar ref="selectorScrollbarRef">
|
||||||
|
<el-row :gutter="10" v-if="fontIconSheetsFilterList.length > 0">
|
||||||
|
<el-col :xs="6" :sm="4" :md="4" :lg="4" :xl="4" @click="onColClick(v)" v-for="(v, k) in fontIconSheetsFilterList" :key="k">
|
||||||
|
<div class="icon-selector-warp-item" :class="{ 'icon-selector-active': fontIconPrefix === v }">
|
||||||
|
<div class="flex-margin">
|
||||||
|
<div class="icon-selector-warp-item-value">
|
||||||
|
<SvgIcon :name="v" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-empty :image-size="100" v-if="fontIconSheetsFilterList.length <= 0" :description="emptyDescription"></el-empty>
|
||||||
|
</el-scrollbar>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-popover>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { ref, toRefs, reactive, onMounted, nextTick, computed, watch, defineComponent } from 'vue';
|
||||||
|
import initIconfont from '/@/utils/getStyleSheets';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'iconSelector',
|
||||||
|
emits: ['update:modelValue', 'get', 'clear'],
|
||||||
|
props: {
|
||||||
|
// 输入框前置内容
|
||||||
|
prepend: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'ele-Pointer',
|
||||||
|
},
|
||||||
|
// 输入框占位文本
|
||||||
|
placeholder: {
|
||||||
|
type: String,
|
||||||
|
default: () => '请输入内容搜索图标或者选择图标',
|
||||||
|
},
|
||||||
|
// 输入框占位文本
|
||||||
|
size: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'default',
|
||||||
|
},
|
||||||
|
// 弹窗标题
|
||||||
|
title: {
|
||||||
|
type: String,
|
||||||
|
default: () => '请选择图标',
|
||||||
|
},
|
||||||
|
// icon 图标类型
|
||||||
|
type: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'ele',
|
||||||
|
},
|
||||||
|
// 禁用
|
||||||
|
disabled: {
|
||||||
|
type: Boolean,
|
||||||
|
default: () => false,
|
||||||
|
},
|
||||||
|
// 是否可清空
|
||||||
|
clearable: {
|
||||||
|
type: Boolean,
|
||||||
|
default: () => true,
|
||||||
|
},
|
||||||
|
// 自定义空状态描述文字
|
||||||
|
emptyDescription: {
|
||||||
|
type: String,
|
||||||
|
default: () => '无相关图标',
|
||||||
|
},
|
||||||
|
// 双向绑定值,默认为 modelValue,
|
||||||
|
// 参考:https://v3.cn.vuejs.org/guide/migration/v-model.html#%E8%BF%81%E7%A7%BB%E7%AD%96%E7%95%A5
|
||||||
|
// 参考:https://v3.cn.vuejs.org/guide/component-custom-events.html#%E5%A4%9A%E4%B8%AA-v-model-%E7%BB%91%E5%AE%9A
|
||||||
|
modelValue: String,
|
||||||
|
},
|
||||||
|
setup(props, { emit }) {
|
||||||
|
const inputWidthRef = ref();
|
||||||
|
const selectorScrollbarRef = ref();
|
||||||
|
const state = reactive({
|
||||||
|
fontIconPrefix: '',
|
||||||
|
fontIconWidth: 0,
|
||||||
|
fontIconSearch: '',
|
||||||
|
fontIconTabsIndex: 0,
|
||||||
|
fontIconSheetsList: [],
|
||||||
|
fontIconPlaceholder: '',
|
||||||
|
fontIconType: 'ali',
|
||||||
|
fontIconShow: true,
|
||||||
|
});
|
||||||
|
// 处理 input 获取焦点时,modelValue 有值时,改变 input 的 placeholder 值
|
||||||
|
const onIconFocus = () => {
|
||||||
|
if (!props.modelValue) return false;
|
||||||
|
state.fontIconSearch = '';
|
||||||
|
state.fontIconPlaceholder = props.modelValue;
|
||||||
|
};
|
||||||
|
// 处理 input 失去焦点时,为空将清空 input 值,为点击选中图标时,将取原先值
|
||||||
|
const onIconBlur = () => {
|
||||||
|
setTimeout(() => {
|
||||||
|
const icon = state.fontIconSheetsList.filter((icon: string) => icon === state.fontIconSearch);
|
||||||
|
if (icon.length <= 0) state.fontIconSearch = '';
|
||||||
|
}, 300);
|
||||||
|
};
|
||||||
|
// 处理 icon 双向绑定数值回显
|
||||||
|
const initModeValueEcho = () => {
|
||||||
|
if (props.modelValue === '') return ((<string | undefined>state.fontIconPlaceholder) = props.placeholder);
|
||||||
|
(<string | undefined>state.fontIconPlaceholder) = props.modelValue;
|
||||||
|
(<string | undefined>state.fontIconPrefix) = props.modelValue;
|
||||||
|
};
|
||||||
|
// 处理 icon type 类型为 all 时,类型 ali、ele、awe 回显问题
|
||||||
|
const initFontIconTypeEcho = () => {
|
||||||
|
if ((<any>props.modelValue)?.indexOf('iconfont') > -1) onIconChange('ali');
|
||||||
|
else if ((<any>props.modelValue)?.indexOf('ele-') > -1) onIconChange('ele');
|
||||||
|
else if ((<any>props.modelValue)?.indexOf('fa') > -1) onIconChange('awe');
|
||||||
|
else onIconChange('ali');
|
||||||
|
};
|
||||||
|
// 图标搜索及图标数据显示
|
||||||
|
const fontIconSheetsFilterList = computed(() => {
|
||||||
|
if (!state.fontIconSearch) return state.fontIconSheetsList;
|
||||||
|
let search = state.fontIconSearch.trim().toLowerCase();
|
||||||
|
return state.fontIconSheetsList.filter((item: any) => {
|
||||||
|
if (item.toLowerCase().indexOf(search) !== -1) return item;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 获取 input 的宽度
|
||||||
|
const getInputWidth = () => {
|
||||||
|
nextTick(() => {
|
||||||
|
state.fontIconWidth = inputWidthRef.value.$el.offsetWidth;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 监听页面宽度改变
|
||||||
|
const initResize = () => {
|
||||||
|
window.addEventListener('resize', () => {
|
||||||
|
getInputWidth();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 初始化数据
|
||||||
|
const initFontIconData = async (type: string) => {
|
||||||
|
state.fontIconSheetsList = [];
|
||||||
|
if (type === 'ali') {
|
||||||
|
await initIconfont.ali().then((res: any) => {
|
||||||
|
// 阿里字体图标使用 `iconfont xxx`
|
||||||
|
state.fontIconSheetsList = res.map((i: string) => `iconfont ${i}`);
|
||||||
|
});
|
||||||
|
} else if (type === 'ele') {
|
||||||
|
await initIconfont.ele().then((res: any) => {
|
||||||
|
state.fontIconSheetsList = res;
|
||||||
|
});
|
||||||
|
} else if (type === 'awe') {
|
||||||
|
await initIconfont.awe().then((res: any) => {
|
||||||
|
// fontawesome字体图标使用 `fa xxx`
|
||||||
|
state.fontIconSheetsList = res.map((i: string) => `fa ${i}`);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 初始化 input 的 placeholder
|
||||||
|
// 参考(单项数据流):https://cn.vuejs.org/v2/guide/components-props.html?#%E5%8D%95%E5%90%91%E6%95%B0%E6%8D%AE%E6%B5%81
|
||||||
|
state.fontIconPlaceholder = props.placeholder;
|
||||||
|
// 初始化双向绑定回显
|
||||||
|
initModeValueEcho();
|
||||||
|
};
|
||||||
|
// 图标点击切换
|
||||||
|
const onIconChange = (type: string) => {
|
||||||
|
state.fontIconType = type;
|
||||||
|
initFontIconData(type);
|
||||||
|
};
|
||||||
|
// 获取当前点击的 icon 图标
|
||||||
|
const onColClick = (v: any) => {
|
||||||
|
state.fontIconPlaceholder = v;
|
||||||
|
state.fontIconPrefix = v;
|
||||||
|
emit('get', state.fontIconPrefix);
|
||||||
|
emit('update:modelValue', state.fontIconPrefix);
|
||||||
|
};
|
||||||
|
// 清空当前点击的 icon 图标
|
||||||
|
const onClearFontIcon = () => {
|
||||||
|
state.fontIconPrefix = '';
|
||||||
|
emit('clear', state.fontIconPrefix);
|
||||||
|
emit('update:modelValue', state.fontIconPrefix);
|
||||||
|
};
|
||||||
|
// 监听 Popover 打开,用于双向绑定值回显
|
||||||
|
const onPopoverShow = () => {
|
||||||
|
initModeValueEcho();
|
||||||
|
initFontIconTypeEcho();
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initModeValueEcho();
|
||||||
|
initResize();
|
||||||
|
getInputWidth();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听双向绑定 modelValue 的变化
|
||||||
|
watch(
|
||||||
|
() => props.modelValue,
|
||||||
|
() => {
|
||||||
|
initModeValueEcho();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
inputWidthRef,
|
||||||
|
selectorScrollbarRef,
|
||||||
|
fontIconSheetsFilterList,
|
||||||
|
onColClick,
|
||||||
|
onIconChange,
|
||||||
|
onClearFontIcon,
|
||||||
|
onIconFocus,
|
||||||
|
onIconBlur,
|
||||||
|
onPopoverShow,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
195
web/src/components/noticeBar/index.vue
Normal file
195
web/src/components/noticeBar/index.vue
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
<template>
|
||||||
|
<div class="notice-bar" :style="{ background, height: `${height}px` }" v-show="!isMode">
|
||||||
|
<div class="notice-bar-warp" :style="{ color, fontSize: `${size}px` }">
|
||||||
|
<i v-if="leftIcon" class="notice-bar-warp-left-icon" :class="leftIcon"></i>
|
||||||
|
<div class="notice-bar-warp-text-box" ref="noticeBarWarpRef">
|
||||||
|
<div class="notice-bar-warp-text" ref="noticeBarTextRef" v-if="!scrollable">{{ text }}</div>
|
||||||
|
<div class="notice-bar-warp-slot" v-else><slot /></div>
|
||||||
|
</div>
|
||||||
|
<SvgIcon :name="rightIcon" v-if="rightIcon" class="notice-bar-warp-right-icon" @click="onRightIconClick" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, defineComponent, ref, onMounted, nextTick } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'noticeBar',
|
||||||
|
props: {
|
||||||
|
// 通知栏模式,可选值为 closeable link
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
// 通知文本内容
|
||||||
|
text: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
// 通知文本颜色
|
||||||
|
color: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'var(--el-color-warning)',
|
||||||
|
},
|
||||||
|
// 通知背景色
|
||||||
|
background: {
|
||||||
|
type: String,
|
||||||
|
default: () => 'var(--el-color-warning-light-9)',
|
||||||
|
},
|
||||||
|
// 字体大小,单位px
|
||||||
|
size: {
|
||||||
|
type: [Number, String],
|
||||||
|
default: () => 14,
|
||||||
|
},
|
||||||
|
// 通知栏高度,单位px
|
||||||
|
height: {
|
||||||
|
type: Number,
|
||||||
|
default: () => 40,
|
||||||
|
},
|
||||||
|
// 动画延迟时间 (s)
|
||||||
|
delay: {
|
||||||
|
type: Number,
|
||||||
|
default: () => 1,
|
||||||
|
},
|
||||||
|
// 滚动速率 (px/s)
|
||||||
|
speed: {
|
||||||
|
type: Number,
|
||||||
|
default: () => 100,
|
||||||
|
},
|
||||||
|
// 是否开启垂直滚动
|
||||||
|
scrollable: {
|
||||||
|
type: Boolean,
|
||||||
|
default: () => false,
|
||||||
|
},
|
||||||
|
// 自定义左侧图标
|
||||||
|
leftIcon: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
// 自定义右侧图标
|
||||||
|
rightIcon: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props, { emit }) {
|
||||||
|
const noticeBarWarpRef = ref();
|
||||||
|
const noticeBarTextRef = ref();
|
||||||
|
const state = reactive({
|
||||||
|
order: 1,
|
||||||
|
oneTime: 0,
|
||||||
|
twoTime: 0,
|
||||||
|
warpOWidth: 0,
|
||||||
|
textOWidth: 0,
|
||||||
|
isMode: false,
|
||||||
|
});
|
||||||
|
// 初始化 animation 各项参数
|
||||||
|
const initAnimation = () => {
|
||||||
|
nextTick(() => {
|
||||||
|
state.warpOWidth = noticeBarWarpRef.value.offsetWidth;
|
||||||
|
state.textOWidth = noticeBarTextRef.value.offsetWidth;
|
||||||
|
document.styleSheets[0].insertRule(`@keyframes oneAnimation {0% {left: 0px;} 100% {left: -${state.textOWidth}px;}}`);
|
||||||
|
document.styleSheets[0].insertRule(`@keyframes twoAnimation {0% {left: ${state.warpOWidth}px;} 100% {left: -${state.textOWidth}px;}}`);
|
||||||
|
computeAnimationTime();
|
||||||
|
setTimeout(() => {
|
||||||
|
changeAnimation();
|
||||||
|
}, props.delay * 1000);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 计算 animation 滚动时长
|
||||||
|
const computeAnimationTime = () => {
|
||||||
|
state.oneTime = state.textOWidth / props.speed;
|
||||||
|
state.twoTime = (state.textOWidth + state.warpOWidth) / props.speed;
|
||||||
|
};
|
||||||
|
// 改变 animation 动画调用
|
||||||
|
const changeAnimation = () => {
|
||||||
|
if (state.order === 1) {
|
||||||
|
noticeBarTextRef.value.style.cssText = `animation: oneAnimation ${state.oneTime}s linear; opactity: 1;}`;
|
||||||
|
state.order = 2;
|
||||||
|
} else {
|
||||||
|
noticeBarTextRef.value.style.cssText = `animation: twoAnimation ${state.twoTime}s linear infinite; opacity: 1;`;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 监听 animation 动画的结束
|
||||||
|
const listenerAnimationend = () => {
|
||||||
|
noticeBarTextRef.value.addEventListener(
|
||||||
|
'animationend',
|
||||||
|
() => {
|
||||||
|
changeAnimation();
|
||||||
|
},
|
||||||
|
false
|
||||||
|
);
|
||||||
|
};
|
||||||
|
// 右侧 icon 图标点击
|
||||||
|
const onRightIconClick = () => {
|
||||||
|
if (!props.mode) return false;
|
||||||
|
if (props.mode === 'closeable') {
|
||||||
|
state.isMode = true;
|
||||||
|
emit('close');
|
||||||
|
} else if (props.mode === 'link') {
|
||||||
|
emit('link');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
if (props.scrollable) return false;
|
||||||
|
initAnimation();
|
||||||
|
listenerAnimationend();
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
noticeBarWarpRef,
|
||||||
|
noticeBarTextRef,
|
||||||
|
onRightIconClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.notice-bar {
|
||||||
|
padding: 0 15px;
|
||||||
|
width: 100%;
|
||||||
|
border-radius: 4px;
|
||||||
|
.notice-bar-warp {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
width: 100%;
|
||||||
|
height: inherit;
|
||||||
|
.notice-bar-warp-text-box {
|
||||||
|
flex: 1;
|
||||||
|
height: inherit;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
.notice-bar-warp-text {
|
||||||
|
white-space: nowrap;
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
.notice-bar-warp-slot {
|
||||||
|
width: 100%;
|
||||||
|
white-space: nowrap;
|
||||||
|
:deep(.el-carousel__item) {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.notice-bar-warp-left-icon {
|
||||||
|
width: 24px;
|
||||||
|
font-size: inherit !important;
|
||||||
|
}
|
||||||
|
.notice-bar-warp-right-icon {
|
||||||
|
width: 24px;
|
||||||
|
text-align: right;
|
||||||
|
font-size: inherit !important;
|
||||||
|
&:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
73
web/src/components/svgIcon/index.vue
Normal file
73
web/src/components/svgIcon/index.vue
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
<template>
|
||||||
|
<i v-if="isShowIconSvg" class="el-icon" :style="setIconSvgStyle">
|
||||||
|
<component :is="getIconName" />
|
||||||
|
</i>
|
||||||
|
<div v-else-if="isShowIconImg" :style="setIconImgOutStyle">
|
||||||
|
<img :src="getIconName" :style="setIconSvgInsStyle" />
|
||||||
|
</div>
|
||||||
|
<i v-else :class="getIconName" :style="setIconSvgStyle" />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'svgIcon',
|
||||||
|
props: {
|
||||||
|
// svg 图标组件名字
|
||||||
|
name: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
// svg 大小
|
||||||
|
size: {
|
||||||
|
type: Number,
|
||||||
|
default: () => 14,
|
||||||
|
},
|
||||||
|
// svg 颜色
|
||||||
|
color: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
// 在线链接、本地引入地址前缀
|
||||||
|
const linesString = ['https', 'http', '/src', '/assets', import.meta.env.VITE_PUBLIC_PATH];
|
||||||
|
|
||||||
|
// 获取 icon 图标名称
|
||||||
|
const getIconName = computed(() => {
|
||||||
|
return props?.name;
|
||||||
|
});
|
||||||
|
// 用于判断 element plus 自带 svg 图标的显示、隐藏
|
||||||
|
const isShowIconSvg = computed(() => {
|
||||||
|
return props?.name?.startsWith('ele-');
|
||||||
|
});
|
||||||
|
// 用于判断在线链接、本地引入等图标显示、隐藏
|
||||||
|
const isShowIconImg = computed(() => {
|
||||||
|
return linesString.find((str) => props.name?.startsWith(str));
|
||||||
|
});
|
||||||
|
// 设置图标样式
|
||||||
|
const setIconSvgStyle = computed(() => {
|
||||||
|
return `font-size: ${props.size}px;color: ${props.color};`;
|
||||||
|
});
|
||||||
|
// 设置图片样式
|
||||||
|
const setIconImgOutStyle = computed(() => {
|
||||||
|
return `width: ${props.size}px;height: ${props.size}px;display: inline-block;overflow: hidden;`;
|
||||||
|
});
|
||||||
|
// 设置图片样式
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I59ND0
|
||||||
|
const setIconSvgInsStyle = computed(() => {
|
||||||
|
const filterStyle: string[] = [];
|
||||||
|
const compatibles: string[] = ['-webkit', '-ms', '-o', '-moz'];
|
||||||
|
compatibles.forEach((j) => filterStyle.push(`${j}-filter: drop-shadow(${props.color} 30px 0);`));
|
||||||
|
return `width: ${props.size}px;height: ${props.size}px;position: relative;left: -${props.size}px;${filterStyle.join('')}`;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
getIconName,
|
||||||
|
isShowIconSvg,
|
||||||
|
isShowIconImg,
|
||||||
|
setIconSvgStyle,
|
||||||
|
setIconImgOutStyle,
|
||||||
|
setIconSvgInsStyle,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
68
web/src/i18n/index.ts
Normal file
68
web/src/i18n/index.ts
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import { createI18n } from 'vue-i18n';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import zhcnLocale from 'element-plus/lib/locale/lang/zh-cn';
|
||||||
|
import enLocale from 'element-plus/lib/locale/lang/en';
|
||||||
|
import zhtwLocale from 'element-plus/lib/locale/lang/zh-tw';
|
||||||
|
|
||||||
|
import nextZhcn from '/@/i18n/lang/zh-cn';
|
||||||
|
import nextEn from '/@/i18n/lang/en';
|
||||||
|
import nextZhtw from '/@/i18n/lang/zh-tw';
|
||||||
|
|
||||||
|
import pagesLoginZhcn from '/@/i18n/pages/login/zh-cn';
|
||||||
|
import pagesLoginEn from '/@/i18n/pages/login/en';
|
||||||
|
import pagesLoginZhtw from '/@/i18n/pages/login/zh-tw';
|
||||||
|
import pagesFormI18nZhcn from '/@/i18n/pages/formI18n/zh-cn';
|
||||||
|
import pagesFormI18nEn from '/@/i18n/pages/formI18n/en';
|
||||||
|
import pagesFormI18nZhtw from '/@/i18n/pages/formI18n/zh-tw';
|
||||||
|
|
||||||
|
// 定义语言国际化内容
|
||||||
|
/**
|
||||||
|
* 说明:
|
||||||
|
* /src/i18n/lang 下的 ts 为框架的国际化内容
|
||||||
|
* /src/i18n/pages 下的 ts 为各界面的国际化内容
|
||||||
|
*/
|
||||||
|
const messages = {
|
||||||
|
[zhcnLocale.name]: {
|
||||||
|
...zhcnLocale,
|
||||||
|
message: {
|
||||||
|
...nextZhcn,
|
||||||
|
...pagesLoginZhcn,
|
||||||
|
...pagesFormI18nZhcn,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
[enLocale.name]: {
|
||||||
|
...enLocale,
|
||||||
|
message: {
|
||||||
|
...nextEn,
|
||||||
|
...pagesLoginEn,
|
||||||
|
...pagesFormI18nEn,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
[zhtwLocale.name]: {
|
||||||
|
...zhtwLocale,
|
||||||
|
message: {
|
||||||
|
...nextZhtw,
|
||||||
|
...pagesLoginZhtw,
|
||||||
|
...pagesFormI18nZhtw,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// 读取 pinia 默认语言
|
||||||
|
const stores = useThemeConfig(pinia);
|
||||||
|
const { themeConfig } = storeToRefs(stores);
|
||||||
|
|
||||||
|
// 导出语言国际化
|
||||||
|
// https://vue-i18n.intlify.dev/guide/essentials/fallback.html#explicit-fallback-with-one-locale
|
||||||
|
export const i18n = createI18n({
|
||||||
|
legacy: false,
|
||||||
|
silentTranslationWarn: true,
|
||||||
|
missingWarn: false,
|
||||||
|
silentFallbackWarn: true,
|
||||||
|
fallbackWarn: false,
|
||||||
|
locale: themeConfig.value.globalI18n,
|
||||||
|
fallbackLocale: zhcnLocale.name,
|
||||||
|
messages,
|
||||||
|
});
|
||||||
181
web/src/i18n/lang/en.ts
Normal file
181
web/src/i18n/lang/en.ts
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
router: {
|
||||||
|
home: 'home',
|
||||||
|
system: 'system',
|
||||||
|
systemMenu: 'systemMenu',
|
||||||
|
systemRole: 'systemRole',
|
||||||
|
systemUser: 'systemUser',
|
||||||
|
systemDept: 'systemDept',
|
||||||
|
systemDic: 'systemDic',
|
||||||
|
limits: 'limits',
|
||||||
|
limitsFrontEnd: 'FrontEnd',
|
||||||
|
limitsFrontEndPage: 'FrontEndPage',
|
||||||
|
limitsFrontEndBtn: 'FrontEndBtn',
|
||||||
|
limitsBackEnd: 'BackEnd',
|
||||||
|
limitsBackEndEndPage: 'BackEndEndPage',
|
||||||
|
menu: 'menu',
|
||||||
|
menu1: 'menu1',
|
||||||
|
menu11: 'menu11',
|
||||||
|
menu12: 'menu12',
|
||||||
|
menu121: 'menu121',
|
||||||
|
menu122: 'menu122',
|
||||||
|
menu13: 'menu13',
|
||||||
|
menu2: 'menu2',
|
||||||
|
funIndex: 'function',
|
||||||
|
funTagsView: 'funTagsView',
|
||||||
|
funCountup: 'countup',
|
||||||
|
funWangEditor: 'wangEditor',
|
||||||
|
funCropper: 'cropper',
|
||||||
|
funQrcode: 'qrcode',
|
||||||
|
funEchartsMap: 'EchartsMap',
|
||||||
|
funPrintJs: 'PrintJs',
|
||||||
|
funClipboard: 'Copy cut',
|
||||||
|
funGridLayout: 'Drag layout',
|
||||||
|
funSplitpanes: 'Pane splitter',
|
||||||
|
funDragVerify: 'Validator',
|
||||||
|
pagesIndex: 'pages',
|
||||||
|
pagesFiltering: 'Filtering',
|
||||||
|
pagesFilteringDetails: 'FilteringDetails',
|
||||||
|
pagesFilteringDetails1: 'FilteringDetails1',
|
||||||
|
pagesIocnfont: 'iconfont icon',
|
||||||
|
pagesElement: 'element icon',
|
||||||
|
pagesAwesome: 'awesome icon',
|
||||||
|
pagesFormAdapt: 'FormAdapt',
|
||||||
|
pagesTableRules: 'pagesTableRules',
|
||||||
|
pagesFormI18n: 'FormI18n',
|
||||||
|
pagesFormRules: 'Multi form validation',
|
||||||
|
pagesDynamicForm: 'Dynamic complex form',
|
||||||
|
pagesWorkflow: 'Workflow',
|
||||||
|
pagesListAdapt: 'ListAdapt',
|
||||||
|
pagesWaterfall: 'Waterfall',
|
||||||
|
pagesSteps: 'Steps',
|
||||||
|
pagesPreview: 'Large preview',
|
||||||
|
pagesWaves: 'Wave effect',
|
||||||
|
pagesTree: 'tree alter table',
|
||||||
|
pagesDrag: 'Drag command',
|
||||||
|
pagesLazyImg: 'Image lazy loading',
|
||||||
|
makeIndex: 'makeIndex',
|
||||||
|
makeSelector: 'Icon selector',
|
||||||
|
makeNoticeBar: 'notification bar',
|
||||||
|
makeSvgDemo: 'Svgicon demo',
|
||||||
|
paramsIndex: 'Routing parameters',
|
||||||
|
paramsCommon: 'General routing',
|
||||||
|
paramsDynamic: 'Dynamic routing',
|
||||||
|
paramsCommonDetails: 'General routing details',
|
||||||
|
paramsDynamicDetails: 'Dynamic routing details',
|
||||||
|
chartIndex: 'chartIndex',
|
||||||
|
visualizingIndex: 'visualizingIndex',
|
||||||
|
visualizingLinkDemo1: 'visualizingLinkDemo1',
|
||||||
|
visualizingLinkDemo2: 'visualizingLinkDemo2',
|
||||||
|
personal: 'personal',
|
||||||
|
tools: 'tools',
|
||||||
|
layoutLinkView: 'LinkView',
|
||||||
|
layoutIfameView: 'IfameView',
|
||||||
|
demo1:'demo1'
|
||||||
|
},
|
||||||
|
staticRoutes: {
|
||||||
|
signIn: 'signIn',
|
||||||
|
notFound: 'notFound',
|
||||||
|
noPower: 'noPower',
|
||||||
|
},
|
||||||
|
user: {
|
||||||
|
title0: 'Component size',
|
||||||
|
title1: 'Language switching',
|
||||||
|
title2: 'Menu search',
|
||||||
|
title3: 'Layout configuration',
|
||||||
|
title4: 'news',
|
||||||
|
title5: 'Full screen on',
|
||||||
|
title6: 'Full screen off',
|
||||||
|
dropdownLarge: 'large',
|
||||||
|
dropdownDefault: 'default',
|
||||||
|
dropdownSmall: 'small',
|
||||||
|
dropdown1: 'home page',
|
||||||
|
dropdown2: 'Personal Center',
|
||||||
|
dropdown3: '404',
|
||||||
|
dropdown4: '401',
|
||||||
|
dropdown5: 'Log out',
|
||||||
|
dropdown6: 'Code warehouse',
|
||||||
|
searchPlaceholder: 'Menu search: support Chinese, routing path',
|
||||||
|
newTitle: 'notice',
|
||||||
|
newBtn: 'All read',
|
||||||
|
newGo: 'Go to the notification center',
|
||||||
|
newDesc: 'No notice',
|
||||||
|
logOutTitle: 'Tips',
|
||||||
|
logOutMessage: 'This operation will log out. Do you want to continue?',
|
||||||
|
logOutConfirm: 'determine',
|
||||||
|
logOutCancel: 'cancel',
|
||||||
|
logOutExit: 'Exiting',
|
||||||
|
},
|
||||||
|
tagsView: {
|
||||||
|
refresh: 'refresh',
|
||||||
|
close: 'close',
|
||||||
|
closeOther: 'closeOther',
|
||||||
|
closeAll: 'closeAll',
|
||||||
|
fullscreen: 'fullscreen',
|
||||||
|
closeFullscreen: 'closeFullscreen',
|
||||||
|
},
|
||||||
|
notFound: {
|
||||||
|
foundTitle: 'Wrong address input, please re-enter the address~',
|
||||||
|
foundMsg: 'You can check the web address first, and then re-enter or give us feedback.',
|
||||||
|
foundBtn: 'Back to home page',
|
||||||
|
},
|
||||||
|
noAccess: {
|
||||||
|
accessTitle: 'You are not authorized to operate~',
|
||||||
|
accessMsg: 'Contact information: add QQ group discussion 665452019',
|
||||||
|
accessBtn: 'Reauthorization',
|
||||||
|
},
|
||||||
|
layout: {
|
||||||
|
configTitle: 'Layout configuration',
|
||||||
|
oneTitle: 'Global Themes',
|
||||||
|
twoTopTitle: 'top bar set up',
|
||||||
|
twoMenuTitle: 'Menu set up',
|
||||||
|
twoColumnsTitle: 'Columns set up',
|
||||||
|
twoTopBar: 'Top bar background',
|
||||||
|
twoTopBarColor: 'Top bar default font color',
|
||||||
|
twoIsTopBarColorGradual: 'Top bar gradient',
|
||||||
|
twoMenuBar: 'Menu background',
|
||||||
|
twoMenuBarColor: 'Menu default font color',
|
||||||
|
twoIsMenuBarColorGradual: 'Menu gradient',
|
||||||
|
twoColumnsMenuBar: 'Column menu background',
|
||||||
|
twoColumnsMenuBarColor: 'Default font color bar menu',
|
||||||
|
twoIsColumnsMenuBarColorGradual: 'Column gradient',
|
||||||
|
threeTitle: 'Interface settings',
|
||||||
|
threeIsCollapse: 'Menu horizontal collapse',
|
||||||
|
threeIsUniqueOpened: 'Menu accordion',
|
||||||
|
threeIsFixedHeader: 'Fixed header',
|
||||||
|
threeIsClassicSplitMenu: 'Classic layout split menu',
|
||||||
|
threeIsLockScreen: 'Open the lock screen',
|
||||||
|
threeLockScreenTime: 'screen locking(s/s)',
|
||||||
|
fourTitle: 'Interface display',
|
||||||
|
fourIsShowLogo: 'Sidebar logo',
|
||||||
|
fourIsBreadcrumb: 'Open breadcrumb',
|
||||||
|
fourIsBreadcrumbIcon: 'Open breadcrumb icon',
|
||||||
|
fourIsTagsview: 'Open tagsview',
|
||||||
|
fourIsTagsviewIcon: 'Open tagsview Icon',
|
||||||
|
fourIsCacheTagsView: 'Enable tagsview cache',
|
||||||
|
fourIsSortableTagsView: 'Enable tagsview drag',
|
||||||
|
fourIsShareTagsView: 'Enable tagsview sharing',
|
||||||
|
fourIsFooter: 'Open footer',
|
||||||
|
fourIsGrayscale: 'Grey model',
|
||||||
|
fourIsInvert: 'Color weak mode',
|
||||||
|
fourIsDark: 'Dark Mode',
|
||||||
|
fourIsWartermark: 'Turn on watermark',
|
||||||
|
fourWartermarkText: 'Watermark copy',
|
||||||
|
fiveTitle: 'Other settings',
|
||||||
|
fiveTagsStyle: 'Tagsview style',
|
||||||
|
fiveAnimation: 'page animation',
|
||||||
|
fiveColumnsAsideStyle: 'Column style',
|
||||||
|
fiveColumnsAsideLayout: 'Column layout',
|
||||||
|
sixTitle: 'Layout switch',
|
||||||
|
sixDefaults: 'One',
|
||||||
|
sixClassic: 'Two',
|
||||||
|
sixTransverse: 'Three',
|
||||||
|
sixColumns: 'Four',
|
||||||
|
tipText: 'Click the button below to copy the layout configuration to `/src/stores/themeConfig.ts` It has been modified in.',
|
||||||
|
copyText: 'replication configuration',
|
||||||
|
resetText: 'restore default',
|
||||||
|
copyTextSuccess: 'Copy succeeded!',
|
||||||
|
copyTextError: 'Copy failed!',
|
||||||
|
},
|
||||||
|
};
|
||||||
181
web/src/i18n/lang/zh-cn.ts
Normal file
181
web/src/i18n/lang/zh-cn.ts
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
router: {
|
||||||
|
home: '首页',
|
||||||
|
system: '系统设置',
|
||||||
|
systemMenu: '菜单管理',
|
||||||
|
systemRole: '角色管理',
|
||||||
|
systemUser: '用户管理',
|
||||||
|
systemDept: '部门管理',
|
||||||
|
systemDic: '字典管理',
|
||||||
|
limits: '权限管理',
|
||||||
|
limitsFrontEnd: '前端控制',
|
||||||
|
limitsFrontEndPage: '页面权限',
|
||||||
|
limitsFrontEndBtn: '按钮权限',
|
||||||
|
limitsBackEnd: '后端控制',
|
||||||
|
limitsBackEndEndPage: '页面权限',
|
||||||
|
menu: '菜单嵌套',
|
||||||
|
menu1: '菜单1',
|
||||||
|
menu11: '菜单11',
|
||||||
|
menu12: '菜单12',
|
||||||
|
menu121: '菜单121',
|
||||||
|
menu122: '菜单122',
|
||||||
|
menu13: '菜单13',
|
||||||
|
menu2: '菜单2',
|
||||||
|
funIndex: '功能',
|
||||||
|
funTagsView: 'tagsView 操作',
|
||||||
|
funCountup: '数字滚动',
|
||||||
|
funWangEditor: 'Editor 编辑器',
|
||||||
|
funCropper: '图片裁剪',
|
||||||
|
funQrcode: '二维码生成',
|
||||||
|
funEchartsMap: '地理坐标/地图',
|
||||||
|
funPrintJs: '页面打印',
|
||||||
|
funClipboard: '复制剪切',
|
||||||
|
funGridLayout: '拖拽布局',
|
||||||
|
funSplitpanes: '窗格拆分器',
|
||||||
|
funDragVerify: '验证器',
|
||||||
|
pagesIndex: '页面',
|
||||||
|
pagesFiltering: '过滤筛选组件',
|
||||||
|
pagesFilteringDetails: '过滤筛选组件详情',
|
||||||
|
pagesFilteringDetails1: '过滤筛选组件详情111',
|
||||||
|
pagesIocnfont: 'ali 字体图标',
|
||||||
|
pagesElement: 'ele 字体图标',
|
||||||
|
pagesAwesome: 'awe 字体图标',
|
||||||
|
pagesFormAdapt: '表单自适应',
|
||||||
|
pagesTableRules: '表单表格验证',
|
||||||
|
pagesFormI18n: '表单国际化',
|
||||||
|
pagesFormRules: '多表单验证',
|
||||||
|
pagesDynamicForm: '动态复杂表单',
|
||||||
|
pagesWorkflow: '工作流',
|
||||||
|
pagesListAdapt: '列表自适应',
|
||||||
|
pagesWaterfall: '瀑布屏',
|
||||||
|
pagesSteps: '步骤条',
|
||||||
|
pagesPreview: '大图预览',
|
||||||
|
pagesWaves: '波浪效果',
|
||||||
|
pagesTree: '树形改表格',
|
||||||
|
pagesDrag: '拖动指令',
|
||||||
|
pagesLazyImg: '图片懒加载',
|
||||||
|
makeIndex: '组件封装',
|
||||||
|
makeSelector: '图标选择器',
|
||||||
|
makeNoticeBar: '滚动通知栏',
|
||||||
|
makeSvgDemo: 'svgIcon 演示',
|
||||||
|
paramsIndex: '路由参数',
|
||||||
|
paramsCommon: '普通路由',
|
||||||
|
paramsDynamic: '动态路由',
|
||||||
|
paramsCommonDetails: '普通路由详情',
|
||||||
|
paramsDynamicDetails: '动态路由详情',
|
||||||
|
chartIndex: '大数据图表',
|
||||||
|
visualizingIndex: '数据可视化',
|
||||||
|
visualizingLinkDemo1: '数据可视化演示1',
|
||||||
|
visualizingLinkDemo2: '数据可视化演示2',
|
||||||
|
personal: '个人中心',
|
||||||
|
tools: '工具类集合',
|
||||||
|
layoutLinkView: '外链',
|
||||||
|
layoutIfameView: '内嵌 iframe',
|
||||||
|
demo1:'demo1'
|
||||||
|
},
|
||||||
|
staticRoutes: {
|
||||||
|
signIn: '登录',
|
||||||
|
notFound: '找不到此页面',
|
||||||
|
noPower: '没有权限',
|
||||||
|
},
|
||||||
|
user: {
|
||||||
|
title0: '组件大小',
|
||||||
|
title1: '语言切换',
|
||||||
|
title2: '菜单搜索',
|
||||||
|
title3: '布局配置',
|
||||||
|
title4: '消息',
|
||||||
|
title5: '开全屏',
|
||||||
|
title6: '关全屏',
|
||||||
|
dropdownLarge: '大型',
|
||||||
|
dropdownDefault: '默认',
|
||||||
|
dropdownSmall: '小型',
|
||||||
|
dropdown1: '首页',
|
||||||
|
dropdown2: '个人中心',
|
||||||
|
dropdown3: '404',
|
||||||
|
dropdown4: '401',
|
||||||
|
dropdown5: '退出登录',
|
||||||
|
dropdown6: '代码仓库',
|
||||||
|
searchPlaceholder: '菜单搜索:支持中文、路由路径',
|
||||||
|
newTitle: '通知',
|
||||||
|
newBtn: '全部已读',
|
||||||
|
newGo: '前往通知中心',
|
||||||
|
newDesc: '暂无通知',
|
||||||
|
logOutTitle: '提示',
|
||||||
|
logOutMessage: '此操作将退出登录, 是否继续?',
|
||||||
|
logOutConfirm: '确定',
|
||||||
|
logOutCancel: '取消',
|
||||||
|
logOutExit: '退出中',
|
||||||
|
},
|
||||||
|
tagsView: {
|
||||||
|
refresh: '刷新',
|
||||||
|
close: '关闭',
|
||||||
|
closeOther: '关闭其它',
|
||||||
|
closeAll: '全部关闭',
|
||||||
|
fullscreen: '当前页全屏',
|
||||||
|
closeFullscreen: '关闭全屏',
|
||||||
|
},
|
||||||
|
notFound: {
|
||||||
|
foundTitle: '地址输入错误,请重新输入地址~',
|
||||||
|
foundMsg: '您可以先检查网址,然后重新输入或给我们反馈问题。',
|
||||||
|
foundBtn: '返回首页',
|
||||||
|
},
|
||||||
|
noAccess: {
|
||||||
|
accessTitle: '您未被授权,没有操作权限~',
|
||||||
|
accessMsg: '联系方式:加QQ群探讨 665452019',
|
||||||
|
accessBtn: '重新授权',
|
||||||
|
},
|
||||||
|
layout: {
|
||||||
|
configTitle: '布局配置',
|
||||||
|
oneTitle: '全局主题',
|
||||||
|
twoTopTitle: '顶栏设置',
|
||||||
|
twoMenuTitle: '菜单设置',
|
||||||
|
twoColumnsTitle: '分栏设置',
|
||||||
|
twoTopBar: '顶栏背景',
|
||||||
|
twoTopBarColor: '顶栏默认字体颜色',
|
||||||
|
twoIsTopBarColorGradual: '顶栏背景渐变',
|
||||||
|
twoMenuBar: '菜单背景',
|
||||||
|
twoMenuBarColor: '菜单默认字体颜色',
|
||||||
|
twoIsMenuBarColorGradual: '菜单背景渐变',
|
||||||
|
twoColumnsMenuBar: '分栏菜单背景',
|
||||||
|
twoColumnsMenuBarColor: '分栏菜单默认字体颜色',
|
||||||
|
twoIsColumnsMenuBarColorGradual: '分栏菜单背景渐变',
|
||||||
|
threeTitle: '界面设置',
|
||||||
|
threeIsCollapse: '菜单水平折叠',
|
||||||
|
threeIsUniqueOpened: '菜单手风琴',
|
||||||
|
threeIsFixedHeader: '固定 Header',
|
||||||
|
threeIsClassicSplitMenu: '经典布局分割菜单',
|
||||||
|
threeIsLockScreen: '开启锁屏',
|
||||||
|
threeLockScreenTime: '自动锁屏(s/秒)',
|
||||||
|
fourTitle: '界面显示',
|
||||||
|
fourIsShowLogo: '侧边栏 Logo',
|
||||||
|
fourIsBreadcrumb: '开启 Breadcrumb',
|
||||||
|
fourIsBreadcrumbIcon: '开启 Breadcrumb 图标',
|
||||||
|
fourIsTagsview: '开启 Tagsview',
|
||||||
|
fourIsTagsviewIcon: '开启 Tagsview 图标',
|
||||||
|
fourIsCacheTagsView: '开启 TagsView 缓存',
|
||||||
|
fourIsSortableTagsView: '开启 TagsView 拖拽',
|
||||||
|
fourIsShareTagsView: '开启 TagsView 共用',
|
||||||
|
fourIsFooter: '开启 Footer',
|
||||||
|
fourIsGrayscale: '灰色模式',
|
||||||
|
fourIsInvert: '色弱模式',
|
||||||
|
fourIsDark: '深色模式',
|
||||||
|
fourIsWartermark: '开启水印',
|
||||||
|
fourWartermarkText: '水印文案',
|
||||||
|
fiveTitle: '其它设置',
|
||||||
|
fiveTagsStyle: 'Tagsview 风格',
|
||||||
|
fiveAnimation: '主页面切换动画',
|
||||||
|
fiveColumnsAsideStyle: '分栏高亮风格',
|
||||||
|
fiveColumnsAsideLayout: '分栏布局风格',
|
||||||
|
sixTitle: '布局切换',
|
||||||
|
sixDefaults: '默认',
|
||||||
|
sixClassic: '经典',
|
||||||
|
sixTransverse: '横向',
|
||||||
|
sixColumns: '分栏',
|
||||||
|
tipText: '点击下方按钮,复制布局配置去 `src/stores/themeConfig.ts` 中修改。',
|
||||||
|
copyText: '一键复制配置',
|
||||||
|
resetText: '一键恢复默认',
|
||||||
|
copyTextSuccess: '复制成功!',
|
||||||
|
copyTextError: '复制失败!',
|
||||||
|
},
|
||||||
|
};
|
||||||
181
web/src/i18n/lang/zh-tw.ts
Normal file
181
web/src/i18n/lang/zh-tw.ts
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
router: {
|
||||||
|
home: '首頁',
|
||||||
|
system: '系統設置',
|
||||||
|
systemMenu: '選單管理',
|
||||||
|
systemRole: '角色管理',
|
||||||
|
systemUser: '用戶管理',
|
||||||
|
systemDept: '部門管理',
|
||||||
|
systemDic: '字典管理',
|
||||||
|
limits: '許可權管理',
|
||||||
|
limitsFrontEnd: '前端控制',
|
||||||
|
limitsFrontEndPage: '頁面許可權',
|
||||||
|
limitsFrontEndBtn: '按鈕許可權',
|
||||||
|
limitsBackEnd: '後端控制',
|
||||||
|
limitsBackEndEndPage: '頁面許可權',
|
||||||
|
menu: '選單嵌套',
|
||||||
|
menu1: '選單1',
|
||||||
|
menu11: '選單11',
|
||||||
|
menu12: '選單12',
|
||||||
|
menu121: '選單121',
|
||||||
|
menu122: '選單122',
|
||||||
|
menu13: '選單13',
|
||||||
|
menu2: '選單2',
|
||||||
|
funIndex: '功能',
|
||||||
|
funTagsView: 'tagsView 操作',
|
||||||
|
funCountup: '數位滾動',
|
||||||
|
funWangEditor: 'Editor 編輯器',
|
||||||
|
funCropper: '圖片裁剪',
|
||||||
|
funQrcode: '二維碼生成',
|
||||||
|
funEchartsMap: '地理座標/地圖',
|
||||||
|
funPrintJs: '頁面列印',
|
||||||
|
funClipboard: '複製剪切',
|
||||||
|
funGridLayout: '拖拽佈局',
|
||||||
|
funSplitpanes: '窗格折開器',
|
||||||
|
funDragVerify: '驗證器',
|
||||||
|
pagesIndex: '頁面',
|
||||||
|
pagesFiltering: '過濾篩選組件',
|
||||||
|
pagesFilteringDetails: '過濾篩選組件詳情',
|
||||||
|
pagesFilteringDetails1: '過濾篩選組件詳情111',
|
||||||
|
pagesIocnfont: 'ali 字體圖標',
|
||||||
|
pagesElement: 'ele 字體圖標',
|
||||||
|
pagesAwesome: 'awe 字體圖標',
|
||||||
|
pagesFormAdapt: '表單自我調整',
|
||||||
|
pagesTableRules: '表單表格驗證',
|
||||||
|
pagesFormI18n: '表單國際化',
|
||||||
|
pagesFormRules: '多表單驗證',
|
||||||
|
pagesDynamicForm: '動態複雜表單',
|
||||||
|
pagesWorkflow: '工作流',
|
||||||
|
pagesListAdapt: '清單自我調整',
|
||||||
|
pagesWaterfall: '瀑布屏',
|
||||||
|
pagesSteps: '步驟條',
|
||||||
|
pagesPreview: '大圖預覽',
|
||||||
|
pagesWaves: '波浪效果',
|
||||||
|
pagesTree: '樹形改表格',
|
||||||
|
pagesDrag: '拖動指令',
|
||||||
|
pagesLazyImg: '圖片懶加載',
|
||||||
|
makeIndex: '組件封裝',
|
||||||
|
makeSelector: '圖標選擇器',
|
||||||
|
makeNoticeBar: '滾動通知欄',
|
||||||
|
makeSvgDemo: 'svgIcon 演示',
|
||||||
|
paramsIndex: '路由參數',
|
||||||
|
paramsCommon: '普通路由',
|
||||||
|
paramsDynamic: '動態路由',
|
||||||
|
paramsCommonDetails: '普通路由詳情',
|
||||||
|
paramsDynamicDetails: '動態路由詳情',
|
||||||
|
chartIndex: '大資料圖表',
|
||||||
|
visualizingIndex: '數據視覺化',
|
||||||
|
visualizingLinkDemo1: '數據視覺化演示1',
|
||||||
|
visualizingLinkDemo2: '數據視覺化演示2',
|
||||||
|
personal: '個人中心',
|
||||||
|
tools: '工具類集合',
|
||||||
|
layoutLinkView: '外鏈',
|
||||||
|
layoutIfameView: '内嵌 iframe',
|
||||||
|
demo1:'demo1'
|
||||||
|
},
|
||||||
|
staticRoutes: {
|
||||||
|
signIn: '登入',
|
||||||
|
notFound: '找不到此頁面',
|
||||||
|
noPower: '沒有許可權',
|
||||||
|
},
|
||||||
|
user: {
|
||||||
|
title0: '組件大小',
|
||||||
|
title1: '語言切換',
|
||||||
|
title2: '選單蒐索',
|
||||||
|
title3: '佈局配寘',
|
||||||
|
title4: '消息',
|
||||||
|
title5: '開全屏',
|
||||||
|
title6: '關全屏',
|
||||||
|
dropdownLarge: '大型',
|
||||||
|
dropdownDefault: '默認',
|
||||||
|
dropdownSmall: '小型',
|
||||||
|
dropdown1: '首頁',
|
||||||
|
dropdown2: '個人中心',
|
||||||
|
dropdown3: '404',
|
||||||
|
dropdown4: '401',
|
||||||
|
dropdown5: '登出',
|
||||||
|
dropdown6: '程式碼倉庫',
|
||||||
|
searchPlaceholder: '選單蒐索:支援中文、路由路徑',
|
||||||
|
newTitle: '通知',
|
||||||
|
newBtn: '全部已讀',
|
||||||
|
newGo: '前往通知中心',
|
||||||
|
newDesc: '暫無通知',
|
||||||
|
logOutTitle: '提示',
|
||||||
|
logOutMessage: '此操作將登出,是否繼續?',
|
||||||
|
logOutConfirm: '確定',
|
||||||
|
logOutCancel: '取消',
|
||||||
|
logOutExit: '退出中',
|
||||||
|
},
|
||||||
|
tagsView: {
|
||||||
|
refresh: '重繪',
|
||||||
|
close: '關閉',
|
||||||
|
closeOther: '關閉其它',
|
||||||
|
closeAll: '全部關閉',
|
||||||
|
fullscreen: '當前頁全屏',
|
||||||
|
closeFullscreen: '關閉全屏',
|
||||||
|
},
|
||||||
|
notFound: {
|
||||||
|
foundTitle: '地址輸入錯誤,請重新輸入地址~',
|
||||||
|
foundMsg: '您可以先檢查網址,然後重新輸入或給我們迴響問題。',
|
||||||
|
foundBtn: '返回首頁',
|
||||||
|
},
|
||||||
|
noAccess: {
|
||||||
|
accessTitle: '您未被授權,沒有操作許可權~',
|
||||||
|
accessMsg: '聯繫方式:加QQ群探討665452019',
|
||||||
|
accessBtn: '重新授權',
|
||||||
|
},
|
||||||
|
layout: {
|
||||||
|
configTitle: '佈局配寘',
|
||||||
|
oneTitle: '全域主題',
|
||||||
|
twoTopTitle: '頂欄設定',
|
||||||
|
twoMenuTitle: '選單設定',
|
||||||
|
twoColumnsTitle: '分欄設定',
|
||||||
|
twoTopBar: '頂欄背景',
|
||||||
|
twoTopBarColor: '頂欄默認字體顏色',
|
||||||
|
twoIsTopBarColorGradual: '頂欄背景漸變',
|
||||||
|
twoMenuBar: '選單背景',
|
||||||
|
twoMenuBarColor: '選單默認字體顏色',
|
||||||
|
twoIsMenuBarColorGradual: '選單背景漸變',
|
||||||
|
twoColumnsMenuBar: '分欄選單背景',
|
||||||
|
twoColumnsMenuBarColor: '分欄選單默認字體顏色',
|
||||||
|
twoIsColumnsMenuBarColorGradual: '分欄選單背景漸變',
|
||||||
|
threeTitle: '介面設定',
|
||||||
|
threeIsCollapse: '選單水准折疊',
|
||||||
|
threeIsUniqueOpened: '選單手風琴',
|
||||||
|
threeIsFixedHeader: '固定 Header',
|
||||||
|
threeIsClassicSplitMenu: '經典佈局分割選單',
|
||||||
|
threeIsLockScreen: '開啟鎖屏',
|
||||||
|
threeLockScreenTime: '自動鎖屏(s/秒)',
|
||||||
|
fourTitle: '介面顯示',
|
||||||
|
fourIsShowLogo: '側邊欄 Logo',
|
||||||
|
fourIsBreadcrumb: '開啟 Breadcrumb',
|
||||||
|
fourIsBreadcrumbIcon: '開啟 Breadcrumb 圖標',
|
||||||
|
fourIsTagsview: '開啟 Tagsview',
|
||||||
|
fourIsTagsviewIcon: '開啟 Tagsview 圖標',
|
||||||
|
fourIsCacheTagsView: '開啟 TagsView 緩存',
|
||||||
|
fourIsSortableTagsView: '開啟 TagsView 拖拽',
|
||||||
|
fourIsShareTagsView: '開啟 TagsView 共用',
|
||||||
|
fourIsFooter: '開啟 Footer',
|
||||||
|
fourIsGrayscale: '灰色模式',
|
||||||
|
fourIsInvert: '色弱模式',
|
||||||
|
fourIsDark: '深色模式',
|
||||||
|
fourIsWartermark: '開啟浮水印',
|
||||||
|
fourWartermarkText: '浮水印文案',
|
||||||
|
fiveTitle: '其它設定',
|
||||||
|
fiveTagsStyle: 'Tagsview 風格',
|
||||||
|
fiveAnimation: '主頁面切換動畫',
|
||||||
|
fiveColumnsAsideStyle: '分欄高亮風格',
|
||||||
|
fiveColumnsAsideLayout: '分欄佈局風格',
|
||||||
|
sixTitle: '佈局切換',
|
||||||
|
sixDefaults: '默認',
|
||||||
|
sixClassic: '經典',
|
||||||
|
sixTransverse: '橫向',
|
||||||
|
sixColumns: '分欄',
|
||||||
|
tipText: '點擊下方按鈕,複製佈局配寘去`src/stores/themeConfig.ts`中修改。',
|
||||||
|
copyText: '一鍵複製配寘',
|
||||||
|
resetText: '一鍵恢復默認',
|
||||||
|
copyTextSuccess: '複製成功!',
|
||||||
|
copyTextError: '複製失敗!',
|
||||||
|
},
|
||||||
|
};
|
||||||
13
web/src/i18n/pages/formI18n/en.ts
Normal file
13
web/src/i18n/pages/formI18n/en.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
formI18nLabel: {
|
||||||
|
name: 'name',
|
||||||
|
email: 'email',
|
||||||
|
autograph: 'autograph',
|
||||||
|
},
|
||||||
|
formI18nPlaceholder: {
|
||||||
|
name: 'Please enter your name',
|
||||||
|
email: 'Please enter the users Department',
|
||||||
|
autograph: 'Please enter the login account name',
|
||||||
|
},
|
||||||
|
};
|
||||||
13
web/src/i18n/pages/formI18n/zh-cn.ts
Normal file
13
web/src/i18n/pages/formI18n/zh-cn.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
formI18nLabel: {
|
||||||
|
name: '姓名',
|
||||||
|
email: '用户归属部门',
|
||||||
|
autograph: '登陆账户名',
|
||||||
|
},
|
||||||
|
formI18nPlaceholder: {
|
||||||
|
name: '请输入姓名',
|
||||||
|
email: '请输入用户归属部门',
|
||||||
|
autograph: '请输入登陆账户名',
|
||||||
|
},
|
||||||
|
};
|
||||||
13
web/src/i18n/pages/formI18n/zh-tw.ts
Normal file
13
web/src/i18n/pages/formI18n/zh-tw.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
formI18nLabel: {
|
||||||
|
name: '姓名',
|
||||||
|
email: '用戶歸屬部門',
|
||||||
|
autograph: '登入帳戶名',
|
||||||
|
},
|
||||||
|
formI18nPlaceholder: {
|
||||||
|
name: '請輸入姓名',
|
||||||
|
email: '請輸入用戶歸屬部門',
|
||||||
|
autograph: '請輸入登入帳戶名',
|
||||||
|
},
|
||||||
|
};
|
||||||
29
web/src/i18n/pages/login/en.ts
Normal file
29
web/src/i18n/pages/login/en.ts
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
label: {
|
||||||
|
one1: 'User name login',
|
||||||
|
two2: 'Mobile number',
|
||||||
|
},
|
||||||
|
link: {
|
||||||
|
one3: 'Third party login',
|
||||||
|
two4: 'Links',
|
||||||
|
},
|
||||||
|
account: {
|
||||||
|
accountPlaceholder1: 'The user name admin or not is common',
|
||||||
|
accountPlaceholder2: 'Password: 123456',
|
||||||
|
accountPlaceholder3: 'Please enter the verification code',
|
||||||
|
accountBtnText: 'Sign in',
|
||||||
|
},
|
||||||
|
mobile: {
|
||||||
|
placeholder1: 'Please input mobile phone number',
|
||||||
|
placeholder2: 'Please enter the verification code',
|
||||||
|
codeText: 'Get code',
|
||||||
|
btnText: 'Sign in',
|
||||||
|
msgText:
|
||||||
|
'Warm tip: it is recommended to use Google, Microsoft edge, version 79.0.1072.62 and above browsers, and 360 browser, please use speed mode',
|
||||||
|
},
|
||||||
|
scan: {
|
||||||
|
text: 'Open the mobile phone to scan and quickly log in / register',
|
||||||
|
},
|
||||||
|
signInText: 'welcome back!',
|
||||||
|
};
|
||||||
28
web/src/i18n/pages/login/zh-cn.ts
Normal file
28
web/src/i18n/pages/login/zh-cn.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
label: {
|
||||||
|
one1: '用户名登录',
|
||||||
|
two2: '手机号登录',
|
||||||
|
},
|
||||||
|
link: {
|
||||||
|
one3: '第三方登录',
|
||||||
|
two4: '友情链接',
|
||||||
|
},
|
||||||
|
account: {
|
||||||
|
accountPlaceholder1: '用户名 admin 或不输均为 common',
|
||||||
|
accountPlaceholder2: '密码:123456',
|
||||||
|
accountPlaceholder3: '请输入验证码',
|
||||||
|
accountBtnText: '登 录',
|
||||||
|
},
|
||||||
|
mobile: {
|
||||||
|
placeholder1: '请输入手机号',
|
||||||
|
placeholder2: '请输入验证码',
|
||||||
|
codeText: '获取验证码',
|
||||||
|
btnText: '登 录',
|
||||||
|
msgText: '* 温馨提示:建议使用谷歌、Microsoft Edge,版本 79.0.1072.62 及以上浏览器,360浏览器请使用极速模式',
|
||||||
|
},
|
||||||
|
scan: {
|
||||||
|
text: '打开手机扫一扫,快速登录/注册',
|
||||||
|
},
|
||||||
|
signInText: '欢迎回来!',
|
||||||
|
};
|
||||||
28
web/src/i18n/pages/login/zh-tw.ts
Normal file
28
web/src/i18n/pages/login/zh-tw.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// 定义内容
|
||||||
|
export default {
|
||||||
|
label: {
|
||||||
|
one1: '用戶名登入',
|
||||||
|
two2: '手機號登入',
|
||||||
|
},
|
||||||
|
link: {
|
||||||
|
one3: '協力廠商登入',
|
||||||
|
two4: '友情連結',
|
||||||
|
},
|
||||||
|
account: {
|
||||||
|
accountPlaceholder1: '用戶名admin或不輸均為common',
|
||||||
|
accountPlaceholder2: '密碼:123456',
|
||||||
|
accountPlaceholder3: '請輸入驗證碼',
|
||||||
|
accountBtnText: '登入',
|
||||||
|
},
|
||||||
|
mobile: {
|
||||||
|
placeholder1: '請輸入手機號',
|
||||||
|
placeholder2: '請輸入驗證碼',
|
||||||
|
codeText: '獲取驗證碼',
|
||||||
|
btnText: '登入',
|
||||||
|
msgText: '* 溫馨提示:建議使用穀歌、Microsoft Edge,版本79.0.1072.62及以上瀏覽器,360瀏覽器請使用極速模式',
|
||||||
|
},
|
||||||
|
scan: {
|
||||||
|
text: '打開手機掃一掃,快速登錄/注册',
|
||||||
|
},
|
||||||
|
signInText: '歡迎回來!',
|
||||||
|
};
|
||||||
163
web/src/layout/component/aside.vue
Normal file
163
web/src/layout/component/aside.vue
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
<template>
|
||||||
|
<div class="h100" v-show="!isTagsViewCurrenFull">
|
||||||
|
<el-aside class="layout-aside" :class="setCollapseStyle">
|
||||||
|
<Logo v-if="setShowLogo" />
|
||||||
|
<el-scrollbar class="flex-auto" ref="layoutAsideScrollbarRef" @mouseenter="onAsideEnterLeave(true)" @mouseleave="onAsideEnterLeave(false)">
|
||||||
|
<Vertical :menuList="menuList" />
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-aside>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, computed, watch, getCurrentInstance, onBeforeMount, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import Logo from '/@/layout/logo/index.vue';
|
||||||
|
import Vertical from '/@/layout/navMenu/vertical.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutAside',
|
||||||
|
components: { Logo, Vertical },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { isTagsViewCurrenFull } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
const state = reactive({
|
||||||
|
menuList: [],
|
||||||
|
clientWidth: 0,
|
||||||
|
});
|
||||||
|
// 设置菜单展开/收起时的宽度
|
||||||
|
const setCollapseStyle = computed(() => {
|
||||||
|
const { layout, isCollapse, menuBar } = themeConfig.value;
|
||||||
|
const asideBrTheme = ['#FFFFFF', '#FFF', '#fff', '#ffffff'];
|
||||||
|
const asideBrColor = asideBrTheme.includes(menuBar) ? 'layout-el-aside-br-color' : '';
|
||||||
|
// 判断是否是手机端
|
||||||
|
if (state.clientWidth <= 1000) {
|
||||||
|
if (isCollapse) {
|
||||||
|
document.body.setAttribute('class', 'el-popup-parent--hidden');
|
||||||
|
const asideEle = document.querySelector('.layout-container') as HTMLElement;
|
||||||
|
const modeDivs = document.createElement('div');
|
||||||
|
modeDivs.setAttribute('class', 'layout-aside-mobile-mode');
|
||||||
|
asideEle.appendChild(modeDivs);
|
||||||
|
modeDivs.addEventListener('click', closeLayoutAsideMobileMode);
|
||||||
|
return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-open'];
|
||||||
|
} else {
|
||||||
|
// 关闭弹窗
|
||||||
|
closeLayoutAsideMobileMode();
|
||||||
|
return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-close'];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (layout === 'columns') {
|
||||||
|
// 分栏布局,菜单收起时宽度给 1px
|
||||||
|
if (isCollapse) return [asideBrColor, 'layout-aside-pc-1'];
|
||||||
|
else return [asideBrColor, 'layout-aside-pc-220'];
|
||||||
|
} else {
|
||||||
|
// 其它布局给 64px
|
||||||
|
if (isCollapse) return [asideBrColor, 'layout-aside-pc-64'];
|
||||||
|
else return [asideBrColor, 'layout-aside-pc-220'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 关闭移动端蒙版
|
||||||
|
const closeLayoutAsideMobileMode = () => {
|
||||||
|
const el = document.querySelector('.layout-aside-mobile-mode');
|
||||||
|
el?.setAttribute('style', 'animation: error-img-two 0.3s');
|
||||||
|
setTimeout(() => {
|
||||||
|
el?.parentNode?.removeChild(el);
|
||||||
|
}, 300);
|
||||||
|
const clientWidth = document.body.clientWidth;
|
||||||
|
if (clientWidth < 1000) themeConfig.value.isCollapse = false;
|
||||||
|
document.body.setAttribute('class', '');
|
||||||
|
};
|
||||||
|
// 设置显示/隐藏 logo
|
||||||
|
const setShowLogo = computed(() => {
|
||||||
|
let { layout, isShowLogo } = themeConfig.value;
|
||||||
|
return (isShowLogo && layout === 'defaults') || (isShowLogo && layout === 'columns');
|
||||||
|
});
|
||||||
|
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
||||||
|
const setFilterRoutes = () => {
|
||||||
|
if (themeConfig.value.layout === 'columns') return false;
|
||||||
|
(state.menuList as any) = filterRoutesFun(routesList.value);
|
||||||
|
};
|
||||||
|
// 路由过滤递归函数
|
||||||
|
const filterRoutesFun = (arr: Array<string>) => {
|
||||||
|
return arr
|
||||||
|
.filter((item: any) => !item.meta.isHide)
|
||||||
|
.map((item: any) => {
|
||||||
|
item = Object.assign({}, item);
|
||||||
|
if (item.children) item.children = filterRoutesFun(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 设置菜单导航是否固定(移动端)
|
||||||
|
const initMenuFixed = (clientWidth: number) => {
|
||||||
|
state.clientWidth = clientWidth;
|
||||||
|
};
|
||||||
|
// 鼠标移入、移出
|
||||||
|
const onAsideEnterLeave = (bool: Boolean) => {
|
||||||
|
let { layout } = themeConfig.value;
|
||||||
|
if (layout !== 'columns') return false;
|
||||||
|
if (!bool) proxy.mittBus.emit('restoreDefault');
|
||||||
|
stores.setColumnsMenuHover(bool);
|
||||||
|
};
|
||||||
|
// 监听 themeConfig 配置文件的变化,更新菜单 el-scrollbar 的高度
|
||||||
|
watch(themeConfig.value, (val) => {
|
||||||
|
if (val.isShowLogoChange !== val.isShowLogo) {
|
||||||
|
if (!proxy.$refs.layoutAsideScrollbarRef) return false;
|
||||||
|
proxy.$refs.layoutAsideScrollbarRef.update();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 监听vuex值的变化,动态赋值给菜单中
|
||||||
|
watch(
|
||||||
|
pinia.state,
|
||||||
|
(val) => {
|
||||||
|
let { layout, isClassicSplitMenu } = val.themeConfig.themeConfig;
|
||||||
|
if (layout === 'classic' && isClassicSplitMenu) return false;
|
||||||
|
setFilterRoutes();
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// 页面加载前
|
||||||
|
onBeforeMount(() => {
|
||||||
|
initMenuFixed(document.body.clientWidth);
|
||||||
|
setFilterRoutes();
|
||||||
|
// 此界面不需要取消监听(proxy.mittBus.off('setSendColumnsChildren))
|
||||||
|
// 因为切换布局时有的监听需要使用,取消了监听,某些操作将不生效
|
||||||
|
proxy.mittBus.on('setSendColumnsChildren', (res: any) => {
|
||||||
|
state.menuList = res.children;
|
||||||
|
});
|
||||||
|
proxy.mittBus.on('setSendClassicChildren', (res: any) => {
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
if (layout === 'classic' && isClassicSplitMenu) {
|
||||||
|
state.menuList = [];
|
||||||
|
state.menuList = res.children;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
proxy.mittBus.on('getBreadcrumbIndexSetFilterRoutes', () => {
|
||||||
|
setFilterRoutes();
|
||||||
|
});
|
||||||
|
proxy.mittBus.on('layoutMobileResize', (res: any) => {
|
||||||
|
initMenuFixed(res.clientWidth);
|
||||||
|
closeLayoutAsideMobileMode();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
setCollapseStyle,
|
||||||
|
setShowLogo,
|
||||||
|
isTagsViewCurrenFull,
|
||||||
|
onAsideEnterLeave,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
292
web/src/layout/component/columnsAside.vue
Normal file
292
web/src/layout/component/columnsAside.vue
Normal file
@@ -0,0 +1,292 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-columns-aside">
|
||||||
|
<el-scrollbar>
|
||||||
|
<ul @mouseleave="onColumnsAsideMenuMouseleave()">
|
||||||
|
<li
|
||||||
|
v-for="(v, k) in columnsAsideList"
|
||||||
|
:key="k"
|
||||||
|
@click="onColumnsAsideMenuClick(v, k)"
|
||||||
|
@mouseenter="onColumnsAsideMenuMouseenter(v, k)"
|
||||||
|
:ref="
|
||||||
|
(el) => {
|
||||||
|
if (el) columnsAsideOffsetTopRefs[k] = el;
|
||||||
|
}
|
||||||
|
"
|
||||||
|
:class="{ 'layout-columns-active': liIndex === k, 'layout-columns-hover': liHoverIndex === k }"
|
||||||
|
:title="$t(v.meta.title)"
|
||||||
|
>
|
||||||
|
<div :class="themeConfig.columnsAsideLayout" v-if="!v.meta.isLink || (v.meta.isLink && v.meta.isIframe)">
|
||||||
|
<SvgIcon :name="v.meta.icon" />
|
||||||
|
<div class="columns-vertical-title font12">
|
||||||
|
{{
|
||||||
|
$t(v.meta.title) && $t(v.meta.title).length >= 4
|
||||||
|
? $t(v.meta.title).substr(0, themeConfig.columnsAsideLayout === 'columns-vertical' ? 4 : 3)
|
||||||
|
: $t(v.meta.title)
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div :class="themeConfig.columnsAsideLayout" v-else>
|
||||||
|
<a :href="v.meta.isLink" target="_blank">
|
||||||
|
<SvgIcon :name="v.meta.icon" />
|
||||||
|
<div class="columns-vertical-title font12">
|
||||||
|
{{
|
||||||
|
$t(v.meta.title) && $t(v.meta.title).length >= 4
|
||||||
|
? $t(v.meta.title).substr(0, themeConfig.columnsAsideLayout === 'columns-vertical' ? 4 : 3)
|
||||||
|
: $t(v.meta.title)
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<div ref="columnsAsideActiveRef" :class="themeConfig.columnsAsideStyle"></div>
|
||||||
|
</ul>
|
||||||
|
</el-scrollbar>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { reactive, toRefs, ref, onMounted, nextTick, getCurrentInstance, watch, onUnmounted, defineComponent } from 'vue';
|
||||||
|
import { useRoute, useRouter, onBeforeRouteUpdate, RouteRecordRaw } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface ColumnsAsideState {
|
||||||
|
columnsAsideList: any[];
|
||||||
|
liIndex: number;
|
||||||
|
liOldIndex: null | number;
|
||||||
|
liHoverIndex: null | number;
|
||||||
|
liOldPath: null | string;
|
||||||
|
difference: number;
|
||||||
|
routeSplit: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutColumnsAside',
|
||||||
|
setup() {
|
||||||
|
const columnsAsideOffsetTopRefs: any = ref([]);
|
||||||
|
const columnsAsideActiveRef = ref();
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { routesList, isColumnsMenuHover, isColumnsNavHover } = storeToRefs(stores);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<ColumnsAsideState>({
|
||||||
|
columnsAsideList: [],
|
||||||
|
liIndex: 0,
|
||||||
|
liOldIndex: null,
|
||||||
|
liHoverIndex: null,
|
||||||
|
liOldPath: null,
|
||||||
|
difference: 0,
|
||||||
|
routeSplit: [],
|
||||||
|
});
|
||||||
|
// 设置菜单高亮位置移动
|
||||||
|
const setColumnsAsideMove = (k: number) => {
|
||||||
|
state.liIndex = k;
|
||||||
|
columnsAsideActiveRef.value.style.top = `${columnsAsideOffsetTopRefs.value[k].offsetTop + state.difference}px`;
|
||||||
|
};
|
||||||
|
// 菜单高亮点击事件
|
||||||
|
const onColumnsAsideMenuClick = (v: Object, k: number) => {
|
||||||
|
setColumnsAsideMove(k);
|
||||||
|
let { path, redirect } = v as any;
|
||||||
|
if (redirect) router.push(redirect);
|
||||||
|
else router.push(path);
|
||||||
|
};
|
||||||
|
// 鼠标移入时,显示当前的子级菜单
|
||||||
|
const onColumnsAsideMenuMouseenter = (v: RouteRecordRaw, k: number) => {
|
||||||
|
let { path } = v;
|
||||||
|
state.liOldPath = path;
|
||||||
|
state.liOldIndex = k;
|
||||||
|
state.liHoverIndex = k;
|
||||||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(path));
|
||||||
|
stores.setColumnsMenuHover(false);
|
||||||
|
stores.setColumnsNavHover(true);
|
||||||
|
};
|
||||||
|
// 鼠标移走时,显示原来的子级菜单
|
||||||
|
const onColumnsAsideMenuMouseleave = async () => {
|
||||||
|
await stores.setColumnsNavHover(false);
|
||||||
|
// 添加延时器,防止拿到的 store.state.routesList 值不是最新的
|
||||||
|
setTimeout(() => {
|
||||||
|
if (!isColumnsMenuHover && !isColumnsNavHover) proxy.mittBus.emit('restoreDefault');
|
||||||
|
}, 100);
|
||||||
|
};
|
||||||
|
// 设置高亮动态位置
|
||||||
|
const onColumnsAsideDown = (k: number) => {
|
||||||
|
nextTick(() => {
|
||||||
|
setColumnsAsideMove(k);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
||||||
|
const setFilterRoutes = () => {
|
||||||
|
state.columnsAsideList = filterRoutesFun(routesList.value);
|
||||||
|
const resData: any = setSendChildren(route.path);
|
||||||
|
if (Object.keys(resData).length <= 0) return false;
|
||||||
|
onColumnsAsideDown(resData.item[0].k);
|
||||||
|
proxy.mittBus.emit('setSendColumnsChildren', resData);
|
||||||
|
};
|
||||||
|
// 传送当前子级数据到菜单中
|
||||||
|
const setSendChildren = (path: string) => {
|
||||||
|
const currentPathSplit = path.split('/');
|
||||||
|
let currentData: any = {};
|
||||||
|
state.columnsAsideList.map((v: any, k: number) => {
|
||||||
|
if (v.path === `/${currentPathSplit[1]}`) {
|
||||||
|
v['k'] = k;
|
||||||
|
currentData['item'] = [{ ...v }];
|
||||||
|
currentData['children'] = [{ ...v }];
|
||||||
|
if (v.children) currentData['children'] = v.children;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return currentData;
|
||||||
|
};
|
||||||
|
// 路由过滤递归函数
|
||||||
|
const filterRoutesFun = (arr: Array<string>) => {
|
||||||
|
return arr
|
||||||
|
.filter((item: any) => !item.meta.isHide)
|
||||||
|
.map((item: any) => {
|
||||||
|
item = Object.assign({}, item);
|
||||||
|
if (item.children) item.children = filterRoutesFun(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// tagsView 点击时,根据路由查找下标 columnsAsideList,实现左侧菜单高亮
|
||||||
|
const setColumnsMenuHighlight = (path: string) => {
|
||||||
|
state.routeSplit = path.split('/');
|
||||||
|
state.routeSplit.shift();
|
||||||
|
const routeFirst = `/${state.routeSplit[0]}`;
|
||||||
|
const currentSplitRoute = state.columnsAsideList.find((v: any) => v.path === routeFirst);
|
||||||
|
if (!currentSplitRoute) return false;
|
||||||
|
// 延迟拿值,防止取不到
|
||||||
|
setTimeout(() => {
|
||||||
|
onColumnsAsideDown((<any>currentSplitRoute).k);
|
||||||
|
}, 0);
|
||||||
|
};
|
||||||
|
// 监听布局配置信息的变化,动态增加菜单高亮位置移动像素
|
||||||
|
watch(
|
||||||
|
pinia.state,
|
||||||
|
(val) => {
|
||||||
|
val.themeConfig.themeConfig.columnsAsideStyle === 'columnsRound' ? (state.difference = 3) : (state.difference = 0);
|
||||||
|
if (!val.routesList.isColumnsMenuHover && !val.routesList.isColumnsNavHover) {
|
||||||
|
state.liHoverIndex = null;
|
||||||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(route.path));
|
||||||
|
} else {
|
||||||
|
state.liHoverIndex = state.liOldIndex;
|
||||||
|
if (!state.liOldPath) return false;
|
||||||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(state.liOldPath));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
setFilterRoutes();
|
||||||
|
// 销毁变量,防止鼠标再次移入时,保留了上次的记录
|
||||||
|
proxy.mittBus.on('restoreDefault', () => {
|
||||||
|
state.liOldIndex = null;
|
||||||
|
state.liOldPath = null;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
proxy.mittBus.off('restoreDefault', () => {});
|
||||||
|
});
|
||||||
|
// 路由更新时
|
||||||
|
onBeforeRouteUpdate((to) => {
|
||||||
|
setColumnsMenuHighlight(to.path);
|
||||||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(to.path));
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
themeConfig,
|
||||||
|
columnsAsideOffsetTopRefs,
|
||||||
|
columnsAsideActiveRef,
|
||||||
|
onColumnsAsideDown,
|
||||||
|
onColumnsAsideMenuClick,
|
||||||
|
onColumnsAsideMenuMouseenter,
|
||||||
|
onColumnsAsideMenuMouseleave,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-columns-aside {
|
||||||
|
width: 70px;
|
||||||
|
height: 100%;
|
||||||
|
background: var(--next-bg-columnsMenuBar);
|
||||||
|
ul {
|
||||||
|
position: relative;
|
||||||
|
li {
|
||||||
|
color: var(--next-bg-columnsMenuBarColor);
|
||||||
|
width: 100%;
|
||||||
|
height: 50px;
|
||||||
|
text-align: center;
|
||||||
|
display: flex;
|
||||||
|
cursor: pointer;
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
.columns-vertical {
|
||||||
|
margin: auto;
|
||||||
|
.columns-vertical-title {
|
||||||
|
padding-top: 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.columns-horizontal {
|
||||||
|
display: flex;
|
||||||
|
height: 50px;
|
||||||
|
width: 100%;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 5px;
|
||||||
|
i {
|
||||||
|
margin-right: 3px;
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
display: flex;
|
||||||
|
.columns-horizontal-title {
|
||||||
|
padding-top: 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
text-decoration: none;
|
||||||
|
color: var(--next-bg-columnsMenuBarColor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-columns-active {
|
||||||
|
color: var(--next-bg-columnsMenuBarColor) !important;
|
||||||
|
transition: 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
.layout-columns-hover {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
a {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.columns-round {
|
||||||
|
background: var(--el-color-primary);
|
||||||
|
color: var(--el-color-white);
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
top: 2px;
|
||||||
|
height: 44px;
|
||||||
|
width: 65px;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
z-index: 0;
|
||||||
|
transition: 0.3s ease-in-out;
|
||||||
|
border-radius: 5px;
|
||||||
|
}
|
||||||
|
.columns-card {
|
||||||
|
@extend .columns-round;
|
||||||
|
top: 0;
|
||||||
|
height: 50px;
|
||||||
|
width: 100%;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
34
web/src/layout/component/header.vue
Normal file
34
web/src/layout/component/header.vue
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<template>
|
||||||
|
<el-header class="layout-header" :height="setHeaderHeight" v-show="!isTagsViewCurrenFull">
|
||||||
|
<NavBarsIndex />
|
||||||
|
</el-header>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import NavBarsIndex from '/@/layout/navBars/index.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutHeader',
|
||||||
|
components: { NavBarsIndex },
|
||||||
|
setup() {
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { isTagsViewCurrenFull } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
// 设置 header 的高度
|
||||||
|
const setHeaderHeight = computed(() => {
|
||||||
|
let { isTagsview, layout } = themeConfig.value;
|
||||||
|
if (isTagsview && layout !== 'classic') return '84px';
|
||||||
|
else return '50px';
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
setHeaderHeight,
|
||||||
|
isTagsViewCurrenFull,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
101
web/src/layout/component/main.vue
Normal file
101
web/src/layout/component/main.vue
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
<template>
|
||||||
|
<el-main class="layout-main">
|
||||||
|
<el-scrollbar
|
||||||
|
ref="layoutScrollbarRef"
|
||||||
|
:class="{
|
||||||
|
'layout-scrollbar':
|
||||||
|
(!isClassicOrTransverse && !currentRouteMeta.isLink && !currentRouteMeta.isIframe) ||
|
||||||
|
(!isClassicOrTransverse && currentRouteMeta.isLink && !currentRouteMeta.isIframe),
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<LayoutParentView
|
||||||
|
:style="{
|
||||||
|
padding: !isClassicOrTransverse || (currentRouteMeta.isLink && currentRouteMeta.isIframe) ? '0' : '15px',
|
||||||
|
transition: 'padding 0.3s ease-in-out',
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
<Footer v-if="themeConfig.isFooter" />
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-main>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent, toRefs, reactive, getCurrentInstance, watch, onMounted, computed } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { NextLoading } from '/@/utils/loading';
|
||||||
|
import LayoutParentView from '/@/layout/routerView/parent.vue';
|
||||||
|
import Footer from '/@/layout/footer/index.vue';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface MainState {
|
||||||
|
headerHeight: string | number;
|
||||||
|
currentRouteMeta: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutMain',
|
||||||
|
components: { LayoutParentView, Footer },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive<MainState>({
|
||||||
|
headerHeight: '',
|
||||||
|
currentRouteMeta: {},
|
||||||
|
});
|
||||||
|
// 判断布局
|
||||||
|
const isClassicOrTransverse = computed(() => {
|
||||||
|
const { layout } = themeConfig.value;
|
||||||
|
return layout === 'classic' || layout === 'transverse';
|
||||||
|
});
|
||||||
|
// 设置 main 的高度
|
||||||
|
const initHeaderHeight = () => {
|
||||||
|
const bool = state.currentRouteMeta.isLink && state.currentRouteMeta.isIframe;
|
||||||
|
let { isTagsview } = themeConfig.value;
|
||||||
|
if (isTagsview) return (state.headerHeight = bool ? `86px` : `115px`);
|
||||||
|
else return (state.headerHeight = `80px`);
|
||||||
|
};
|
||||||
|
// 初始化获取当前路由 meta,用于设置 iframes padding
|
||||||
|
const initGetMeta = () => {
|
||||||
|
state.currentRouteMeta = route.meta;
|
||||||
|
};
|
||||||
|
// 页面加载前
|
||||||
|
onMounted(async () => {
|
||||||
|
await initGetMeta();
|
||||||
|
initHeaderHeight();
|
||||||
|
NextLoading.done();
|
||||||
|
});
|
||||||
|
// 监听路由变化
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
state.currentRouteMeta = route.meta;
|
||||||
|
const bool = state.currentRouteMeta.isLink && state.currentRouteMeta.isIframe;
|
||||||
|
state.headerHeight = bool ? `86px` : `115px`;
|
||||||
|
proxy.$refs.layoutScrollbarRef.update();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// 监听 themeConfig 配置文件的变化,更新菜单 el-scrollbar 的高度
|
||||||
|
watch(
|
||||||
|
themeConfig,
|
||||||
|
(val) => {
|
||||||
|
state.currentRouteMeta = route.meta;
|
||||||
|
const bool = state.currentRouteMeta.isLink && state.currentRouteMeta.isIframe;
|
||||||
|
state.headerHeight = val.isTagsview ? (bool ? `86px` : `115px`) : '51px';
|
||||||
|
proxy.$refs?.layoutScrollbarRef?.update();
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
themeConfig,
|
||||||
|
isClassicOrTransverse,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
47
web/src/layout/footer/index.vue
Normal file
47
web/src/layout/footer/index.vue
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-footer mt15" v-show="isDelayFooter">
|
||||||
|
<div class="layout-footer-warp">
|
||||||
|
<div>vue-next-admin,Made by lyt with ❤️</div>
|
||||||
|
<div class="mt5">深圳市 xxx 公司版权所有</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, defineComponent } from 'vue';
|
||||||
|
import { onBeforeRouteUpdate } from 'vue-router';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutFooter',
|
||||||
|
setup() {
|
||||||
|
const state = reactive({
|
||||||
|
isDelayFooter: true,
|
||||||
|
});
|
||||||
|
// 路由改变时,等主界面动画加载完毕再显示 footer
|
||||||
|
onBeforeRouteUpdate(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
state.isDelayFooter = false;
|
||||||
|
setTimeout(() => {
|
||||||
|
state.isDelayFooter = true;
|
||||||
|
}, 800);
|
||||||
|
}, 0);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-footer {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
&-warp {
|
||||||
|
margin: auto;
|
||||||
|
color: var(--el-text-color-secondary);
|
||||||
|
text-align: center;
|
||||||
|
animation: error-num 1s ease-in-out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
54
web/src/layout/index.vue
Normal file
54
web/src/layout/index.vue
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
<template>
|
||||||
|
<component :is="themeConfig.layout" />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { onBeforeMount, onUnmounted, getCurrentInstance, defineComponent, defineAsyncComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layout',
|
||||||
|
components: {
|
||||||
|
defaults: defineAsyncComponent(() => import('/@/layout/main/defaults.vue')),
|
||||||
|
classic: defineAsyncComponent(() => import('/@/layout/main/classic.vue')),
|
||||||
|
transverse: defineAsyncComponent(() => import('/@/layout/main/transverse.vue')),
|
||||||
|
columns: defineAsyncComponent(() => import('/@/layout/main/columns.vue')),
|
||||||
|
},
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
// 窗口大小改变时(适配移动端)
|
||||||
|
const onLayoutResize = () => {
|
||||||
|
if (!Local.get('oldLayout')) Local.set('oldLayout', themeConfig.value.layout);
|
||||||
|
const clientWidth = document.body.clientWidth;
|
||||||
|
if (clientWidth < 1000) {
|
||||||
|
themeConfig.value.isCollapse = false;
|
||||||
|
proxy.mittBus.emit('layoutMobileResize', {
|
||||||
|
layout: 'defaults',
|
||||||
|
clientWidth,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
proxy.mittBus.emit('layoutMobileResize', {
|
||||||
|
layout: Local.get('oldLayout') ? Local.get('oldLayout') : themeConfig.value.layout,
|
||||||
|
clientWidth,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 页面加载前
|
||||||
|
onBeforeMount(() => {
|
||||||
|
onLayoutResize();
|
||||||
|
window.addEventListener('resize', onLayoutResize);
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
window.removeEventListener('resize', onLayoutResize);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
themeConfig,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
372
web/src/layout/lockScreen/index.vue
Normal file
372
web/src/layout/lockScreen/index.vue
Normal file
@@ -0,0 +1,372 @@
|
|||||||
|
<template>
|
||||||
|
<div v-show="isShowLockScreen">
|
||||||
|
<div class="layout-lock-screen-mask"></div>
|
||||||
|
<div class="layout-lock-screen-img" :class="{ 'layout-lock-screen-filter': isShowLoockLogin }"></div>
|
||||||
|
<div class="layout-lock-screen">
|
||||||
|
<div
|
||||||
|
class="layout-lock-screen-date"
|
||||||
|
ref="layoutLockScreenDateRef"
|
||||||
|
@mousedown="onDown"
|
||||||
|
@mousemove="onMove"
|
||||||
|
@mouseup="onEnd"
|
||||||
|
@touchstart.stop="onDown"
|
||||||
|
@touchmove.stop="onMove"
|
||||||
|
@touchend.stop="onEnd"
|
||||||
|
>
|
||||||
|
<div class="layout-lock-screen-date-box">
|
||||||
|
<div class="layout-lock-screen-date-box-time">
|
||||||
|
{{ time.hm }}<span class="layout-lock-screen-date-box-minutes">{{ time.s }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="layout-lock-screen-date-box-info">{{ time.mdq }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-lock-screen-date-top">
|
||||||
|
<SvgIcon name="ele-Top" />
|
||||||
|
<div class="layout-lock-screen-date-top-text">上滑解锁</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<transition name="el-zoom-in-center">
|
||||||
|
<div v-show="isShowLoockLogin" class="layout-lock-screen-login">
|
||||||
|
<div class="layout-lock-screen-login-box">
|
||||||
|
<div class="layout-lock-screen-login-box-img">
|
||||||
|
<img src="https://img2.baidu.com/it/u=1978192862,2048448374&fm=253&fmt=auto&app=138&f=JPEG?w=504&h=500" />
|
||||||
|
</div>
|
||||||
|
<div class="layout-lock-screen-login-box-name">Administrator</div>
|
||||||
|
<div class="layout-lock-screen-login-box-value">
|
||||||
|
<el-input
|
||||||
|
placeholder="请输入密码"
|
||||||
|
ref="layoutLockScreenInputRef"
|
||||||
|
v-model="lockScreenPassword"
|
||||||
|
@keyup.enter.native.stop="onLockScreenSubmit()"
|
||||||
|
>
|
||||||
|
<template #append>
|
||||||
|
<el-button @click="onLockScreenSubmit">
|
||||||
|
<el-icon class="el-input__icon">
|
||||||
|
<ele-Right />
|
||||||
|
</el-icon>
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-lock-screen-login-icon">
|
||||||
|
<SvgIcon name="ele-Microphone" :size="20" />
|
||||||
|
<SvgIcon name="ele-AlarmClock" :size="20" />
|
||||||
|
<SvgIcon name="ele-SwitchButton" :size="20" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</transition>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { nextTick, onMounted, reactive, toRefs, ref, onUnmounted, getCurrentInstance, defineComponent } from 'vue';
|
||||||
|
import { formatDate } from '/@/utils/formatTime';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface LockScreenState {
|
||||||
|
transparency: number;
|
||||||
|
downClientY: number;
|
||||||
|
moveDifference: number;
|
||||||
|
isShowLoockLogin: boolean;
|
||||||
|
isFlags: boolean;
|
||||||
|
querySelectorEl: HTMLElement | string;
|
||||||
|
time: {
|
||||||
|
hm: string;
|
||||||
|
s: string;
|
||||||
|
mdq: string;
|
||||||
|
};
|
||||||
|
setIntervalTime: number;
|
||||||
|
isShowLockScreen: boolean;
|
||||||
|
isShowLockScreenIntervalTime: number;
|
||||||
|
lockScreenPassword: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutLockScreen',
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const layoutLockScreenInputRef = ref();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const state = reactive<LockScreenState>({
|
||||||
|
transparency: 1,
|
||||||
|
downClientY: 0,
|
||||||
|
moveDifference: 0,
|
||||||
|
isShowLoockLogin: false,
|
||||||
|
isFlags: false,
|
||||||
|
querySelectorEl: '',
|
||||||
|
time: {
|
||||||
|
hm: '',
|
||||||
|
s: '',
|
||||||
|
mdq: '',
|
||||||
|
},
|
||||||
|
setIntervalTime: 0,
|
||||||
|
isShowLockScreen: false,
|
||||||
|
isShowLockScreenIntervalTime: 0,
|
||||||
|
lockScreenPassword: '',
|
||||||
|
});
|
||||||
|
// 鼠标按下
|
||||||
|
const onDown = (down: any) => {
|
||||||
|
state.isFlags = true;
|
||||||
|
state.downClientY = down.touches ? down.touches[0].clientY : down.clientY;
|
||||||
|
};
|
||||||
|
// 鼠标移动
|
||||||
|
const onMove = (move: any) => {
|
||||||
|
if (state.isFlags) {
|
||||||
|
const el = <HTMLElement>state.querySelectorEl;
|
||||||
|
const opacitys = (state.transparency -= 1 / 200);
|
||||||
|
if (move.touches) {
|
||||||
|
state.moveDifference = move.touches[0].clientY - state.downClientY;
|
||||||
|
} else {
|
||||||
|
state.moveDifference = move.clientY - state.downClientY;
|
||||||
|
}
|
||||||
|
if (state.moveDifference >= 0) return false;
|
||||||
|
el.setAttribute('style', `top:${state.moveDifference}px;cursor:pointer;opacity:${opacitys};`);
|
||||||
|
if (state.moveDifference < -400) {
|
||||||
|
el.setAttribute('style', `top:${-el.clientHeight}px;cursor:pointer;transition:all 0.3s ease;`);
|
||||||
|
state.moveDifference = -el.clientHeight;
|
||||||
|
setTimeout(() => {
|
||||||
|
el && el.parentNode?.removeChild(el);
|
||||||
|
}, 300);
|
||||||
|
}
|
||||||
|
if (state.moveDifference === -el.clientHeight) {
|
||||||
|
state.isShowLoockLogin = true;
|
||||||
|
layoutLockScreenInputRef.value.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 鼠标松开
|
||||||
|
const onEnd = () => {
|
||||||
|
state.isFlags = false;
|
||||||
|
state.transparency = 1;
|
||||||
|
if (state.moveDifference >= -400) {
|
||||||
|
(<HTMLElement>state.querySelectorEl).setAttribute('style', `top:0px;opacity:1;transition:all 0.3s ease;`);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 获取要拖拽的初始元素
|
||||||
|
const initGetElement = () => {
|
||||||
|
nextTick(() => {
|
||||||
|
state.querySelectorEl = proxy.$refs.layoutLockScreenDateRef;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 时间初始化
|
||||||
|
const initTime = () => {
|
||||||
|
state.time.hm = formatDate(new Date(), 'HH:MM');
|
||||||
|
state.time.s = formatDate(new Date(), 'SS');
|
||||||
|
state.time.mdq = formatDate(new Date(), 'mm月dd日,WWW');
|
||||||
|
};
|
||||||
|
// 时间初始化定时器
|
||||||
|
const initSetTime = () => {
|
||||||
|
initTime();
|
||||||
|
state.setIntervalTime = window.setInterval(() => {
|
||||||
|
initTime();
|
||||||
|
}, 1000);
|
||||||
|
};
|
||||||
|
// 锁屏时间定时器
|
||||||
|
const initLockScreen = () => {
|
||||||
|
if (themeConfig.value.isLockScreen) {
|
||||||
|
state.isShowLockScreenIntervalTime = window.setInterval(() => {
|
||||||
|
if (themeConfig.value.lockScreenTime <= 1) {
|
||||||
|
state.isShowLockScreen = true;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
themeConfig.value.lockScreenTime--;
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
clearInterval(state.isShowLockScreenIntervalTime);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 存储布局配置
|
||||||
|
const setLocalThemeConfig = () => {
|
||||||
|
themeConfig.value.isDrawer = false;
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
};
|
||||||
|
// 密码输入点击事件
|
||||||
|
const onLockScreenSubmit = () => {
|
||||||
|
themeConfig.value.isLockScreen = false;
|
||||||
|
themeConfig.value.lockScreenTime = 30;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initGetElement();
|
||||||
|
initSetTime();
|
||||||
|
initLockScreen();
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
window.clearInterval(state.setIntervalTime);
|
||||||
|
window.clearInterval(state.isShowLockScreenIntervalTime);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
layoutLockScreenInputRef,
|
||||||
|
onDown,
|
||||||
|
onMove,
|
||||||
|
onEnd,
|
||||||
|
onLockScreenSubmit,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-lock-screen-fixed {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.layout-lock-screen-filter {
|
||||||
|
filter: blur(1px);
|
||||||
|
}
|
||||||
|
.layout-lock-screen-mask {
|
||||||
|
background: var(--el-color-white);
|
||||||
|
@extend .layout-lock-screen-fixed;
|
||||||
|
z-index: 9999990;
|
||||||
|
}
|
||||||
|
.layout-lock-screen-img {
|
||||||
|
@extend .layout-lock-screen-fixed;
|
||||||
|
background-image: url('https://img-blog.csdnimg.cn/afa9c317667f47d5bea34b85af45979e.png#pic_center');
|
||||||
|
background-size: 100% 100%;
|
||||||
|
z-index: 9999991;
|
||||||
|
}
|
||||||
|
.layout-lock-screen {
|
||||||
|
@extend .layout-lock-screen-fixed;
|
||||||
|
z-index: 9999992;
|
||||||
|
&-date {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
color: var(--el-color-white);
|
||||||
|
z-index: 9999993;
|
||||||
|
user-select: none;
|
||||||
|
&-box {
|
||||||
|
position: absolute;
|
||||||
|
left: 30px;
|
||||||
|
bottom: 50px;
|
||||||
|
&-time {
|
||||||
|
font-size: 100px;
|
||||||
|
color: var(--el-color-white);
|
||||||
|
}
|
||||||
|
&-info {
|
||||||
|
font-size: 40px;
|
||||||
|
color: var(--el-color-white);
|
||||||
|
}
|
||||||
|
&-minutes {
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-top {
|
||||||
|
width: 40px;
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
border-radius: 100%;
|
||||||
|
border: 1px solid var(--el-border-color-light, #ebeef5);
|
||||||
|
background: rgba(255, 255, 255, 0.1);
|
||||||
|
color: var(--el-color-white);
|
||||||
|
opacity: 0.8;
|
||||||
|
position: absolute;
|
||||||
|
right: 30px;
|
||||||
|
bottom: 50px;
|
||||||
|
text-align: center;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
i {
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
&-text {
|
||||||
|
opacity: 0;
|
||||||
|
position: absolute;
|
||||||
|
top: 150%;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--el-color-white);
|
||||||
|
left: 50%;
|
||||||
|
line-height: 1.2;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
width: 35px;
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.5);
|
||||||
|
background: rgba(255, 255, 255, 0.2);
|
||||||
|
box-shadow: 0 0 12px 0 rgba(255, 255, 255, 0.5);
|
||||||
|
color: var(--el-color-white);
|
||||||
|
opacity: 1;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
i {
|
||||||
|
transform: translateY(-40px);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
.layout-lock-screen-date-top-text {
|
||||||
|
opacity: 1;
|
||||||
|
top: 50%;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-login {
|
||||||
|
position: relative;
|
||||||
|
z-index: 9999994;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
color: var(--el-color-white);
|
||||||
|
&-box {
|
||||||
|
text-align: center;
|
||||||
|
margin: auto;
|
||||||
|
&-img {
|
||||||
|
width: 180px;
|
||||||
|
height: 180px;
|
||||||
|
margin: auto;
|
||||||
|
img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-name {
|
||||||
|
font-size: 26px;
|
||||||
|
margin: 15px 0 30px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-icon {
|
||||||
|
position: absolute;
|
||||||
|
right: 30px;
|
||||||
|
bottom: 30px;
|
||||||
|
i {
|
||||||
|
font-size: 20px;
|
||||||
|
margin-left: 15px;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.8;
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-input-group__append) {
|
||||||
|
background: var(--el-color-white);
|
||||||
|
padding: 0px 15px;
|
||||||
|
}
|
||||||
|
:deep(.el-input__inner) {
|
||||||
|
border-right-color: var(--el-border-color-extra-light);
|
||||||
|
&:hover {
|
||||||
|
border-color: var(--el-border-color-extra-light);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
85
web/src/layout/logo/index.vue
Normal file
85
web/src/layout/logo/index.vue
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-logo" v-if="setShowLogo" @click="onThemeConfigChange">
|
||||||
|
<img :src="logoMini" class="layout-logo-medium-img" />
|
||||||
|
<span>{{ themeConfig.globalTitle }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="layout-logo-size" v-else @click="onThemeConfigChange">
|
||||||
|
<img :src="logoMini" class="layout-logo-size-img" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
|
||||||
|
import logoMini from '/@/assets/logo-mini.svg';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutLogo',
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
// 设置 logo 的显示。classic 经典布局默认显示 logo
|
||||||
|
const setShowLogo = computed(() => {
|
||||||
|
let { isCollapse, layout } = themeConfig.value;
|
||||||
|
return !isCollapse || layout === 'classic' || document.body.clientWidth < 1000;
|
||||||
|
});
|
||||||
|
// logo 点击实现菜单展开/收起
|
||||||
|
const onThemeConfigChange = () => {
|
||||||
|
if (themeConfig.value.layout === 'transverse') return false;
|
||||||
|
themeConfig.value.isCollapse = !themeConfig.value.isCollapse;
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
logoMini,
|
||||||
|
setShowLogo,
|
||||||
|
themeConfig,
|
||||||
|
onThemeConfigChange,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-logo {
|
||||||
|
width: 220px;
|
||||||
|
height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
box-shadow: rgb(0 21 41 / 2%) 0px 1px 4px;
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
font-size: 16px;
|
||||||
|
cursor: pointer;
|
||||||
|
animation: logoAnimation 0.3s ease-in-out;
|
||||||
|
span {
|
||||||
|
white-space: nowrap;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
span {
|
||||||
|
color: var(--color-primary-light-2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-medium-img {
|
||||||
|
width: 20px;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-logo-size {
|
||||||
|
width: 100%;
|
||||||
|
height: 50px;
|
||||||
|
display: flex;
|
||||||
|
cursor: pointer;
|
||||||
|
animation: logoAnimation 0.3s ease-in-out;
|
||||||
|
&-img {
|
||||||
|
width: 20px;
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
img {
|
||||||
|
animation: logoAnimation 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
35
web/src/layout/main/classic.vue
Normal file
35
web/src/layout/main/classic.vue
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<template>
|
||||||
|
<el-container class="layout-container flex-center">
|
||||||
|
<Header />
|
||||||
|
<el-container class="layout-mian-height-50">
|
||||||
|
<Aside />
|
||||||
|
<div class="flex-center layout-backtop">
|
||||||
|
<TagsView v-if="themeConfig.isTagsview" />
|
||||||
|
<Main style="padding: 10px !important" />
|
||||||
|
</div>
|
||||||
|
</el-container>
|
||||||
|
<el-backtop target=".layout-backtop .el-main .el-scrollbar__wrap"></el-backtop>
|
||||||
|
</el-container>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import Aside from '/@/layout/component/aside.vue';
|
||||||
|
import Header from '/@/layout/component/header.vue';
|
||||||
|
import Main from '/@/layout/component/main.vue';
|
||||||
|
import TagsView from '/@/layout/navBars/tagsView/tagsView.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutClassic',
|
||||||
|
components: { Aside, Header, Main, TagsView },
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
return {
|
||||||
|
themeConfig,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
41
web/src/layout/main/columns.vue
Normal file
41
web/src/layout/main/columns.vue
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<template>
|
||||||
|
<el-container class="layout-container">
|
||||||
|
<ColumnsAside />
|
||||||
|
<div class="layout-columns-warp">
|
||||||
|
<Aside />
|
||||||
|
<el-container class="flex-center layout-backtop" :class="{ 'layout-backtop': !isFixedHeader }">
|
||||||
|
<Header v-if="isFixedHeader" />
|
||||||
|
<el-scrollbar :class="{ 'layout-backtop': isFixedHeader }">
|
||||||
|
<Header v-if="!isFixedHeader" />
|
||||||
|
<Main style="padding: 10px !important;height: 100vh" />
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-container>
|
||||||
|
</div>
|
||||||
|
<el-backtop target=".layout-backtop .el-scrollbar__wrap"></el-backtop>
|
||||||
|
</el-container>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import Aside from '/@/layout/component/aside.vue';
|
||||||
|
import Header from '/@/layout/component/header.vue';
|
||||||
|
import Main from '/@/layout/component/main.vue';
|
||||||
|
import ColumnsAside from '/@/layout/component/columnsAside.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutColumns',
|
||||||
|
components: { Aside, Header, Main, ColumnsAside },
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const isFixedHeader = computed(() => {
|
||||||
|
return themeConfig.value.isFixedHeader;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
isFixedHeader,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
47
web/src/layout/main/defaults.vue
Normal file
47
web/src/layout/main/defaults.vue
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<template>
|
||||||
|
<el-container class="layout-container">
|
||||||
|
<Aside />
|
||||||
|
<el-container class="flex-center" :class="{ 'layout-backtop': !isFixedHeader }">
|
||||||
|
<Header v-if="isFixedHeader" />
|
||||||
|
<el-scrollbar ref="layoutDefaultsScrollbarRef" :class="{ 'layout-backtop': isFixedHeader }">
|
||||||
|
<Header v-if="!isFixedHeader" />
|
||||||
|
<Main style="padding: 10px !important;height: 100vh" />
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-container>
|
||||||
|
<el-backtop target=".layout-backtop .el-scrollbar__wrap"></el-backtop>
|
||||||
|
</el-container>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, getCurrentInstance, watch, defineComponent } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import Aside from '/@/layout/component/aside.vue';
|
||||||
|
import Header from '/@/layout/component/header.vue';
|
||||||
|
import Main from '/@/layout/component/main.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutDefaults',
|
||||||
|
components: { Aside, Header, Main },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const route = useRoute();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const isFixedHeader = computed(() => {
|
||||||
|
return themeConfig.value.isFixedHeader;
|
||||||
|
});
|
||||||
|
// 监听路由的变化
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
proxy.$refs.layoutDefaultsScrollbarRef.wrap$.scrollTop = 0;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
isFixedHeader,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
17
web/src/layout/main/transverse.vue
Normal file
17
web/src/layout/main/transverse.vue
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<template>
|
||||||
|
<el-container class="layout-container flex-center layout-backtop">
|
||||||
|
<Header />
|
||||||
|
<Main style="padding: 10px !important" />
|
||||||
|
<el-backtop target=".layout-backtop .el-main .el-scrollbar__wrap"></el-backtop>
|
||||||
|
</el-container>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import Header from '/@/layout/component/header.vue';
|
||||||
|
import Main from '/@/layout/component/main.vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'layoutTransverse',
|
||||||
|
components: { Header, Main },
|
||||||
|
};
|
||||||
|
</script>
|
||||||
163
web/src/layout/navBars/breadcrumb/breadcrumb.vue
Normal file
163
web/src/layout/navBars/breadcrumb/breadcrumb.vue
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
<template>
|
||||||
|
<div v-if="isShowBreadcrumb" class="layout-navbars-breadcrumb">
|
||||||
|
<SvgIcon
|
||||||
|
class="layout-navbars-breadcrumb-icon"
|
||||||
|
:name="themeConfig.isCollapse ? 'ele-Expand' : 'ele-Fold'"
|
||||||
|
:size="16"
|
||||||
|
@click="onThemeConfigChange"
|
||||||
|
/>
|
||||||
|
<el-breadcrumb class="layout-navbars-breadcrumb-hide">
|
||||||
|
<transition-group name="breadcrumb">
|
||||||
|
<el-breadcrumb-item v-for="(v, k) in breadcrumbList" :key="!v.meta.tagsViewName ? v.meta.title : v.meta.tagsViewName">
|
||||||
|
<span v-if="k === breadcrumbList.length - 1" class="layout-navbars-breadcrumb-span">
|
||||||
|
<SvgIcon :name="v.meta.icon" class="layout-navbars-breadcrumb-iconfont" v-if="themeConfig.isBreadcrumbIcon" />
|
||||||
|
<div v-if="!v.meta.tagsViewName">{{ $t(v.meta.title) }}</div>
|
||||||
|
<div v-else>{{ v.meta.tagsViewName }}</div>
|
||||||
|
</span>
|
||||||
|
<a v-else @click.prevent="onBreadcrumbClick(v)">
|
||||||
|
<SvgIcon :name="v.meta.icon" class="layout-navbars-breadcrumb-iconfont" v-if="themeConfig.isBreadcrumbIcon" />{{ $t(v.meta.title) }}
|
||||||
|
</a>
|
||||||
|
</el-breadcrumb-item>
|
||||||
|
</transition-group>
|
||||||
|
</el-breadcrumb>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, computed, onMounted, defineComponent } from 'vue';
|
||||||
|
import { onBeforeRouteUpdate, useRoute, useRouter } from 'vue-router';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface BreadcrumbState {
|
||||||
|
breadcrumbList: Array<any>;
|
||||||
|
routeSplit: Array<string>;
|
||||||
|
routeSplitFirst: string;
|
||||||
|
routeSplitIndex: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumb',
|
||||||
|
setup() {
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<BreadcrumbState>({
|
||||||
|
breadcrumbList: [],
|
||||||
|
routeSplit: [],
|
||||||
|
routeSplitFirst: '',
|
||||||
|
routeSplitIndex: 1,
|
||||||
|
});
|
||||||
|
// 动态设置经典、横向布局不显示
|
||||||
|
const isShowBreadcrumb = computed(() => {
|
||||||
|
initRouteSplit(route.path);
|
||||||
|
const { layout, isBreadcrumb } = themeConfig.value;
|
||||||
|
if (layout === 'classic' || layout === 'transverse') return false;
|
||||||
|
else return isBreadcrumb ? true : false;
|
||||||
|
});
|
||||||
|
// 面包屑点击时
|
||||||
|
const onBreadcrumbClick = (v: any) => {
|
||||||
|
const { redirect, path } = v;
|
||||||
|
if (redirect) router.push(redirect);
|
||||||
|
else router.push(path);
|
||||||
|
};
|
||||||
|
// 展开/收起左侧菜单点击
|
||||||
|
const onThemeConfigChange = () => {
|
||||||
|
themeConfig.value.isCollapse = !themeConfig.value.isCollapse;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 存储布局配置
|
||||||
|
const setLocalThemeConfig = () => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
};
|
||||||
|
// 处理面包屑数据
|
||||||
|
const getBreadcrumbList = (arr: Array<string>) => {
|
||||||
|
arr.forEach((item: any) => {
|
||||||
|
state.routeSplit.forEach((v: any, k: number, arrs: any) => {
|
||||||
|
if (state.routeSplitFirst === item.path) {
|
||||||
|
state.routeSplitFirst += `/${arrs[state.routeSplitIndex]}`;
|
||||||
|
state.breadcrumbList.push(item);
|
||||||
|
state.routeSplitIndex++;
|
||||||
|
if (item.children) getBreadcrumbList(item.children);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 当前路由字符串切割成数组,并删除第一项空内容
|
||||||
|
const initRouteSplit = (path: string) => {
|
||||||
|
if (!themeConfig.value.isBreadcrumb) return false;
|
||||||
|
state.breadcrumbList = [routesList.value[0]];
|
||||||
|
state.routeSplit = path.split('/');
|
||||||
|
state.routeSplit.shift();
|
||||||
|
state.routeSplitFirst = `/${state.routeSplit[0]}`;
|
||||||
|
state.routeSplitIndex = 1;
|
||||||
|
getBreadcrumbList(routesList.value);
|
||||||
|
if (route.name === 'home' || (route.name === 'notFound' && state.breadcrumbList.length > 1)) state.breadcrumbList.shift();
|
||||||
|
if (state.breadcrumbList.length > 0) state.breadcrumbList[state.breadcrumbList.length - 1].meta.tagsViewName = other.setTagsViewNameI18n(route);
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initRouteSplit(route.path);
|
||||||
|
});
|
||||||
|
// 路由更新时
|
||||||
|
onBeforeRouteUpdate((to) => {
|
||||||
|
initRouteSplit(to.path);
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
onThemeConfigChange,
|
||||||
|
isShowBreadcrumb,
|
||||||
|
themeConfig,
|
||||||
|
onBreadcrumbClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb {
|
||||||
|
flex: 1;
|
||||||
|
height: inherit;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
.layout-navbars-breadcrumb-icon {
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 18px;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
height: 100%;
|
||||||
|
width: 40px;
|
||||||
|
opacity: 0.8;
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-navbars-breadcrumb-span {
|
||||||
|
display: flex;
|
||||||
|
opacity: 0.7;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
.layout-navbars-breadcrumb-iconfont {
|
||||||
|
font-size: 14px;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
:deep(.el-breadcrumb__separator) {
|
||||||
|
opacity: 0.7;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
:deep(.el-breadcrumb__inner a, .el-breadcrumb__inner.is-link) {
|
||||||
|
font-weight: unset !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
&:hover {
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
61
web/src/layout/navBars/breadcrumb/closeFull.vue
Normal file
61
web/src/layout/navBars/breadcrumb/closeFull.vue
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-close-full" v-if="isTagsViewCurrenFull">
|
||||||
|
<div class="layout-navbars-close-full-icon">
|
||||||
|
<SvgIcon name="ele-Close" :title="$t('message.tagsView.closeFullscreen')" @click="onCloseFullscreen" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutCloseFull',
|
||||||
|
setup() {
|
||||||
|
const stores = useTagsViewRoutes();
|
||||||
|
const { isTagsViewCurrenFull } = storeToRefs(stores);
|
||||||
|
// 关闭当前全屏
|
||||||
|
const onCloseFullscreen = () => {
|
||||||
|
stores.setCurrenFullscreen(false);
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
isTagsViewCurrenFull,
|
||||||
|
onCloseFullscreen,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-close-full {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 9999999999;
|
||||||
|
right: -30px;
|
||||||
|
top: -30px;
|
||||||
|
.layout-navbars-close-full-icon {
|
||||||
|
width: 60px;
|
||||||
|
height: 60px;
|
||||||
|
border-radius: 100%;
|
||||||
|
cursor: pointer;
|
||||||
|
background: rgba(0, 0, 0, 0.1);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
position: relative;
|
||||||
|
:deep(i) {
|
||||||
|
position: absolute;
|
||||||
|
left: 10px;
|
||||||
|
top: 35px;
|
||||||
|
color: #333333;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
:deep(i) {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
119
web/src/layout/navBars/breadcrumb/index.vue
Normal file
119
web/src/layout/navBars/breadcrumb/index.vue
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-index">
|
||||||
|
<Logo v-if="setIsShowLogo" />
|
||||||
|
<Breadcrumb />
|
||||||
|
<Horizontal :menuList="menuList" v-if="isLayoutTransverse" />
|
||||||
|
<User />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, reactive, toRefs, onMounted, onUnmounted, getCurrentInstance, defineComponent } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import Breadcrumb from '/@/layout/navBars/breadcrumb/breadcrumb.vue';
|
||||||
|
import User from '/@/layout/navBars/breadcrumb/user.vue';
|
||||||
|
import Logo from '/@/layout/logo/index.vue';
|
||||||
|
import Horizontal from '/@/layout/navMenu/horizontal.vue';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface IndexState {
|
||||||
|
menuList: object[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumbIndex',
|
||||||
|
components: { Breadcrumb, User, Logo, Horizontal },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive<IndexState>({
|
||||||
|
menuList: [],
|
||||||
|
});
|
||||||
|
// 设置 logo 显示/隐藏
|
||||||
|
const setIsShowLogo = computed(() => {
|
||||||
|
let { isShowLogo, layout } = themeConfig.value;
|
||||||
|
return (isShowLogo && layout === 'classic') || (isShowLogo && layout === 'transverse');
|
||||||
|
});
|
||||||
|
// 设置是否显示横向导航菜单
|
||||||
|
const isLayoutTransverse = computed(() => {
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
return layout === 'transverse' || (isClassicSplitMenu && layout === 'classic');
|
||||||
|
});
|
||||||
|
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
||||||
|
const setFilterRoutes = () => {
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
if (layout === 'classic' && isClassicSplitMenu) {
|
||||||
|
state.menuList = delClassicChildren(filterRoutesFun(routesList.value));
|
||||||
|
const resData = setSendClassicChildren(route.path);
|
||||||
|
proxy.mittBus.emit('setSendClassicChildren', resData);
|
||||||
|
} else {
|
||||||
|
state.menuList = filterRoutesFun(routesList.value);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 设置了分割菜单时,删除底下 children
|
||||||
|
const delClassicChildren = (arr: Array<object>) => {
|
||||||
|
arr.map((v: any) => {
|
||||||
|
if (v.children) delete v.children;
|
||||||
|
});
|
||||||
|
return arr;
|
||||||
|
};
|
||||||
|
// 路由过滤递归函数
|
||||||
|
const filterRoutesFun = (arr: Array<string>) => {
|
||||||
|
return arr
|
||||||
|
.filter((item: any) => !item.meta.isHide)
|
||||||
|
.map((item: any) => {
|
||||||
|
item = Object.assign({}, item);
|
||||||
|
if (item.children) item.children = filterRoutesFun(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 传送当前子级数据到菜单中
|
||||||
|
const setSendClassicChildren = (path: string) => {
|
||||||
|
const currentPathSplit = path.split('/');
|
||||||
|
let currentData: any = {};
|
||||||
|
filterRoutesFun(routesList.value).map((v, k) => {
|
||||||
|
if (v.path === `/${currentPathSplit[1]}`) {
|
||||||
|
v['k'] = k;
|
||||||
|
currentData['item'] = [{ ...v }];
|
||||||
|
currentData['children'] = [{ ...v }];
|
||||||
|
if (v.children) currentData['children'] = v.children;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return currentData;
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
setFilterRoutes();
|
||||||
|
proxy.mittBus.on('getBreadcrumbIndexSetFilterRoutes', () => {
|
||||||
|
setFilterRoutes();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
proxy.mittBus.off('getBreadcrumbIndexSetFilterRoutes', () => {});
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
setIsShowLogo,
|
||||||
|
isLayoutTransverse,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-index {
|
||||||
|
height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
background: var(--next-bg-topBar);
|
||||||
|
border-bottom: 1px solid var(--next-border-color-light);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
138
web/src/layout/navBars/breadcrumb/search.vue
Normal file
138
web/src/layout/navBars/breadcrumb/search.vue
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-search-dialog">
|
||||||
|
<el-dialog v-model="isShowSearch" width="300px" destroy-on-close :modal="false" fullscreen :show-close="false">
|
||||||
|
<el-autocomplete
|
||||||
|
v-model="menuQuery"
|
||||||
|
:fetch-suggestions="menuSearch"
|
||||||
|
:placeholder="$t('message.user.searchPlaceholder')"
|
||||||
|
ref="layoutMenuAutocompleteRef"
|
||||||
|
@select="onHandleSelect"
|
||||||
|
@blur="onSearchBlur"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<el-icon class="el-input__icon">
|
||||||
|
<ele-Search />
|
||||||
|
</el-icon>
|
||||||
|
</template>
|
||||||
|
<template #default="{ item }">
|
||||||
|
<div>
|
||||||
|
<SvgIcon :name="item.meta.icon" class="mr5" />
|
||||||
|
{{ $t(item.meta.title) }}
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-autocomplete>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { reactive, toRefs, defineComponent, ref, nextTick } from 'vue';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface SearchState {
|
||||||
|
isShowSearch: boolean;
|
||||||
|
menuQuery: string;
|
||||||
|
tagsViewList: object[];
|
||||||
|
}
|
||||||
|
interface Restaurant {
|
||||||
|
path: string;
|
||||||
|
meta: {
|
||||||
|
title: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumbSearch',
|
||||||
|
setup() {
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const { tagsViewRoutes } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
const layoutMenuAutocompleteRef = ref();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<SearchState>({
|
||||||
|
isShowSearch: false,
|
||||||
|
menuQuery: '',
|
||||||
|
tagsViewList: [],
|
||||||
|
});
|
||||||
|
// 搜索弹窗打开
|
||||||
|
const openSearch = () => {
|
||||||
|
state.menuQuery = '';
|
||||||
|
state.isShowSearch = true;
|
||||||
|
initTageView();
|
||||||
|
nextTick(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
layoutMenuAutocompleteRef.value.focus();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 搜索弹窗关闭
|
||||||
|
const closeSearch = () => {
|
||||||
|
state.isShowSearch = false;
|
||||||
|
};
|
||||||
|
// 菜单搜索数据过滤
|
||||||
|
const menuSearch = (queryString: string, cb: Function) => {
|
||||||
|
let results = queryString ? state.tagsViewList.filter(createFilter(queryString)) : state.tagsViewList;
|
||||||
|
cb(results);
|
||||||
|
};
|
||||||
|
// 菜单搜索过滤
|
||||||
|
const createFilter: any = (queryString: string) => {
|
||||||
|
return (restaurant: Restaurant) => {
|
||||||
|
return (
|
||||||
|
restaurant.path.toLowerCase().indexOf(queryString.toLowerCase()) > -1 ||
|
||||||
|
restaurant.meta.title.toLowerCase().indexOf(queryString.toLowerCase()) > -1 ||
|
||||||
|
t(restaurant.meta.title).indexOf(queryString.toLowerCase()) > -1
|
||||||
|
);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
// 初始化菜单数据
|
||||||
|
const initTageView = () => {
|
||||||
|
if (state.tagsViewList.length > 0) return false;
|
||||||
|
tagsViewRoutes.value.map((v: any) => {
|
||||||
|
if (!v.meta.isHide) state.tagsViewList.push({ ...v });
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 当前菜单选中时
|
||||||
|
const onHandleSelect = (item: any) => {
|
||||||
|
let { path, redirect } = item;
|
||||||
|
if (item.meta.isLink && !item.meta.isIframe) window.open(item.meta.isLink);
|
||||||
|
else if (redirect) router.push(redirect);
|
||||||
|
else router.push(path);
|
||||||
|
closeSearch();
|
||||||
|
};
|
||||||
|
// input 失去焦点时
|
||||||
|
const onSearchBlur = () => {
|
||||||
|
closeSearch();
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
layoutMenuAutocompleteRef,
|
||||||
|
openSearch,
|
||||||
|
closeSearch,
|
||||||
|
menuSearch,
|
||||||
|
onHandleSelect,
|
||||||
|
onSearchBlur,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-search-dialog {
|
||||||
|
:deep(.el-dialog) {
|
||||||
|
box-shadow: unset !important;
|
||||||
|
border-radius: 0 !important;
|
||||||
|
background: rgba(0, 0, 0, 0.5);
|
||||||
|
}
|
||||||
|
:deep(.el-autocomplete) {
|
||||||
|
width: 560px;
|
||||||
|
position: absolute;
|
||||||
|
top: 100px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
816
web/src/layout/navBars/breadcrumb/setings.vue
Normal file
816
web/src/layout/navBars/breadcrumb/setings.vue
Normal file
@@ -0,0 +1,816 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-breadcrumb-seting">
|
||||||
|
<el-drawer
|
||||||
|
:title="$t('message.layout.configTitle')"
|
||||||
|
v-model="getThemeConfig.isDrawer"
|
||||||
|
direction="rtl"
|
||||||
|
destroy-on-close
|
||||||
|
size="260px"
|
||||||
|
@close="onDrawerClose"
|
||||||
|
>
|
||||||
|
<el-scrollbar class="layout-breadcrumb-seting-bar">
|
||||||
|
<!-- 全局主题 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.oneTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">primary</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.primary" size="default" @change="onColorPickerChange"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsDark') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isIsDark" size="small" @change="onAddDarkChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 顶栏设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.twoTopTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoTopBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.topBar" size="default" @change="onBgColorPickerChange('topBar')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoTopBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.topBarColor" size="default" @change="onBgColorPickerChange('topBarColor')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt10">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsTopBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTopBarColorGradual" size="small" @change="onTopBarGradualChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 菜单设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.twoMenuTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoMenuBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.menuBar" size="default" @change="onBgColorPickerChange('menuBar')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoMenuBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.menuBarColor" size="default" @change="onBgColorPickerChange('menuBarColor')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsMenuBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isMenuBarColorGradual" size="small" @change="onMenuBarGradualChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 分栏设置 -->
|
||||||
|
<el-divider content-position="left" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">{{
|
||||||
|
$t('message.layout.twoColumnsTitle')
|
||||||
|
}}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoColumnsMenuBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker
|
||||||
|
v-model="getThemeConfig.columnsMenuBar"
|
||||||
|
size="default"
|
||||||
|
@change="onBgColorPickerChange('columnsMenuBar')"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
>
|
||||||
|
</el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoColumnsMenuBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker
|
||||||
|
v-model="getThemeConfig.columnsMenuBarColor"
|
||||||
|
size="default"
|
||||||
|
@change="onBgColorPickerChange('columnsMenuBarColor')"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
>
|
||||||
|
</el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsColumnsMenuBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isColumnsMenuBarColorGradual"
|
||||||
|
size="small"
|
||||||
|
@change="onColumnsMenuBarGradualChange"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 界面设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.threeTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout === 'transverse' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsCollapse') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isCollapse"
|
||||||
|
:disabled="getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="onThemeConfigChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout === 'transverse' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsUniqueOpened') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isUniqueOpened"
|
||||||
|
:disabled="getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsFixedHeader') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isFixedHeader" size="small" @change="onIsFixedHeaderChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout !== 'classic' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsClassicSplitMenu') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isClassicSplitMenu"
|
||||||
|
:disabled="getThemeConfig.layout !== 'classic'"
|
||||||
|
size="small"
|
||||||
|
@change="onClassicSplitMenuChange"
|
||||||
|
>
|
||||||
|
</el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsLockScreen') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isLockScreen" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt11">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeLockScreenTime') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-input-number
|
||||||
|
v-model="getThemeConfig.lockScreenTime"
|
||||||
|
controls-position="right"
|
||||||
|
:min="1"
|
||||||
|
:max="9999"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
>
|
||||||
|
</el-input-number>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 界面显示 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.fourTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsShowLogo') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isShowLogo" size="small" @change="onIsShowLogoChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="layout-breadcrumb-seting-bar-flex mt15"
|
||||||
|
:style="{ opacity: getThemeConfig.layout === 'classic' || getThemeConfig.layout === 'transverse' ? 0.5 : 1 }"
|
||||||
|
>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsBreadcrumb') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isBreadcrumb"
|
||||||
|
:disabled="getThemeConfig.layout === 'classic' || getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="onIsBreadcrumbChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsBreadcrumbIcon') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isBreadcrumbIcon" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsTagsview') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTagsview" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsTagsviewIcon') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTagsviewIcon" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsCacheTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isCacheTagsView" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: isMobile ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsSortableTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isSortableTagsView"
|
||||||
|
:disabled="isMobile ? true : false"
|
||||||
|
size="small"
|
||||||
|
@change="onSortableTagsViewChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsShareTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isShareTagsView" size="small" @change="onShareTagsViewChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsFooter') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isFooter" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsGrayscale') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isGrayscale" size="small" @change="onAddFilterChange('grayscale')"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsInvert') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isInvert" size="small" @change="onAddFilterChange('invert')"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsWartermark') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isWartermark" size="small" @change="onWartermarkChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourWartermarkText') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-input v-model="getThemeConfig.wartermarkText" size="default" style="width: 90px" @input="onWartermarkTextInput($event)"></el-input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 其它设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.fiveTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveTagsStyle') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select v-model="getThemeConfig.tagsStyle" placeholder="请选择" size="default" style="width: 90px" @change="setLocalThemeConfig">
|
||||||
|
<el-option label="风格1" value="tags-style-one"></el-option>
|
||||||
|
<el-option label="风格4" value="tags-style-four"></el-option>
|
||||||
|
<el-option label="风格5" value="tags-style-five"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveAnimation') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select v-model="getThemeConfig.animation" placeholder="请选择" size="default" style="width: 90px" @change="setLocalThemeConfig">
|
||||||
|
<el-option label="slide-right" value="slide-right"></el-option>
|
||||||
|
<el-option label="slide-left" value="slide-left"></el-option>
|
||||||
|
<el-option label="opacitys" value="opacitys"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveColumnsAsideStyle') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select
|
||||||
|
v-model="getThemeConfig.columnsAsideStyle"
|
||||||
|
placeholder="请选择"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns' ? true : false"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
>
|
||||||
|
<el-option label="圆角" value="columns-round"></el-option>
|
||||||
|
<el-option label="卡片" value="columns-card"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15 mb27" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveColumnsAsideLayout') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select
|
||||||
|
v-model="getThemeConfig.columnsAsideLayout"
|
||||||
|
placeholder="请选择"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns' ? true : false"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
>
|
||||||
|
<el-option label="水平" value="columns-horizontal"></el-option>
|
||||||
|
<el-option label="垂直" value="columns-vertical"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 布局切换 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.sixTitle') }}</el-divider>
|
||||||
|
<div class="layout-drawer-content-flex">
|
||||||
|
<!-- defaults 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('defaults')">
|
||||||
|
<section class="el-container el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'defaults' }">
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'defaults' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixDefaults') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- classic 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('classic')">
|
||||||
|
<section class="el-container is-vertical el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'classic' }">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<section class="el-container">
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'classic' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixClassic') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- transverse 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('transverse')">
|
||||||
|
<section class="el-container is-vertical el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'transverse' }">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<section class="el-container">
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'transverse' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixTransverse') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- columns 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('columns')">
|
||||||
|
<section class="el-container el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'columns' }">
|
||||||
|
<aside class="el-aside-dark" style="width: 10px"></aside>
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'columns' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixColumns') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="copy-config">
|
||||||
|
<el-alert :title="$t('message.layout.tipText')" type="warning" :closable="false"> </el-alert>
|
||||||
|
<el-button size="default" class="copy-config-btn" type="primary" ref="copyConfigBtnRef" @click="onCopyConfigClick">
|
||||||
|
<el-icon class="mr5">
|
||||||
|
<ele-CopyDocument />
|
||||||
|
</el-icon>
|
||||||
|
{{ $t('message.layout.copyText') }}
|
||||||
|
</el-button>
|
||||||
|
<el-button size="default" class="copy-config-btn-reset" type="info" @click="onResetConfigClick">
|
||||||
|
<el-icon class="mr5">
|
||||||
|
<ele-RefreshRight />
|
||||||
|
</el-icon>
|
||||||
|
{{ $t('message.layout.resetText') }}
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-drawer>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { nextTick, onUnmounted, onMounted, getCurrentInstance, defineComponent, computed, reactive, toRefs } from 'vue';
|
||||||
|
import { ElMessage } from 'element-plus';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { getLightColor, getDarkColor } from '/@/utils/theme';
|
||||||
|
import { verifyAndSpace } from '/@/utils/toolsValidate';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
import Watermark from '/@/utils/wartermark';
|
||||||
|
import commonFunction from '/@/utils/commonFunction';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumbSeting',
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { copyText } = commonFunction();
|
||||||
|
const state = reactive({
|
||||||
|
isMobile: false,
|
||||||
|
});
|
||||||
|
// 获取布局配置信息
|
||||||
|
const getThemeConfig = computed(() => {
|
||||||
|
return themeConfig.value;
|
||||||
|
});
|
||||||
|
// 1、全局主题
|
||||||
|
const onColorPickerChange = () => {
|
||||||
|
if (!getThemeConfig.value.primary) return ElMessage.warning('全局主题 primary 颜色值不能为空');
|
||||||
|
// 颜色加深
|
||||||
|
document.documentElement.style.setProperty('--el-color-primary-dark-2', `${getDarkColor(getThemeConfig.value.primary, 0.1)}`);
|
||||||
|
document.documentElement.style.setProperty('--el-color-primary', getThemeConfig.value.primary);
|
||||||
|
// 颜色变浅
|
||||||
|
for (let i = 1; i <= 9; i++) {
|
||||||
|
document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, `${getLightColor(getThemeConfig.value.primary, i / 10)}`);
|
||||||
|
}
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏
|
||||||
|
const onBgColorPickerChange = (bg: string) => {
|
||||||
|
document.documentElement.style.setProperty(`--next-bg-${bg}`, (<any>getThemeConfig.value)[bg]);
|
||||||
|
if (bg === 'menuBar') {
|
||||||
|
document.documentElement.style.setProperty(`--next-bg-menuBar-light-1`, <any>getLightColor(getThemeConfig.value.menuBar, 0.05));
|
||||||
|
}
|
||||||
|
onTopBarGradualChange();
|
||||||
|
onMenuBarGradualChange();
|
||||||
|
onColumnsMenuBarGradualChange();
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 顶栏背景渐变
|
||||||
|
const onTopBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-navbars-breadcrumb-index', getThemeConfig.value.isTopBarColorGradual, getThemeConfig.value.topBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 菜单背景渐变
|
||||||
|
const onMenuBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-container .el-aside', getThemeConfig.value.isMenuBarColorGradual, getThemeConfig.value.menuBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 分栏菜单背景渐变
|
||||||
|
const onColumnsMenuBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-container .layout-columns-aside', getThemeConfig.value.isColumnsMenuBarColorGradual, getThemeConfig.value.columnsMenuBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 背景渐变函数
|
||||||
|
const setGraduaFun = (el: string, bool: boolean, color: string) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
let els = document.querySelector(el);
|
||||||
|
if (!els) return false;
|
||||||
|
document.documentElement.style.setProperty('--el-menu-bg-color', document.documentElement.style.getPropertyValue('--next-bg-menuBar'));
|
||||||
|
if (bool) els.setAttribute('style', `background:linear-gradient(to bottom left , ${color}, ${getLightColor(color, 0.6)}) !important;`);
|
||||||
|
else els.setAttribute('style', ``);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
}, 200);
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 菜单水平折叠
|
||||||
|
const onThemeConfigChange = () => {
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 固定 Header
|
||||||
|
const onIsFixedHeaderChange = () => {
|
||||||
|
getThemeConfig.value.isFixedHeaderChange = getThemeConfig.value.isFixedHeader ? false : true;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 经典布局分割菜单
|
||||||
|
const onClassicSplitMenuChange = () => {
|
||||||
|
getThemeConfig.value.isBreadcrumb = false;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
proxy.mittBus.emit('getBreadcrumbIndexSetFilterRoutes');
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 侧边栏 Logo
|
||||||
|
const onIsShowLogoChange = () => {
|
||||||
|
getThemeConfig.value.isShowLogoChange = getThemeConfig.value.isShowLogo ? false : true;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 面包屑 Breadcrumb
|
||||||
|
const onIsBreadcrumbChange = () => {
|
||||||
|
if (getThemeConfig.value.layout === 'classic') {
|
||||||
|
getThemeConfig.value.isClassicSplitMenu = false;
|
||||||
|
}
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启 TagsView 拖拽
|
||||||
|
const onSortableTagsViewChange = () => {
|
||||||
|
proxy.mittBus.emit('openOrCloseSortable');
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启 TagsView 共用
|
||||||
|
const onShareTagsViewChange = () => {
|
||||||
|
proxy.mittBus.emit('openShareTagsView');
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 灰色模式/色弱模式
|
||||||
|
const onAddFilterChange = (attr: string) => {
|
||||||
|
if (attr === 'grayscale') {
|
||||||
|
if (getThemeConfig.value.isGrayscale) getThemeConfig.value.isInvert = false;
|
||||||
|
} else {
|
||||||
|
if (getThemeConfig.value.isInvert) getThemeConfig.value.isGrayscale = false;
|
||||||
|
}
|
||||||
|
const cssAttr =
|
||||||
|
attr === 'grayscale' ? `grayscale(${getThemeConfig.value.isGrayscale ? 1 : 0})` : `invert(${getThemeConfig.value.isInvert ? '80%' : '0%'})`;
|
||||||
|
const appEle: any = document.body;
|
||||||
|
appEle.setAttribute('style', `filter: ${cssAttr}`);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 深色模式
|
||||||
|
const onAddDarkChange = () => {
|
||||||
|
const body = document.documentElement as HTMLElement;
|
||||||
|
if (getThemeConfig.value.isIsDark) body.setAttribute('data-theme', 'dark');
|
||||||
|
else body.setAttribute('data-theme', '');
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启水印
|
||||||
|
const onWartermarkChange = () => {
|
||||||
|
getThemeConfig.value.isWartermark ? Watermark.set(getThemeConfig.value.wartermarkText) : Watermark.del();
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 水印文案
|
||||||
|
const onWartermarkTextInput = (val: any) => {
|
||||||
|
getThemeConfig.value.wartermarkText = verifyAndSpace(val);
|
||||||
|
if (getThemeConfig.value.wartermarkText === '') return false;
|
||||||
|
if (getThemeConfig.value.isWartermark) Watermark.set(getThemeConfig.value.wartermarkText);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 5、布局切换
|
||||||
|
const onSetLayout = (layout: string) => {
|
||||||
|
Local.set('oldLayout', layout);
|
||||||
|
if (getThemeConfig.value.layout === layout) return false;
|
||||||
|
if (layout === 'transverse') getThemeConfig.value.isCollapse = false;
|
||||||
|
getThemeConfig.value.layout = layout;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
initLayoutChangeFun();
|
||||||
|
};
|
||||||
|
// 设置布局切换函数
|
||||||
|
const initLayoutChangeFun = () => {
|
||||||
|
onBgColorPickerChange('menuBar');
|
||||||
|
onBgColorPickerChange('menuBarColor');
|
||||||
|
onBgColorPickerChange('topBar');
|
||||||
|
onBgColorPickerChange('topBarColor');
|
||||||
|
onBgColorPickerChange('columnsMenuBar');
|
||||||
|
onBgColorPickerChange('columnsMenuBarColor');
|
||||||
|
};
|
||||||
|
// 关闭弹窗时,初始化变量。变量用于处理 proxy.$refs.layoutScrollbarRef.update()
|
||||||
|
const onDrawerClose = () => {
|
||||||
|
getThemeConfig.value.isFixedHeaderChange = false;
|
||||||
|
getThemeConfig.value.isShowLogoChange = false;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 布局配置弹窗打开
|
||||||
|
const openDrawer = () => {
|
||||||
|
getThemeConfig.value.isDrawer = true;
|
||||||
|
};
|
||||||
|
// 触发 store 布局配置更新
|
||||||
|
const setDispatchThemeConfig = () => {
|
||||||
|
setLocalThemeConfig();
|
||||||
|
setLocalThemeConfigStyle();
|
||||||
|
};
|
||||||
|
// 存储布局配置
|
||||||
|
const setLocalThemeConfig = () => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
Local.set('themeConfig', getThemeConfig.value);
|
||||||
|
};
|
||||||
|
// 存储布局配置全局主题样式(html根标签)
|
||||||
|
const setLocalThemeConfigStyle = () => {
|
||||||
|
Local.set('themeConfigStyle', document.documentElement.style.cssText);
|
||||||
|
};
|
||||||
|
// 一键复制配置
|
||||||
|
const onCopyConfigClick = () => {
|
||||||
|
let copyThemeConfig = Local.get('themeConfig');
|
||||||
|
copyThemeConfig.isDrawer = false;
|
||||||
|
copyText(JSON.stringify(copyThemeConfig)).then(() => {
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 一键恢复默认
|
||||||
|
const onResetConfigClick = () => {
|
||||||
|
Local.clear();
|
||||||
|
window.location.reload();
|
||||||
|
};
|
||||||
|
// 初始化菜单样式等
|
||||||
|
const initSetStyle = () => {
|
||||||
|
// 2、菜单 / 顶栏 --> 顶栏背景渐变
|
||||||
|
onTopBarGradualChange();
|
||||||
|
// 2、菜单 / 顶栏 --> 菜单背景渐变
|
||||||
|
onMenuBarGradualChange();
|
||||||
|
// 2、菜单 / 顶栏 --> 分栏菜单背景渐变
|
||||||
|
onColumnsMenuBarGradualChange();
|
||||||
|
};
|
||||||
|
onMounted(() => {
|
||||||
|
nextTick(() => {
|
||||||
|
// 判断当前布局是否不相同,不相同则初始化当前布局的样式,防止监听窗口大小改变时,布局配置logo、菜单背景等部分布局失效问题
|
||||||
|
if (!Local.get('frequency')) initLayoutChangeFun();
|
||||||
|
Local.set('frequency', 1);
|
||||||
|
// 监听窗口大小改变,非默认布局,设置成默认布局(适配移动端)
|
||||||
|
proxy.mittBus.on('layoutMobileResize', (res: any) => {
|
||||||
|
getThemeConfig.value.layout = res.layout;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
initLayoutChangeFun();
|
||||||
|
state.isMobile = other.isMobile();
|
||||||
|
});
|
||||||
|
setTimeout(() => {
|
||||||
|
// 默认样式
|
||||||
|
onColorPickerChange();
|
||||||
|
// 灰色模式
|
||||||
|
if (getThemeConfig.value.isGrayscale) onAddFilterChange('grayscale');
|
||||||
|
// 色弱模式
|
||||||
|
if (getThemeConfig.value.isInvert) onAddFilterChange('invert');
|
||||||
|
// 深色模式
|
||||||
|
if (getThemeConfig.value.isIsDark) onAddDarkChange();
|
||||||
|
// 开启水印
|
||||||
|
onWartermarkChange();
|
||||||
|
// 语言国际化
|
||||||
|
if (Local.get('themeConfig')) proxy.$i18n.locale = Local.get('themeConfig').globalI18n;
|
||||||
|
// 初始化菜单样式等
|
||||||
|
initSetStyle();
|
||||||
|
}, 100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
onUnmounted(() => {
|
||||||
|
proxy.mittBus.off('layoutMobileResize', () => {});
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
openDrawer,
|
||||||
|
onColorPickerChange,
|
||||||
|
onBgColorPickerChange,
|
||||||
|
onTopBarGradualChange,
|
||||||
|
onMenuBarGradualChange,
|
||||||
|
onColumnsMenuBarGradualChange,
|
||||||
|
onThemeConfigChange,
|
||||||
|
onIsFixedHeaderChange,
|
||||||
|
onIsShowLogoChange,
|
||||||
|
getThemeConfig,
|
||||||
|
onDrawerClose,
|
||||||
|
onAddFilterChange,
|
||||||
|
onAddDarkChange,
|
||||||
|
onWartermarkChange,
|
||||||
|
onWartermarkTextInput,
|
||||||
|
onSetLayout,
|
||||||
|
setLocalThemeConfig,
|
||||||
|
onClassicSplitMenuChange,
|
||||||
|
onIsBreadcrumbChange,
|
||||||
|
onSortableTagsViewChange,
|
||||||
|
onShareTagsViewChange,
|
||||||
|
onCopyConfigClick,
|
||||||
|
onResetConfigClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-breadcrumb-seting-bar {
|
||||||
|
height: calc(100vh - 50px);
|
||||||
|
padding: 0 15px;
|
||||||
|
:deep(.el-scrollbar__view) {
|
||||||
|
overflow-x: hidden !important;
|
||||||
|
}
|
||||||
|
.layout-breadcrumb-seting-bar-flex {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
&-label {
|
||||||
|
flex: 1;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-drawer-content-flex {
|
||||||
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-content: flex-start;
|
||||||
|
margin: 0 -5px;
|
||||||
|
.layout-drawer-content-item {
|
||||||
|
width: 50%;
|
||||||
|
height: 70px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
position: relative;
|
||||||
|
padding: 5px;
|
||||||
|
.el-container {
|
||||||
|
height: 100%;
|
||||||
|
.el-aside-dark {
|
||||||
|
background-color: var(--next-color-seting-header);
|
||||||
|
}
|
||||||
|
.el-aside {
|
||||||
|
background-color: var(--next-color-seting-aside);
|
||||||
|
}
|
||||||
|
.el-header {
|
||||||
|
background-color: var(--next-color-seting-header);
|
||||||
|
}
|
||||||
|
.el-main {
|
||||||
|
background-color: var(--next-color-seting-main);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-circular {
|
||||||
|
border-radius: 2px;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
.drawer-layout-active {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
.layout-tips-warp,
|
||||||
|
.layout-tips-warp-active {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
border-radius: 100%;
|
||||||
|
padding: 4px;
|
||||||
|
.layout-tips-box {
|
||||||
|
transition: inherit;
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
z-index: 9;
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
border-radius: 100%;
|
||||||
|
.layout-tips-txt {
|
||||||
|
transition: inherit;
|
||||||
|
position: relative;
|
||||||
|
top: 5px;
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 1;
|
||||||
|
letter-spacing: 2px;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: var(--el-color-primary-light-5);
|
||||||
|
text-align: center;
|
||||||
|
transform: rotate(30deg);
|
||||||
|
left: -1px;
|
||||||
|
background-color: var(--next-color-seting-main);
|
||||||
|
width: 32px;
|
||||||
|
height: 17px;
|
||||||
|
line-height: 17px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-tips-warp-active {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-box {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-txt {
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
background-color: var(--next-color-seting-main) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
.el-circular {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
.layout-tips-warp {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-box {
|
||||||
|
transition: inherit;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-txt {
|
||||||
|
transition: inherit;
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
background-color: var(--next-color-seting-main) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.copy-config {
|
||||||
|
margin: 10px 0;
|
||||||
|
.copy-config-btn {
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
.copy-config-btn-reset {
|
||||||
|
width: 100%;
|
||||||
|
margin: 10px 0 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
300
web/src/layout/navBars/breadcrumb/user.vue
Normal file
300
web/src/layout/navBars/breadcrumb/user.vue
Normal file
@@ -0,0 +1,300 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-user pr15" :style="{ flex: layoutUserFlexNum }">
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" trigger="click" @command="onComponentSizeChange">
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon">
|
||||||
|
<i class="iconfont icon-ziti" :title="$t('message.user.title0')"></i>
|
||||||
|
</div>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="large" :disabled="disabledSize === 'large'">{{ $t('message.user.dropdownLarge') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="default" :disabled="disabledSize === 'default'">{{ $t('message.user.dropdownDefault') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="small" :disabled="disabledSize === 'small'">{{ $t('message.user.dropdownSmall') }}</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" trigger="click" @command="onLanguageChange">
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon">
|
||||||
|
<i class="iconfont" :class="disabledI18n === 'en' ? 'icon-fuhao-yingwen' : 'icon-fuhao-zhongwen'" :title="$t('message.user.title1')"></i>
|
||||||
|
</div>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="zh-cn" :disabled="disabledI18n === 'zh-cn'">简体中文</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="en" :disabled="disabledI18n === 'en'">English</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="zh-tw" :disabled="disabledI18n === 'zh-tw'">繁體中文</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon" @click="onSearchClick">
|
||||||
|
<el-icon :title="$t('message.user.title2')">
|
||||||
|
<ele-Search />
|
||||||
|
</el-icon>
|
||||||
|
</div>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon" @click="onLayoutSetingClick">
|
||||||
|
<i class="icon-xitongshezhi iconfont" :title="$t('message.user.title3')"></i>
|
||||||
|
</div>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon">
|
||||||
|
<el-popover placement="bottom" trigger="click" transition="el-zoom-in-top" :width="300" :persistent="false">
|
||||||
|
<template #reference>
|
||||||
|
<el-badge :is-dot="true">
|
||||||
|
<el-icon :title="$t('message.user.title4')">
|
||||||
|
<ele-Bell />
|
||||||
|
</el-icon>
|
||||||
|
</el-badge>
|
||||||
|
</template>
|
||||||
|
<template #default>
|
||||||
|
<UserNews />
|
||||||
|
</template>
|
||||||
|
</el-popover>
|
||||||
|
</div>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon mr10" @click="onScreenfullClick">
|
||||||
|
<i
|
||||||
|
class="iconfont"
|
||||||
|
:title="isScreenfull ? $t('message.user.title6') : $t('message.user.title5')"
|
||||||
|
:class="!isScreenfull ? 'icon-fullscreen' : 'icon-tuichuquanping'"
|
||||||
|
></i>
|
||||||
|
</div>
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" @command="onHandleCommandClick">
|
||||||
|
<span class="layout-navbars-breadcrumb-user-link">
|
||||||
|
<img :src="userInfos.photo" class="layout-navbars-breadcrumb-user-link-photo mr5" />
|
||||||
|
{{ userInfos.userName === '' ? 'common' : userInfos.userName }}
|
||||||
|
<el-icon class="el-icon--right">
|
||||||
|
<ele-ArrowDown />
|
||||||
|
</el-icon>
|
||||||
|
</span>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="/home">{{ $t('message.user.dropdown1') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="wareHouse">{{ $t('message.user.dropdown6') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/personal">{{ $t('message.user.dropdown2') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/404">{{ $t('message.user.dropdown3') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/401">{{ $t('message.user.dropdown4') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item divided command="logOut">{{ $t('message.user.dropdown5') }}</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<Search ref="searchRef" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { ref, getCurrentInstance, computed, reactive, toRefs, onMounted, defineComponent } from 'vue';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
import { ElMessageBox, ElMessage } from 'element-plus';
|
||||||
|
import screenfull from 'screenfull';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import { Session, Local } from '/@/utils/storage';
|
||||||
|
import UserNews from '/@/layout/navBars/breadcrumb/userNews.vue';
|
||||||
|
import Search from '/@/layout/navBars/breadcrumb/search.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumbUser',
|
||||||
|
components: { UserNews, Search },
|
||||||
|
setup() {
|
||||||
|
const { t } = useI18n();
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const router = useRouter();
|
||||||
|
const stores = useUserInfo();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const searchRef = ref();
|
||||||
|
const state = reactive({
|
||||||
|
isScreenfull: false,
|
||||||
|
disabledI18n: 'zh-cn',
|
||||||
|
disabledSize: 'large',
|
||||||
|
});
|
||||||
|
// 设置分割样式
|
||||||
|
const layoutUserFlexNum = computed(() => {
|
||||||
|
let num: string | number = '';
|
||||||
|
const { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
const layoutArr: string[] = ['defaults', 'columns'];
|
||||||
|
if (layoutArr.includes(layout) || (layout === 'classic' && !isClassicSplitMenu)) num = '1';
|
||||||
|
else num = '';
|
||||||
|
return num;
|
||||||
|
});
|
||||||
|
// 全屏点击时
|
||||||
|
const onScreenfullClick = () => {
|
||||||
|
if (!screenfull.isEnabled) {
|
||||||
|
ElMessage.warning('暂不不支持全屏');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
screenfull.toggle();
|
||||||
|
screenfull.on('change', () => {
|
||||||
|
if (screenfull.isFullscreen) state.isScreenfull = true;
|
||||||
|
else state.isScreenfull = false;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 布局配置 icon 点击时
|
||||||
|
const onLayoutSetingClick = () => {
|
||||||
|
proxy.mittBus.emit('openSetingsDrawer');
|
||||||
|
};
|
||||||
|
// 下拉菜单点击时
|
||||||
|
const onHandleCommandClick = (path: string) => {
|
||||||
|
if (path === 'logOut') {
|
||||||
|
ElMessageBox({
|
||||||
|
closeOnClickModal: false,
|
||||||
|
closeOnPressEscape: false,
|
||||||
|
title: t('message.user.logOutTitle'),
|
||||||
|
message: t('message.user.logOutMessage'),
|
||||||
|
showCancelButton: true,
|
||||||
|
confirmButtonText: t('message.user.logOutConfirm'),
|
||||||
|
cancelButtonText: t('message.user.logOutCancel'),
|
||||||
|
buttonSize: 'default',
|
||||||
|
beforeClose: (action, instance, done) => {
|
||||||
|
if (action === 'confirm') {
|
||||||
|
instance.confirmButtonLoading = true;
|
||||||
|
instance.confirmButtonText = t('message.user.logOutExit');
|
||||||
|
setTimeout(() => {
|
||||||
|
done();
|
||||||
|
setTimeout(() => {
|
||||||
|
instance.confirmButtonLoading = false;
|
||||||
|
}, 300);
|
||||||
|
}, 700);
|
||||||
|
} else {
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.then(async () => {
|
||||||
|
// 清除缓存/token等
|
||||||
|
Session.clear();
|
||||||
|
// 使用 reload 时,不需要调用 resetRoute() 重置路由
|
||||||
|
window.location.reload();
|
||||||
|
})
|
||||||
|
.catch(() => {});
|
||||||
|
} else if (path === 'wareHouse') {
|
||||||
|
window.open('https://gitee.com/lyt-top/vue-next-admin');
|
||||||
|
} else {
|
||||||
|
router.push(path);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 菜单搜索点击
|
||||||
|
const onSearchClick = () => {
|
||||||
|
searchRef.value.openSearch();
|
||||||
|
};
|
||||||
|
// 组件大小改变
|
||||||
|
const onComponentSizeChange = (size: string) => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
themeConfig.value.globalComponentSize = size;
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
initComponentSize();
|
||||||
|
window.location.reload();
|
||||||
|
};
|
||||||
|
// 语言切换
|
||||||
|
const onLanguageChange = (lang: string) => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
themeConfig.value.globalI18n = lang;
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
proxy.$i18n.locale = lang;
|
||||||
|
initI18n();
|
||||||
|
other.useTitle();
|
||||||
|
};
|
||||||
|
// 设置 element plus 组件的国际化
|
||||||
|
const setI18nConfig = (locale: string) => {
|
||||||
|
proxy.$i18n={messages:{}}
|
||||||
|
proxy.mittBus.emit('getI18nConfig', proxy.$i18n.messages[locale]);
|
||||||
|
};
|
||||||
|
// 初始化言语国际化
|
||||||
|
const initI18n = () => {
|
||||||
|
switch (Local.get('themeConfig').globalI18n) {
|
||||||
|
case 'zh-cn':
|
||||||
|
state.disabledI18n = 'zh-cn';
|
||||||
|
setI18nConfig('zh-cn');
|
||||||
|
break;
|
||||||
|
case 'en':
|
||||||
|
state.disabledI18n = 'en';
|
||||||
|
setI18nConfig('en');
|
||||||
|
break;
|
||||||
|
case 'zh-tw':
|
||||||
|
state.disabledI18n = 'zh-tw';
|
||||||
|
setI18nConfig('zh-tw');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 初始化全局组件大小
|
||||||
|
const initComponentSize = () => {
|
||||||
|
switch (Local.get('themeConfig').globalComponentSize) {
|
||||||
|
case 'large':
|
||||||
|
state.disabledSize = 'large';
|
||||||
|
break;
|
||||||
|
case 'default':
|
||||||
|
state.disabledSize = 'default';
|
||||||
|
break;
|
||||||
|
case 'small':
|
||||||
|
state.disabledSize = 'small';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
if (Local.get('themeConfig')) {
|
||||||
|
initI18n();
|
||||||
|
initComponentSize();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
userInfos,
|
||||||
|
onLayoutSetingClick,
|
||||||
|
onHandleCommandClick,
|
||||||
|
onScreenfullClick,
|
||||||
|
onSearchClick,
|
||||||
|
onComponentSizeChange,
|
||||||
|
onLanguageChange,
|
||||||
|
searchRef,
|
||||||
|
layoutUserFlexNum,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-user {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
&-link {
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
white-space: nowrap;
|
||||||
|
&-photo {
|
||||||
|
width: 25px;
|
||||||
|
height: 25px;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-icon {
|
||||||
|
padding: 0 10px;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
height: 50px;
|
||||||
|
line-height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
&:hover {
|
||||||
|
background: var(--next-color-user-hover);
|
||||||
|
i {
|
||||||
|
display: inline-block;
|
||||||
|
animation: logoAnimation 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-dropdown) {
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
:deep(.el-badge) {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
:deep(.el-badge__content.is-fixed) {
|
||||||
|
top: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
115
web/src/layout/navBars/breadcrumb/userNews.vue
Normal file
115
web/src/layout/navBars/breadcrumb/userNews.vue
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-news">
|
||||||
|
<div class="head-box">
|
||||||
|
<div class="head-box-title">{{ $t('message.user.newTitle') }}</div>
|
||||||
|
<div class="head-box-btn" v-if="newsList.length > 0" @click="onAllReadClick">{{ $t('message.user.newBtn') }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="content-box">
|
||||||
|
<template v-if="newsList.length > 0">
|
||||||
|
<div class="content-box-item" v-for="(v, k) in newsList" :key="k">
|
||||||
|
<div>{{ v.label }}</div>
|
||||||
|
<div class="content-box-msg">
|
||||||
|
{{ v.value }}
|
||||||
|
</div>
|
||||||
|
<div class="content-box-time">{{ v.time }}</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<el-empty :description="$t('message.user.newDesc')" v-else></el-empty>
|
||||||
|
</div>
|
||||||
|
<div class="foot-box" @click="onGoToGiteeClick" v-if="newsList.length > 0">{{ $t('message.user.newGo') }}</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { reactive, toRefs, defineComponent } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutBreadcrumbUserNews',
|
||||||
|
setup() {
|
||||||
|
const state = reactive({
|
||||||
|
newsList: [
|
||||||
|
{
|
||||||
|
label: '关于版本发布的通知',
|
||||||
|
value: 'vue-next-admin,基于 vue3 + CompositionAPI + typescript + vite + element plus,正式发布时间:2021年02月28日!',
|
||||||
|
time: '2020-12-08',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '关于学习交流的通知',
|
||||||
|
value: 'QQ群号码 665452019,欢迎小伙伴入群学习交流探讨!',
|
||||||
|
time: '2020-12-08',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
// 全部已读点击
|
||||||
|
const onAllReadClick = () => {
|
||||||
|
state.newsList = [];
|
||||||
|
};
|
||||||
|
// 前往通知中心点击
|
||||||
|
const onGoToGiteeClick = () => {
|
||||||
|
window.open('https://gitee.com/lyt-top/vue-next-admin');
|
||||||
|
};
|
||||||
|
return {
|
||||||
|
onAllReadClick,
|
||||||
|
onGoToGiteeClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-user-news {
|
||||||
|
.head-box {
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 1px solid var(--el-border-color-lighter);
|
||||||
|
box-sizing: border-box;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
justify-content: space-between;
|
||||||
|
height: 35px;
|
||||||
|
align-items: center;
|
||||||
|
.head-box-btn {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
font-size: 13px;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.8;
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.content-box {
|
||||||
|
font-size: 13px;
|
||||||
|
.content-box-item {
|
||||||
|
padding-top: 12px;
|
||||||
|
&:last-of-type {
|
||||||
|
padding-bottom: 12px;
|
||||||
|
}
|
||||||
|
.content-box-msg {
|
||||||
|
color: var(--el-text-color-secondary);
|
||||||
|
margin-top: 5px;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
.content-box-time {
|
||||||
|
color: var(--el-text-color-secondary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.foot-box {
|
||||||
|
height: 35px;
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
font-size: 13px;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.8;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-top: 1px solid var(--el-border-color-lighter);
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-empty__description p) {
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
40
web/src/layout/navBars/index.vue
Normal file
40
web/src/layout/navBars/index.vue
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-container">
|
||||||
|
<BreadcrumbIndex />
|
||||||
|
<TagsView v-if="setShowTagsView" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import BreadcrumbIndex from '/@/layout/navBars/breadcrumb/index.vue';
|
||||||
|
import TagsView from '/@/layout/navBars/tagsView/tagsView.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutNavBars',
|
||||||
|
components: { BreadcrumbIndex, TagsView },
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
// 是否显示 tagsView
|
||||||
|
const setShowTagsView = computed(() => {
|
||||||
|
let { layout, isTagsview } = themeConfig.value;
|
||||||
|
return layout !== 'classic' && isTagsview;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
setShowTagsView,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
138
web/src/layout/navBars/tagsView/contextmenu.vue
Normal file
138
web/src/layout/navBars/tagsView/contextmenu.vue
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
<template>
|
||||||
|
<transition name="el-zoom-in-center">
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
class="el-dropdown__popper el-popper is-light is-pure custom-contextmenu"
|
||||||
|
role="tooltip"
|
||||||
|
data-popper-placement="bottom"
|
||||||
|
:style="`top: ${dropdowns.y + 5}px;left: ${dropdowns.x}px;`"
|
||||||
|
:key="Math.random()"
|
||||||
|
v-show="isShow"
|
||||||
|
>
|
||||||
|
<ul class="el-dropdown-menu">
|
||||||
|
<template v-for="(v, k) in dropdownList">
|
||||||
|
<li
|
||||||
|
class="el-dropdown-menu__item"
|
||||||
|
aria-disabled="false"
|
||||||
|
tabindex="-1"
|
||||||
|
:key="k"
|
||||||
|
v-if="!v.affix"
|
||||||
|
@click="onCurrentContextmenuClick(v.contextMenuClickId)"
|
||||||
|
>
|
||||||
|
<SvgIcon :name="v.icon" />
|
||||||
|
<span>{{ $t(v.txt) }}</span>
|
||||||
|
</li>
|
||||||
|
</template>
|
||||||
|
</ul>
|
||||||
|
<div class="el-popper__arrow" :style="{ left: `${arrowLeft}px` }"></div>
|
||||||
|
</div>
|
||||||
|
</transition>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent, reactive, toRefs, onMounted, onUnmounted, watch } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutTagsViewContextmenu',
|
||||||
|
props: {
|
||||||
|
dropdown: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {
|
||||||
|
return {
|
||||||
|
x: 0,
|
||||||
|
y: 0,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props, { emit }) {
|
||||||
|
const state = reactive({
|
||||||
|
isShow: false,
|
||||||
|
dropdownList: [
|
||||||
|
{ contextMenuClickId: 0, txt: 'message.tagsView.refresh', affix: false, icon: 'ele-RefreshRight' },
|
||||||
|
{ contextMenuClickId: 1, txt: 'message.tagsView.close', affix: false, icon: 'ele-Close' },
|
||||||
|
{ contextMenuClickId: 2, txt: 'message.tagsView.closeOther', affix: false, icon: 'ele-CircleClose' },
|
||||||
|
{ contextMenuClickId: 3, txt: 'message.tagsView.closeAll', affix: false, icon: 'ele-FolderDelete' },
|
||||||
|
{
|
||||||
|
contextMenuClickId: 4,
|
||||||
|
txt: 'message.tagsView.fullscreen',
|
||||||
|
affix: false,
|
||||||
|
icon: 'iconfont icon-fullscreen',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
item: {},
|
||||||
|
arrowLeft: 10,
|
||||||
|
});
|
||||||
|
// 父级传过来的坐标 x,y 值
|
||||||
|
const dropdowns = computed(() => {
|
||||||
|
// 117 为 `Dropdown 下拉菜单` 的宽度
|
||||||
|
if (props.dropdown.x + 117 > document.documentElement.clientWidth) {
|
||||||
|
return {
|
||||||
|
x: document.documentElement.clientWidth - 117 - 5,
|
||||||
|
y: props.dropdown.y,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return props.dropdown;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 当前项菜单点击
|
||||||
|
const onCurrentContextmenuClick = (contextMenuClickId: number) => {
|
||||||
|
emit('currentContextmenuClick', Object.assign({}, { contextMenuClickId }, state.item));
|
||||||
|
};
|
||||||
|
// 打开右键菜单:判断是否固定,固定则不显示关闭按钮
|
||||||
|
const openContextmenu = (item: any) => {
|
||||||
|
state.item = item;
|
||||||
|
item.meta.isAffix ? (state.dropdownList[1].affix = true) : (state.dropdownList[1].affix = false);
|
||||||
|
closeContextmenu();
|
||||||
|
setTimeout(() => {
|
||||||
|
state.isShow = true;
|
||||||
|
}, 10);
|
||||||
|
};
|
||||||
|
// 关闭右键菜单
|
||||||
|
const closeContextmenu = () => {
|
||||||
|
state.isShow = false;
|
||||||
|
};
|
||||||
|
// 监听页面监听进行右键菜单的关闭
|
||||||
|
onMounted(() => {
|
||||||
|
document.body.addEventListener('click', closeContextmenu);
|
||||||
|
});
|
||||||
|
// 页面卸载时,移除右键菜单监听事件
|
||||||
|
onUnmounted(() => {
|
||||||
|
document.body.removeEventListener('click', closeContextmenu);
|
||||||
|
});
|
||||||
|
// 监听下拉菜单位置
|
||||||
|
watch(
|
||||||
|
() => props.dropdown,
|
||||||
|
({ x }) => {
|
||||||
|
if (x + 117 > document.documentElement.clientWidth) state.arrowLeft = 117 - (document.documentElement.clientWidth - x);
|
||||||
|
else state.arrowLeft = 10;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
dropdowns,
|
||||||
|
openContextmenu,
|
||||||
|
closeContextmenu,
|
||||||
|
onCurrentContextmenuClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.custom-contextmenu {
|
||||||
|
transform-origin: center top;
|
||||||
|
z-index: 2190;
|
||||||
|
position: fixed;
|
||||||
|
.el-dropdown-menu__item {
|
||||||
|
font-size: 12px !important;
|
||||||
|
white-space: nowrap;
|
||||||
|
i {
|
||||||
|
font-size: 12px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
734
web/src/layout/navBars/tagsView/tagsView.vue
Normal file
734
web/src/layout/navBars/tagsView/tagsView.vue
Normal file
@@ -0,0 +1,734 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-tagsview" :class="{ 'layout-navbars-tagsview-shadow': getThemeConfig.layout === 'classic' }">
|
||||||
|
<el-scrollbar ref="scrollbarRef" @wheel.prevent="onHandleScroll">
|
||||||
|
<ul class="layout-navbars-tagsview-ul" :class="setTagsStyle" ref="tagsUlRef">
|
||||||
|
<li
|
||||||
|
v-for="(v, k) in tagsViewList"
|
||||||
|
:key="k"
|
||||||
|
class="layout-navbars-tagsview-ul-li"
|
||||||
|
:data-url="v.url"
|
||||||
|
:class="{ 'is-active': isActive(v) }"
|
||||||
|
@contextmenu.prevent="onContextmenu(v, $event)"
|
||||||
|
@click="onTagsClick(v, k)"
|
||||||
|
:ref="
|
||||||
|
(el) => {
|
||||||
|
if (el) tagsRefs[k] = el;
|
||||||
|
}
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<i class="iconfont icon-webicon318 layout-navbars-tagsview-ul-li-iconfont" v-if="isActive(v)"></i>
|
||||||
|
<SvgIcon :name="v.meta.icon" v-if="!isActive(v) && getThemeConfig.isTagsviewIcon" class="pr5" />
|
||||||
|
<span>{{ setTagsViewNameI18n(v) }}</span>
|
||||||
|
<template v-if="isActive(v)">
|
||||||
|
<SvgIcon
|
||||||
|
name="ele-RefreshRight"
|
||||||
|
class="ml5 layout-navbars-tagsview-ul-li-refresh"
|
||||||
|
@click.stop="refreshCurrentTagsView($route.fullPath)"
|
||||||
|
/>
|
||||||
|
<SvgIcon
|
||||||
|
name="ele-Close"
|
||||||
|
class="layout-navbars-tagsview-ul-li-icon layout-icon-active"
|
||||||
|
v-if="!v.meta.isAffix"
|
||||||
|
@click.stop="closeCurrentTagsView(getThemeConfig.isShareTagsView ? v.path : v.url)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
<SvgIcon
|
||||||
|
name="ele-Close"
|
||||||
|
class="layout-navbars-tagsview-ul-li-icon layout-icon-three"
|
||||||
|
v-if="!v.meta.isAffix"
|
||||||
|
@click.stop="closeCurrentTagsView(getThemeConfig.isShareTagsView ? v.path : v.url)"
|
||||||
|
/>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</el-scrollbar>
|
||||||
|
<Contextmenu :dropdown="dropdown" ref="contextmenuRef" @currentContextmenuClick="onCurrentContextmenuClick" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import {
|
||||||
|
toRefs,
|
||||||
|
reactive,
|
||||||
|
onMounted,
|
||||||
|
computed,
|
||||||
|
ref,
|
||||||
|
nextTick,
|
||||||
|
onBeforeUpdate,
|
||||||
|
onBeforeMount,
|
||||||
|
onUnmounted,
|
||||||
|
getCurrentInstance,
|
||||||
|
watch,
|
||||||
|
defineComponent,
|
||||||
|
} from 'vue';
|
||||||
|
import { useRoute, useRouter, onBeforeRouteUpdate } from 'vue-router';
|
||||||
|
import Sortable from 'sortablejs';
|
||||||
|
import { ElMessage } from 'element-plus';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useKeepALiveNames } from '/@/stores/keepAliveNames';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
import { isObjectValueEqual } from '/@/utils/arrayOperation';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import Contextmenu from '/@/layout/navBars/tagsView/contextmenu.vue';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface TagsViewState {
|
||||||
|
routeActive: string;
|
||||||
|
routePath: string | unknown;
|
||||||
|
dropdown: {
|
||||||
|
x: string | number;
|
||||||
|
y: string | number;
|
||||||
|
};
|
||||||
|
sortable: any;
|
||||||
|
tagsRefsIndex: number;
|
||||||
|
tagsViewList: any[];
|
||||||
|
tagsViewRoutesList: any[];
|
||||||
|
}
|
||||||
|
interface RouteParams {
|
||||||
|
path: string;
|
||||||
|
url: string;
|
||||||
|
query: object;
|
||||||
|
params: object;
|
||||||
|
}
|
||||||
|
interface CurrentContextmenu {
|
||||||
|
meta: {
|
||||||
|
isDynamic: boolean;
|
||||||
|
};
|
||||||
|
params: any;
|
||||||
|
query: any;
|
||||||
|
path: string;
|
||||||
|
contextMenuClickId: string | number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutTagsView',
|
||||||
|
components: { Contextmenu },
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const tagsRefs = ref<any[]>([]);
|
||||||
|
const scrollbarRef = ref();
|
||||||
|
const contextmenuRef = ref();
|
||||||
|
const tagsUlRef = ref();
|
||||||
|
const stores = useTagsViewRoutes();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { tagsViewRoutes } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
const storesKeepALiveNames = useKeepALiveNames();
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<TagsViewState>({
|
||||||
|
routeActive: '',
|
||||||
|
routePath: route.path,
|
||||||
|
dropdown: { x: '', y: '' },
|
||||||
|
sortable: '',
|
||||||
|
tagsRefsIndex: 0,
|
||||||
|
tagsViewList: [],
|
||||||
|
tagsViewRoutesList: [],
|
||||||
|
});
|
||||||
|
// 动态设置 tagsView 风格样式
|
||||||
|
const setTagsStyle = computed(() => {
|
||||||
|
return themeConfig.value.tagsStyle;
|
||||||
|
});
|
||||||
|
// 获取布局配置信息
|
||||||
|
const getThemeConfig = computed(() => {
|
||||||
|
return themeConfig.value;
|
||||||
|
});
|
||||||
|
// 设置 自定义 tagsView 名称、 自定义 tagsView 名称国际化
|
||||||
|
const setTagsViewNameI18n = computed(() => {
|
||||||
|
return (v: any) => {
|
||||||
|
return other.setTagsViewNameI18n(v);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
// 设置 tagsView 高亮
|
||||||
|
const isActive = (v: RouteParams) => {
|
||||||
|
if (getThemeConfig.value.isShareTagsView) {
|
||||||
|
return v.path === state.routePath;
|
||||||
|
} else {
|
||||||
|
if ((v.query && Object.keys(v.query).length) || (v.params && Object.keys(v.params).length)) {
|
||||||
|
// 普通传参
|
||||||
|
return v.url ? v.url === state.routeActive : v.path === state.routeActive;
|
||||||
|
} else {
|
||||||
|
// 通过 name 传参,params 取值,刷新页面参数消失
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I51RS9
|
||||||
|
return v.path === state.routePath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 存储 tagsViewList 到浏览器临时缓存中,页面刷新时,保留记录
|
||||||
|
const addBrowserSetSession = (tagsViewList: Array<object>) => {
|
||||||
|
Session.set('tagsViewList', tagsViewList);
|
||||||
|
};
|
||||||
|
// 获取 vuex 中的 tagsViewRoutes 列表
|
||||||
|
const getTagsViewRoutes = async () => {
|
||||||
|
state.routeActive = await setTagsViewHighlight(route);
|
||||||
|
state.routePath = (await route.meta.isDynamic) ? route.meta.isDynamicPath : route.path;
|
||||||
|
state.tagsViewList = [];
|
||||||
|
state.tagsViewRoutesList = tagsViewRoutes.value;
|
||||||
|
initTagsView();
|
||||||
|
};
|
||||||
|
// vuex 中获取路由信息:如果是设置了固定的(isAffix),进行初始化显示
|
||||||
|
const initTagsView = async () => {
|
||||||
|
if (Session.get('tagsViewList') && getThemeConfig.value.isCacheTagsView) {
|
||||||
|
state.tagsViewList = await Session.get('tagsViewList');
|
||||||
|
} else {
|
||||||
|
await state.tagsViewRoutesList.map((v: any) => {
|
||||||
|
if (v.meta.isAffix && !v.meta.isHide) {
|
||||||
|
v.url = setTagsViewHighlight(v);
|
||||||
|
state.tagsViewList.push({ ...v });
|
||||||
|
storesKeepALiveNames.addCachedView(v);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
await addTagsView(route.path, route);
|
||||||
|
}
|
||||||
|
// 初始化当前元素(li)的下标
|
||||||
|
getTagsRefsIndex(getThemeConfig.value.isShareTagsView ? state.routePath : state.routeActive);
|
||||||
|
};
|
||||||
|
// 处理可开启多标签详情,单标签详情(动态路由(xxx/:id/:name"),普通路由处理)
|
||||||
|
const solveAddTagsView = async (path: string, to?: any) => {
|
||||||
|
let isDynamicPath = to.meta.isDynamic ? to.meta.isDynamicPath : path;
|
||||||
|
let current = state.tagsViewList.filter(
|
||||||
|
(v: any) =>
|
||||||
|
v.path === isDynamicPath &&
|
||||||
|
isObjectValueEqual(
|
||||||
|
to.meta.isDynamic ? (v.params ? v.params : null) : v.query ? v.query : null,
|
||||||
|
to.meta.isDynamic ? (to?.params ? to?.params : null) : to?.query ? to?.query : null
|
||||||
|
)
|
||||||
|
);
|
||||||
|
if (current.length <= 0) {
|
||||||
|
// 防止:Avoid app logic that relies on enumerating keys on a component instance. The keys will be empty in production mode to avoid performance overhead.
|
||||||
|
let findItem = state.tagsViewRoutesList.find((v: any) => v.path === isDynamicPath);
|
||||||
|
if (!findItem) return false;
|
||||||
|
if (findItem.meta.isAffix) return false;
|
||||||
|
if (findItem.meta.isLink && !findItem.meta.isIframe) return false;
|
||||||
|
to.meta.isDynamic ? (findItem.params = to.params) : (findItem.query = to.query);
|
||||||
|
findItem.url = setTagsViewHighlight(findItem);
|
||||||
|
state.tagsViewList.push({ ...findItem });
|
||||||
|
await storesKeepALiveNames.addCachedView(findItem);
|
||||||
|
addBrowserSetSession(state.tagsViewList);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 处理单标签时,第二次的值未覆盖第一次的 tagsViewList 值(Session Storage)
|
||||||
|
const singleAddTagsView = (path: string, to?: any) => {
|
||||||
|
let isDynamicPath = to.meta.isDynamic ? to.meta.isDynamicPath : path;
|
||||||
|
state.tagsViewList.forEach((v) => {
|
||||||
|
if (
|
||||||
|
v.path === isDynamicPath &&
|
||||||
|
!isObjectValueEqual(
|
||||||
|
to.meta.isDynamic ? (v.params ? v.params : null) : v.query ? v.query : null,
|
||||||
|
to.meta.isDynamic ? (to?.params ? to?.params : null) : to?.query ? to?.query : null
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
to.meta.isDynamic ? (v.params = to.params) : (v.query = to.query);
|
||||||
|
v.url = setTagsViewHighlight(v);
|
||||||
|
addBrowserSetSession(state.tagsViewList);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 1、添加 tagsView:未设置隐藏(isHide)也添加到在 tagsView 中(可开启多标签详情,单标签详情)
|
||||||
|
const addTagsView = (path: string, to?: any) => {
|
||||||
|
// 防止拿取不到路由信息
|
||||||
|
nextTick(async () => {
|
||||||
|
// 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G
|
||||||
|
let item: any = '';
|
||||||
|
if (to && to.meta.isDynamic) {
|
||||||
|
// 动态路由(xxx/:id/:name"):参数不同,开启多个 tagsview
|
||||||
|
if (!getThemeConfig.value.isShareTagsView) await solveAddTagsView(path, to);
|
||||||
|
else await singleAddTagsView(path, to);
|
||||||
|
if (state.tagsViewList.some((v: any) => v.path === to.meta.isDynamicPath)) return false;
|
||||||
|
item = state.tagsViewRoutesList.find((v: any) => v.path === to.meta.isDynamicPath);
|
||||||
|
} else {
|
||||||
|
// 普通路由:参数不同,开启多个 tagsview
|
||||||
|
if (!getThemeConfig.value.isShareTagsView) await solveAddTagsView(path, to);
|
||||||
|
else await singleAddTagsView(path, to);
|
||||||
|
if (state.tagsViewList.some((v: any) => v.path === path)) return false;
|
||||||
|
item = state.tagsViewRoutesList.find((v: any) => v.path === path);
|
||||||
|
}
|
||||||
|
if (!item) return false;
|
||||||
|
if (item.meta.isLink && !item.meta.isIframe) return false;
|
||||||
|
if (to && to.meta.isDynamic) item.params = to?.params ? to?.params : route.params;
|
||||||
|
else item.query = to?.query ? to?.query : route.query;
|
||||||
|
item.url = setTagsViewHighlight(item);
|
||||||
|
await storesKeepALiveNames.addCachedView(item);
|
||||||
|
await state.tagsViewList.push({ ...item });
|
||||||
|
await addBrowserSetSession(state.tagsViewList);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 2、刷新当前 tagsView:
|
||||||
|
const refreshCurrentTagsView = async (fullPath: string) => {
|
||||||
|
const item = state.tagsViewList.find((v: any) => (getThemeConfig.value.isShareTagsView ? v.path === fullPath : v.url === fullPath));
|
||||||
|
if (item != null) {
|
||||||
|
await storesKeepALiveNames.delCachedView(item);
|
||||||
|
proxy.mittBus.emit('onTagsViewRefreshRouterView', fullPath);
|
||||||
|
if (item.meta.isKeepAlive) storesKeepALiveNames.addCachedView(item);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 3、关闭当前 tagsView:如果是设置了固定的(isAffix),不可以关闭
|
||||||
|
const closeCurrentTagsView = (path: string) => {
|
||||||
|
state.tagsViewList.map((v: any, k: number, arr: any) => {
|
||||||
|
if (!v.meta.isAffix) {
|
||||||
|
if (getThemeConfig.value.isShareTagsView ? v.path === path : v.url === path) {
|
||||||
|
storesKeepALiveNames.delCachedView(v);
|
||||||
|
state.tagsViewList.splice(k, 1);
|
||||||
|
setTimeout(() => {
|
||||||
|
if (state.tagsViewList.length === k && getThemeConfig.value.isShareTagsView ? state.routePath === path : state.routeActive === path) {
|
||||||
|
// 最后一个且高亮时
|
||||||
|
if (arr[arr.length - 1].meta.isDynamic) {
|
||||||
|
// 动态路由(xxx/:id/:name")
|
||||||
|
if (k !== arr.length) router.push({ name: arr[k].name, params: arr[k].params });
|
||||||
|
else router.push({ name: arr[arr.length - 1].name, params: arr[arr.length - 1].params });
|
||||||
|
} else {
|
||||||
|
// 普通路由
|
||||||
|
if (k !== arr.length) router.push({ path: arr[k].path, query: arr[k].query });
|
||||||
|
else router.push({ path: arr[arr.length - 1].path, query: arr[arr.length - 1].query });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 非最后一个且高亮时,跳转到下一个
|
||||||
|
if (state.tagsViewList.length !== k && getThemeConfig.value.isShareTagsView ? state.routePath === path : state.routeActive === path) {
|
||||||
|
if (arr[k].meta.isDynamic) {
|
||||||
|
// 动态路由(xxx/:id/:name")
|
||||||
|
router.push({ name: arr[k].name, params: arr[k].params });
|
||||||
|
} else {
|
||||||
|
// 普通路由
|
||||||
|
router.push({ path: arr[k].path, query: arr[k].query });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
addBrowserSetSession(state.tagsViewList);
|
||||||
|
};
|
||||||
|
// 4、关闭其它 tagsView:如果是设置了固定的(isAffix),不进行关闭
|
||||||
|
const closeOtherTagsView = (path: string) => {
|
||||||
|
if (Session.get('tagsViewList')) {
|
||||||
|
state.tagsViewList = [];
|
||||||
|
Session.get('tagsViewList').map((v: any) => {
|
||||||
|
if (v.meta.isAffix && !v.meta.isHide) {
|
||||||
|
v.url = setTagsViewHighlight(v);
|
||||||
|
storesKeepALiveNames.delOthersCachedViews(v);
|
||||||
|
state.tagsViewList.push({ ...v });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
addTagsView(path, route);
|
||||||
|
addBrowserSetSession(state.tagsViewList);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 5、关闭全部 tagsView:如果是设置了固定的(isAffix),不进行关闭
|
||||||
|
const closeAllTagsView = () => {
|
||||||
|
if (Session.get('tagsViewList')) {
|
||||||
|
storesKeepALiveNames.delAllCachedViews();
|
||||||
|
state.tagsViewList = [];
|
||||||
|
Session.get('tagsViewList').map((v: any) => {
|
||||||
|
if (v.meta.isAffix && !v.meta.isHide) {
|
||||||
|
v.url = setTagsViewHighlight(v);
|
||||||
|
state.tagsViewList.push({ ...v });
|
||||||
|
router.push({ path: state.tagsViewList[state.tagsViewList.length - 1].path });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
addBrowserSetSession(state.tagsViewList);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 6、开启当前页面全屏
|
||||||
|
const openCurrenFullscreen = async (path: string) => {
|
||||||
|
const item = state.tagsViewList.find((v: any) => (getThemeConfig.value.isShareTagsView ? v.path === path : v.url === path));
|
||||||
|
if (item.meta.isDynamic) await router.push({ name: item.name, params: item.params });
|
||||||
|
else await router.push({ name: item.name, query: item.query });
|
||||||
|
stores.setCurrenFullscreen(true);
|
||||||
|
};
|
||||||
|
// 当前项右键菜单点击,拿当前点击的路由路径对比 浏览器缓存中的 tagsView 路由数组,取当前点击项的详细路由信息
|
||||||
|
// 防止 tagsView 非当前页演示时,操作异常
|
||||||
|
const getCurrentRouteItem = (path: string, cParams: any) => {
|
||||||
|
const itemRoute = Session.get('tagsViewList') ? Session.get('tagsViewList') : state.tagsViewList;
|
||||||
|
return itemRoute.find((v: any) => {
|
||||||
|
if (
|
||||||
|
v.path === path &&
|
||||||
|
isObjectValueEqual(
|
||||||
|
v.meta.isDynamic ? (v.params ? v.params : null) : v.query ? v.query : null,
|
||||||
|
cParams && Object.keys(cParams ? cParams : {}).length > 0 ? cParams : null
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return v;
|
||||||
|
} else if (v.path === path && Object.keys(cParams ? cParams : {}).length <= 0) {
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 当前项右键菜单点击
|
||||||
|
const onCurrentContextmenuClick = async (item: CurrentContextmenu) => {
|
||||||
|
const cParams = item.meta.isDynamic ? item.params : item.query;
|
||||||
|
if (!getCurrentRouteItem(item.path, cParams)) return ElMessage({ type: 'warning', message: '请正确输入路径及完整参数(query、params)' });
|
||||||
|
const { path, name, params, query, meta, url } = getCurrentRouteItem(item.path, cParams);
|
||||||
|
switch (item.contextMenuClickId) {
|
||||||
|
case 0:
|
||||||
|
// 刷新当前
|
||||||
|
if (meta.isDynamic) await router.push({ name, params });
|
||||||
|
else await router.push({ path, query });
|
||||||
|
refreshCurrentTagsView(route.fullPath);
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
// 关闭当前
|
||||||
|
closeCurrentTagsView(getThemeConfig.value.isShareTagsView ? path : url);
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
// 关闭其它
|
||||||
|
if (meta.isDynamic) await router.push({ name, params });
|
||||||
|
else await router.push({ path, query });
|
||||||
|
closeOtherTagsView(path);
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
// 关闭全部
|
||||||
|
closeAllTagsView();
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
// 开启当前页面全屏
|
||||||
|
openCurrenFullscreen(getThemeConfig.value.isShareTagsView ? path : url);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 右键点击时:传 x,y 坐标值到子组件中(props)
|
||||||
|
const onContextmenu = (v: any, e: any) => {
|
||||||
|
const { clientX, clientY } = e;
|
||||||
|
state.dropdown.x = clientX;
|
||||||
|
state.dropdown.y = clientY;
|
||||||
|
contextmenuRef.value.openContextmenu(v);
|
||||||
|
};
|
||||||
|
// 当前的 tagsView 项点击时
|
||||||
|
const onTagsClick = (v: any, k: number) => {
|
||||||
|
state.tagsRefsIndex = k;
|
||||||
|
router.push(v);
|
||||||
|
};
|
||||||
|
// 处理 tagsView 高亮(多标签详情时使用,单标签详情未使用)
|
||||||
|
const setTagsViewHighlight = (v: any) => {
|
||||||
|
let params = v.query && Object.keys(v.query).length > 0 ? v.query : v.params;
|
||||||
|
if (!params || Object.keys(params).length <= 0) return v.path;
|
||||||
|
let path = '';
|
||||||
|
for (let i in params) {
|
||||||
|
path += params[i];
|
||||||
|
}
|
||||||
|
// 判断是否是动态路由(xxx/:id/:name")
|
||||||
|
return `${v.meta.isDynamic ? v.meta.isDynamicPath : v.path}-${path}`;
|
||||||
|
};
|
||||||
|
// 更新滚动条显示
|
||||||
|
const updateScrollbar = () => {
|
||||||
|
proxy.$refs.scrollbarRef.update();
|
||||||
|
};
|
||||||
|
// 鼠标滚轮滚动
|
||||||
|
const onHandleScroll = (e: any) => {
|
||||||
|
proxy.$refs.scrollbarRef.$refs.wrap$.scrollLeft += e.wheelDelta / 4;
|
||||||
|
};
|
||||||
|
// tagsView 横向滚动
|
||||||
|
const tagsViewmoveToCurrentTag = () => {
|
||||||
|
nextTick(() => {
|
||||||
|
if (tagsRefs.value.length <= 0) return false;
|
||||||
|
// 当前 li 元素
|
||||||
|
let liDom = tagsRefs.value[state.tagsRefsIndex];
|
||||||
|
// 当前 li 元素下标
|
||||||
|
let liIndex = state.tagsRefsIndex;
|
||||||
|
// 当前 ul 下 li 元素总长度
|
||||||
|
let liLength = tagsRefs.value.length;
|
||||||
|
// 最前 li
|
||||||
|
let liFirst: any = tagsRefs.value[0];
|
||||||
|
// 最后 li
|
||||||
|
let liLast: any = tagsRefs.value[tagsRefs.value.length - 1];
|
||||||
|
// 当前滚动条的值
|
||||||
|
let scrollRefs = proxy.$refs.scrollbarRef.$refs.wrap$;
|
||||||
|
// 当前滚动条滚动宽度
|
||||||
|
let scrollS = scrollRefs.scrollWidth;
|
||||||
|
// 当前滚动条偏移宽度
|
||||||
|
let offsetW = scrollRefs.offsetWidth;
|
||||||
|
// 当前滚动条偏移距离
|
||||||
|
let scrollL = scrollRefs.scrollLeft;
|
||||||
|
// 上一个 tags li dom
|
||||||
|
let liPrevTag: any = tagsRefs.value[state.tagsRefsIndex - 1];
|
||||||
|
// 下一个 tags li dom
|
||||||
|
let liNextTag: any = tagsRefs.value[state.tagsRefsIndex + 1];
|
||||||
|
// 上一个 tags li dom 的偏移距离
|
||||||
|
let beforePrevL: any = '';
|
||||||
|
// 下一个 tags li dom 的偏移距离
|
||||||
|
let afterNextL: any = '';
|
||||||
|
if (liDom === liFirst) {
|
||||||
|
// 头部
|
||||||
|
scrollRefs.scrollLeft = 0;
|
||||||
|
} else if (liDom === liLast) {
|
||||||
|
// 尾部
|
||||||
|
scrollRefs.scrollLeft = scrollS - offsetW;
|
||||||
|
} else {
|
||||||
|
// 非头/尾部
|
||||||
|
if (liIndex === 0) beforePrevL = liFirst.offsetLeft - 5;
|
||||||
|
else beforePrevL = liPrevTag?.offsetLeft - 5;
|
||||||
|
if (liIndex === liLength) afterNextL = liLast.offsetLeft + liLast.offsetWidth + 5;
|
||||||
|
else afterNextL = liNextTag.offsetLeft + liNextTag.offsetWidth + 5;
|
||||||
|
if (afterNextL > scrollL + offsetW) {
|
||||||
|
scrollRefs.scrollLeft = afterNextL - offsetW;
|
||||||
|
} else if (beforePrevL < scrollL) {
|
||||||
|
scrollRefs.scrollLeft = beforePrevL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 更新滚动条,防止不出现
|
||||||
|
updateScrollbar();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 获取 tagsView 的下标:用于处理 tagsView 点击时的横向滚动
|
||||||
|
const getTagsRefsIndex = (path: string | unknown) => {
|
||||||
|
nextTick(async () => {
|
||||||
|
// await 使用该写法,防止拿取不到 tagsViewList 列表数据不完整
|
||||||
|
let tagsViewList = await state.tagsViewList;
|
||||||
|
state.tagsRefsIndex = tagsViewList.findIndex((v: any) => {
|
||||||
|
if (getThemeConfig.value.isShareTagsView) {
|
||||||
|
return v.path === path;
|
||||||
|
} else {
|
||||||
|
return v.url === path;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 添加初始化横向滚动条移动到对应位置
|
||||||
|
tagsViewmoveToCurrentTag();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 设置 tagsView 可以进行拖拽
|
||||||
|
const initSortable = async () => {
|
||||||
|
const el = <HTMLElement>document.querySelector('.layout-navbars-tagsview-ul');
|
||||||
|
if (!el) return false;
|
||||||
|
state.sortable.el && state.sortable.destroy();
|
||||||
|
state.sortable = Sortable.create(el, {
|
||||||
|
animation: 300,
|
||||||
|
dataIdAttr: 'data-url',
|
||||||
|
disabled: getThemeConfig.value.isSortableTagsView ? false : true,
|
||||||
|
onEnd: () => {
|
||||||
|
const sortEndList: any = [];
|
||||||
|
state.sortable.toArray().map((val: any) => {
|
||||||
|
state.tagsViewList.map((v: any) => {
|
||||||
|
if (v.url === val) sortEndList.push({ ...v });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
addBrowserSetSession(sortEndList);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 拖动问题,https://gitee.com/lyt-top/vue-next-admin/issues/I3ZRRI
|
||||||
|
const onSortableResize = async () => {
|
||||||
|
await initSortable();
|
||||||
|
if (other.isMobile()) state.sortable.el && state.sortable.destroy();
|
||||||
|
};
|
||||||
|
// 页面加载前
|
||||||
|
onBeforeMount(() => {
|
||||||
|
// 初始化,防止手机端直接访问时还可以拖拽
|
||||||
|
onSortableResize();
|
||||||
|
// 拖动问题,https://gitee.com/lyt-top/vue-next-admin/issues/I3ZRRI
|
||||||
|
window.addEventListener('resize', onSortableResize);
|
||||||
|
// 监听非本页面调用 0 刷新当前,1 关闭当前,2 关闭其它,3 关闭全部 4 当前页全屏
|
||||||
|
proxy.mittBus.on('onCurrentContextmenuClick', (data: CurrentContextmenu) => {
|
||||||
|
onCurrentContextmenuClick(data);
|
||||||
|
});
|
||||||
|
// 监听布局配置界面开启/关闭拖拽
|
||||||
|
proxy.mittBus.on('openOrCloseSortable', () => {
|
||||||
|
initSortable();
|
||||||
|
});
|
||||||
|
// 监听布局配置开启 TagsView 共用,为了演示还原默认值
|
||||||
|
proxy.mittBus.on('openShareTagsView', () => {
|
||||||
|
if (getThemeConfig.value.isShareTagsView) {
|
||||||
|
router.push('/home');
|
||||||
|
state.tagsViewList = [];
|
||||||
|
state.tagsViewRoutesList.map((v: any) => {
|
||||||
|
if (v.meta.isAffix && !v.meta.isHide) {
|
||||||
|
v.url = setTagsViewHighlight(v);
|
||||||
|
state.tagsViewList.push({ ...v });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
// 取消非本页面调用监听
|
||||||
|
proxy.mittBus.off('onCurrentContextmenuClick', () => {});
|
||||||
|
// 取消监听布局配置界面开启/关闭拖拽
|
||||||
|
proxy.mittBus.off('openOrCloseSortable', () => {});
|
||||||
|
// 取消监听布局配置开启 TagsView 共用
|
||||||
|
proxy.mittBus.off('openShareTagsView', () => {});
|
||||||
|
// 取消窗口 resize 监听
|
||||||
|
window.removeEventListener('resize', onSortableResize);
|
||||||
|
});
|
||||||
|
// 页面更新时
|
||||||
|
onBeforeUpdate(() => {
|
||||||
|
tagsRefs.value = [];
|
||||||
|
});
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
// 初始化 pinia 中的 tagsViewRoutes 列表
|
||||||
|
getTagsViewRoutes();
|
||||||
|
initSortable();
|
||||||
|
});
|
||||||
|
// 路由更新时(组件内生命钩子)
|
||||||
|
onBeforeRouteUpdate(async (to) => {
|
||||||
|
state.routeActive = setTagsViewHighlight(to);
|
||||||
|
state.routePath = to.meta.isDynamic ? to.meta.isDynamicPath : to.path;
|
||||||
|
await addTagsView(to.path, to);
|
||||||
|
getTagsRefsIndex(getThemeConfig.value.isShareTagsView ? state.routePath : state.routeActive);
|
||||||
|
});
|
||||||
|
// 监听路由的变化,动态赋值给 tagsView
|
||||||
|
watch(
|
||||||
|
pinia.state,
|
||||||
|
(val) => {
|
||||||
|
if (val.tagsViewRoutes.tagsViewRoutes.length === state.tagsViewRoutesList.length) return false;
|
||||||
|
getTagsViewRoutes();
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
isActive,
|
||||||
|
onContextmenu,
|
||||||
|
onTagsClick,
|
||||||
|
tagsRefs,
|
||||||
|
contextmenuRef,
|
||||||
|
scrollbarRef,
|
||||||
|
tagsUlRef,
|
||||||
|
onHandleScroll,
|
||||||
|
getThemeConfig,
|
||||||
|
setTagsStyle,
|
||||||
|
setTagsViewNameI18n,
|
||||||
|
refreshCurrentTagsView,
|
||||||
|
closeCurrentTagsView,
|
||||||
|
onCurrentContextmenuClick,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-tagsview {
|
||||||
|
background-color: var(--el-color-white);
|
||||||
|
border-bottom: 1px solid var(--next-border-color-light);
|
||||||
|
position: relative;
|
||||||
|
z-index: 4;
|
||||||
|
:deep(.el-scrollbar__wrap) {
|
||||||
|
overflow-x: auto !important;
|
||||||
|
}
|
||||||
|
&-ul {
|
||||||
|
list-style: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
height: 34px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
color: var(--el-text-color-regular);
|
||||||
|
font-size: 12px;
|
||||||
|
white-space: nowrap;
|
||||||
|
padding: 0 15px;
|
||||||
|
&-li {
|
||||||
|
height: 26px;
|
||||||
|
line-height: 26px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
border: 1px solid var(--el-border-color-lighter);
|
||||||
|
padding: 0 15px;
|
||||||
|
margin-right: 5px;
|
||||||
|
border-radius: 2px;
|
||||||
|
position: relative;
|
||||||
|
z-index: 0;
|
||||||
|
cursor: pointer;
|
||||||
|
justify-content: space-between;
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--el-color-primary-light-9);
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
}
|
||||||
|
&-iconfont {
|
||||||
|
position: relative;
|
||||||
|
left: -5px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
&-icon {
|
||||||
|
border-radius: 100%;
|
||||||
|
position: relative;
|
||||||
|
height: 14px;
|
||||||
|
width: 14px;
|
||||||
|
text-align: center;
|
||||||
|
line-height: 14px;
|
||||||
|
right: -5px;
|
||||||
|
&:hover {
|
||||||
|
color: var(--el-color-white);
|
||||||
|
background-color: var(--el-color-primary-light-3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-icon-active {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.layout-icon-three {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.is-active {
|
||||||
|
color: var(--el-color-white);
|
||||||
|
background: var(--el-color-primary);
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
transition: border-color 3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 风格4
|
||||||
|
.tags-style-four {
|
||||||
|
.layout-navbars-tagsview-ul-li {
|
||||||
|
margin-right: 0 !important;
|
||||||
|
border: none !important;
|
||||||
|
position: relative;
|
||||||
|
border-radius: 3px !important;
|
||||||
|
.layout-icon-active {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.layout-icon-three {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
background: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.is-active {
|
||||||
|
background: none !important;
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 风格5
|
||||||
|
.tags-style-five {
|
||||||
|
align-items: flex-end;
|
||||||
|
.tags-style-five-svg {
|
||||||
|
-webkit-mask-box-image: url("data:image/svg+xml,%3Csvg width='68' height='34' viewBox='0 0 68 34' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='m27,0c-7.99582,0 -11.95105,0.00205 -12,12l0,6c0,8.284 -0.48549,16.49691 -8.76949,16.49691l54.37857,-0.11145c-8.284,0 -8.60908,-8.10146 -8.60908,-16.38546l0,-6c0.11145,-12.08445 -4.38441,-12 -12,-12l-13,0z' fill='%23409eff'/%3E%3C/svg%3E")
|
||||||
|
12 27 15;
|
||||||
|
}
|
||||||
|
.layout-navbars-tagsview-ul-li {
|
||||||
|
padding: 0 5px;
|
||||||
|
border-width: 15px 27px 15px;
|
||||||
|
border-style: solid;
|
||||||
|
border-color: transparent;
|
||||||
|
margin: 0 -15px;
|
||||||
|
.layout-icon-active,
|
||||||
|
.layout-navbars-tagsview-ul-li-iconfont,
|
||||||
|
.layout-navbars-tagsview-ul-li-refresh {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.layout-icon-three {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
@extend .tags-style-five-svg;
|
||||||
|
background: var(--el-color-primary-light-9);
|
||||||
|
color: unset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.is-active {
|
||||||
|
@extend .tags-style-five-svg;
|
||||||
|
background: var(--el-color-primary-light-9) !important;
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-navbars-tagsview-shadow {
|
||||||
|
box-shadow: rgb(0 21 41 / 4%) 0px 1px 4px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
157
web/src/layout/navMenu/horizontal.vue
Normal file
157
web/src/layout/navMenu/horizontal.vue
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
<template>
|
||||||
|
<div class="el-menu-horizontal-warp">
|
||||||
|
<el-scrollbar @wheel.native.prevent="onElMenuHorizontalScroll" ref="elMenuHorizontalScrollRef">
|
||||||
|
<el-menu router :default-active="defaultActive" :ellipsis="false" background-color="transparent" mode="horizontal">
|
||||||
|
<template v-for="val in menuLists">
|
||||||
|
<el-sub-menu :index="val.path" v-if="val.children && val.children.length > 0" :key="val.path">
|
||||||
|
<template #title>
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
<span>{{ $t(val.meta.title) }}</span>
|
||||||
|
</template>
|
||||||
|
<SubItem :chil="val.children" />
|
||||||
|
</el-sub-menu>
|
||||||
|
<template v-else>
|
||||||
|
<el-menu-item :index="val.path" :key="val.path">
|
||||||
|
<template #title v-if="!val.meta.isLink || (val.meta.isLink && val.meta.isIframe)">
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
{{ $t(val.meta.title) }}
|
||||||
|
</template>
|
||||||
|
<template #title v-else>
|
||||||
|
<a :href="val.meta.isLink" target="_blank" rel="opener" class="w100">
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
{{ $t(val.meta.title) }}
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
</el-menu-item>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
</el-menu>
|
||||||
|
</el-scrollbar>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, computed, defineComponent, getCurrentInstance, onMounted, nextTick, onBeforeMount } from 'vue';
|
||||||
|
import { useRoute, onBeforeRouteUpdate } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import SubItem from '/@/layout/navMenu/subItem.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'navMenuHorizontal',
|
||||||
|
components: { SubItem },
|
||||||
|
props: {
|
||||||
|
menuList: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive({
|
||||||
|
defaultActive: null,
|
||||||
|
});
|
||||||
|
// 获取父级菜单数据
|
||||||
|
const menuLists = computed(() => {
|
||||||
|
return <any>props.menuList;
|
||||||
|
});
|
||||||
|
// 设置横向滚动条可以鼠标滚轮滚动
|
||||||
|
const onElMenuHorizontalScroll = (e: any) => {
|
||||||
|
const eventDelta = e.wheelDelta || -e.deltaY * 40;
|
||||||
|
proxy.$refs.elMenuHorizontalScrollRef.$refs.wrap$.scrollLeft = proxy.$refs.elMenuHorizontalScrollRef.$refs.wrap$.scrollLeft + eventDelta / 4;
|
||||||
|
};
|
||||||
|
// 初始化数据,页面刷新时,滚动条滚动到对应位置
|
||||||
|
const initElMenuOffsetLeft = () => {
|
||||||
|
nextTick(() => {
|
||||||
|
let els: any = document.querySelector('.el-menu.el-menu--horizontal li.is-active');
|
||||||
|
if (!els) return false;
|
||||||
|
proxy.$refs.elMenuHorizontalScrollRef.$refs.wrap$.scrollLeft = els.offsetLeft;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 路由过滤递归函数
|
||||||
|
const filterRoutesFun = (arr: Array<string>) => {
|
||||||
|
return arr
|
||||||
|
.filter((item: any) => !item.meta.isHide)
|
||||||
|
.map((item: any) => {
|
||||||
|
item = Object.assign({}, item);
|
||||||
|
if (item.children) item.children = filterRoutesFun(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 传送当前子级数据到菜单中
|
||||||
|
const setSendClassicChildren = (path: string) => {
|
||||||
|
const currentPathSplit = path.split('/');
|
||||||
|
let currentData: any = {};
|
||||||
|
filterRoutesFun(routesList.value).map((v, k) => {
|
||||||
|
if (v.path === `/${currentPathSplit[1]}`) {
|
||||||
|
v['k'] = k;
|
||||||
|
currentData['item'] = [{ ...v }];
|
||||||
|
currentData['children'] = [{ ...v }];
|
||||||
|
if (v.children) currentData['children'] = v.children;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return currentData;
|
||||||
|
};
|
||||||
|
// 设置页面当前路由高亮
|
||||||
|
const setCurrentRouterHighlight = (currentRoute: any) => {
|
||||||
|
const { path, meta } = currentRoute;
|
||||||
|
if (themeConfig.value.layout === 'classic') {
|
||||||
|
(<any>state.defaultActive) = `/${path.split('/')[1]}`;
|
||||||
|
} else {
|
||||||
|
const pathSplit = meta.isDynamic ? meta.isDynamicPath.split('/') : path.split('/');
|
||||||
|
if (pathSplit.length >= 4 && meta.isHide) state.defaultActive = pathSplit.splice(0, 3).join('/');
|
||||||
|
else state.defaultActive = path;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 页面加载前
|
||||||
|
onBeforeMount(() => {
|
||||||
|
setCurrentRouterHighlight(route);
|
||||||
|
});
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initElMenuOffsetLeft();
|
||||||
|
});
|
||||||
|
// 路由更新时
|
||||||
|
onBeforeRouteUpdate((to) => {
|
||||||
|
// 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G
|
||||||
|
setCurrentRouterHighlight(to);
|
||||||
|
// 修复经典布局开启切割菜单时,点击tagsView后左侧导航菜单数据不变的问题
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
if (layout === 'classic' && isClassicSplitMenu) {
|
||||||
|
proxy.mittBus.emit('setSendClassicChildren', setSendClassicChildren(to.path));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
menuLists,
|
||||||
|
onElMenuHorizontalScroll,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.el-menu-horizontal-warp {
|
||||||
|
flex: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
margin-right: 30px;
|
||||||
|
:deep(.el-scrollbar__bar.is-vertical) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
:deep(a) {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.el-menu.el-menu--horizontal {
|
||||||
|
display: flex;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
48
web/src/layout/navMenu/subItem.vue
Normal file
48
web/src/layout/navMenu/subItem.vue
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<template>
|
||||||
|
<template v-for="val in chils">
|
||||||
|
<el-sub-menu :index="val.path" :key="val.path" v-if="val.children && val.children.length > 0">
|
||||||
|
<template #title>
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
<span>{{ $t(val.meta.title) }}</span>
|
||||||
|
</template>
|
||||||
|
<sub-item :chil="val.children" />
|
||||||
|
</el-sub-menu>
|
||||||
|
<template v-else>
|
||||||
|
<el-menu-item :index="val.path" :key="val.path">
|
||||||
|
<template v-if="!val.meta.isLink || (val.meta.isLink && val.meta.isIframe)">
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
<span>{{ $t(val.meta.title) }}</span>
|
||||||
|
</template>
|
||||||
|
<template v-else>
|
||||||
|
<a :href="val.meta.isLink" target="_blank" rel="opener" class="w100">
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
{{ $t(val.meta.title) }}
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
</el-menu-item>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'navMenuSubItem',
|
||||||
|
props: {
|
||||||
|
chil: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
// 获取父级菜单数据
|
||||||
|
const chils = computed(() => {
|
||||||
|
return <any>props.chil;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
chils,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
101
web/src/layout/navMenu/vertical.vue
Normal file
101
web/src/layout/navMenu/vertical.vue
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
<template>
|
||||||
|
<el-menu
|
||||||
|
router
|
||||||
|
:default-active="defaultActive"
|
||||||
|
background-color="transparent"
|
||||||
|
:collapse="isCollapse"
|
||||||
|
:unique-opened="getThemeConfig.isUniqueOpened"
|
||||||
|
:collapse-transition="false"
|
||||||
|
>
|
||||||
|
<template v-for="val in menuLists">
|
||||||
|
<el-sub-menu :index="val.path" v-if="val.children && val.children.length > 0" :key="val.path">
|
||||||
|
<template #title>
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
<span>{{ $t(val.meta.title) }}</span>
|
||||||
|
</template>
|
||||||
|
<SubItem :chil="val.children" />
|
||||||
|
</el-sub-menu>
|
||||||
|
<template v-else>
|
||||||
|
<el-menu-item :index="val.path" :key="val.path">
|
||||||
|
<SvgIcon :name="val.meta.icon" />
|
||||||
|
<template #title v-if="!val.meta.isLink || (val.meta.isLink && val.meta.isIframe)">
|
||||||
|
<span>{{ $t(val.meta.title) }}</span>
|
||||||
|
</template>
|
||||||
|
<template #title v-else>
|
||||||
|
<a :href="val.meta.isLink" target="_blank" rel="opener" class="w100">{{ $t(val.meta.title) }}</a>
|
||||||
|
</template>
|
||||||
|
</el-menu-item>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
</el-menu>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { toRefs, reactive, computed, defineComponent, onMounted, watch } from 'vue';
|
||||||
|
import { useRoute, onBeforeRouteUpdate } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import SubItem from '/@/layout/navMenu/subItem.vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'navMenuVertical',
|
||||||
|
components: { SubItem },
|
||||||
|
props: {
|
||||||
|
menuList: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
setup(props) {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive({
|
||||||
|
// 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G
|
||||||
|
defaultActive: route.meta.isDynamic ? route.meta.isDynamicPath : route.path,
|
||||||
|
isCollapse: false,
|
||||||
|
});
|
||||||
|
// 获取父级菜单数据
|
||||||
|
const menuLists = computed(() => {
|
||||||
|
return <any>props.menuList;
|
||||||
|
});
|
||||||
|
// 获取布局配置信息
|
||||||
|
const getThemeConfig = computed(() => {
|
||||||
|
return themeConfig.value;
|
||||||
|
});
|
||||||
|
// 菜单高亮(详情时,父级高亮)
|
||||||
|
const setParentHighlight = (currentRoute: any) => {
|
||||||
|
const { path, meta } = currentRoute;
|
||||||
|
const pathSplit = meta.isDynamic ? meta.isDynamicPath.split('/') : path.split('/');
|
||||||
|
if (pathSplit.length >= 4 && meta.isHide) return pathSplit.splice(0, 3).join('/');
|
||||||
|
else return path;
|
||||||
|
};
|
||||||
|
// 设置菜单的收起/展开
|
||||||
|
watch(
|
||||||
|
themeConfig.value,
|
||||||
|
() => {
|
||||||
|
document.body.clientWidth <= 1000 ? (state.isCollapse = false) : (state.isCollapse = themeConfig.value.isCollapse);
|
||||||
|
},
|
||||||
|
{
|
||||||
|
immediate: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
state.defaultActive = setParentHighlight(route);
|
||||||
|
});
|
||||||
|
// 路由更新时
|
||||||
|
onBeforeRouteUpdate((to) => {
|
||||||
|
// 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G
|
||||||
|
state.defaultActive = setParentHighlight(to);
|
||||||
|
const clientWidth = document.body.clientWidth;
|
||||||
|
if (clientWidth < 1000) themeConfig.value.isCollapse = false;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
menuLists,
|
||||||
|
getThemeConfig,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
66
web/src/layout/routerView/iframes.vue
Normal file
66
web/src/layout/routerView/iframes.vue
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-view-bg-white flex mt1" :style="{ height: `calc(100vh - ${setIframeHeight}`, border: 'none' }" v-loading="iframeLoading">
|
||||||
|
<iframe :src="iframeUrl" frameborder="0" height="100%" width="100%" ref="iframeDom" v-show="!iframeLoading"></iframe>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent, reactive, toRefs, onMounted, nextTick, watch, computed } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutIfameView',
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { isTagsViewCurrenFull } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive({
|
||||||
|
iframeDom: null as HTMLIFrameElement | null,
|
||||||
|
iframeLoading: true,
|
||||||
|
iframeUrl: '',
|
||||||
|
});
|
||||||
|
// 初始化页面加载 loading
|
||||||
|
const initIframeLoad = () => {
|
||||||
|
state.iframeUrl = <any>route.meta.isLink;
|
||||||
|
nextTick(() => {
|
||||||
|
state.iframeLoading = true;
|
||||||
|
const iframe = state.iframeDom;
|
||||||
|
if (!iframe) return false;
|
||||||
|
iframe.onload = () => {
|
||||||
|
state.iframeLoading = false;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 设置 iframe 的高度
|
||||||
|
const setIframeHeight = computed(() => {
|
||||||
|
let { isTagsview } = themeConfig.value;
|
||||||
|
if (isTagsViewCurrenFull.value) {
|
||||||
|
return `1px`;
|
||||||
|
} else {
|
||||||
|
if (isTagsview) return `86px`;
|
||||||
|
else return `51px`;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initIframeLoad();
|
||||||
|
});
|
||||||
|
// 监听路由变化,多个 iframe 时使用
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
initIframeLoad();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
setIframeHeight,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
61
web/src/layout/routerView/link.vue
Normal file
61
web/src/layout/routerView/link.vue
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-view-bg-white flex layout-view-link" :style="{ height: `calc(100vh - ${setLinkHeight}` }">
|
||||||
|
<a :href="currentRouteMeta.isLink" target="_blank" rel="opener" class="flex-margin">
|
||||||
|
{{ $t(currentRouteMeta.title) }}:{{ currentRouteMeta.isLink }}
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent, toRefs, reactive, computed, watch } from 'vue';
|
||||||
|
import { useRoute, RouteMeta } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface LinkViewState {
|
||||||
|
currentRouteMeta: {
|
||||||
|
isLink: string;
|
||||||
|
title: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
interface LinkViewRouteMeta extends RouteMeta {
|
||||||
|
isLink: string;
|
||||||
|
title: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutLinkView',
|
||||||
|
setup() {
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive<LinkViewState>({
|
||||||
|
currentRouteMeta: {
|
||||||
|
isLink: '',
|
||||||
|
title: '',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
// 设置 link 的高度
|
||||||
|
const setLinkHeight = computed(() => {
|
||||||
|
let { isTagsview } = themeConfig.value;
|
||||||
|
if (isTagsview) return `115px`;
|
||||||
|
else return `80px`;
|
||||||
|
});
|
||||||
|
// 监听路由的变化,设置内容
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
state.currentRouteMeta = <LinkViewRouteMeta>route.meta;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
immediate: true,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
setLinkHeight,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
88
web/src/layout/routerView/parent.vue
Normal file
88
web/src/layout/routerView/parent.vue
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
<template>
|
||||||
|
<div class="h100">
|
||||||
|
<router-view v-slot="{ Component }">
|
||||||
|
<transition :name="setTransitionName" mode="out-in">
|
||||||
|
<keep-alive :include="getKeepAliveNames">
|
||||||
|
<component :is="Component" :key="refreshRouterViewKey" class="w100" />
|
||||||
|
</keep-alive>
|
||||||
|
</transition>
|
||||||
|
</router-view>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { computed, defineComponent, toRefs, reactive, getCurrentInstance, onBeforeMount, onUnmounted, nextTick, watch, onMounted } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useKeepALiveNames } from '/@/stores/keepAliveNames';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
|
||||||
|
// 定义接口来定义对象的类型
|
||||||
|
interface ParentViewState {
|
||||||
|
refreshRouterViewKey: null | string;
|
||||||
|
keepAliveNameList: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
name: 'layoutParentView',
|
||||||
|
setup() {
|
||||||
|
const { proxy } = <any>getCurrentInstance();
|
||||||
|
const route = useRoute();
|
||||||
|
const storesKeepAliveNames = useKeepALiveNames();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { keepAliveNames, cachedViews } = storeToRefs(storesKeepAliveNames);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const state = reactive<ParentViewState>({
|
||||||
|
refreshRouterViewKey: null,
|
||||||
|
keepAliveNameList: [],
|
||||||
|
});
|
||||||
|
// 设置主界面切换动画
|
||||||
|
const setTransitionName = computed(() => {
|
||||||
|
return themeConfig.value.animation;
|
||||||
|
});
|
||||||
|
// 获取组件缓存列表(name值)
|
||||||
|
const getKeepAliveNames = computed(() => {
|
||||||
|
return themeConfig.value.isTagsview ? cachedViews.value : state.keepAliveNameList;
|
||||||
|
});
|
||||||
|
// 页面加载前,处理缓存,页面刷新时路由缓存处理
|
||||||
|
onBeforeMount(() => {
|
||||||
|
state.keepAliveNameList = keepAliveNames.value;
|
||||||
|
proxy.mittBus.on('onTagsViewRefreshRouterView', (fullPath: string) => {
|
||||||
|
state.keepAliveNameList = keepAliveNames.value.filter((name: string) => route.name !== name);
|
||||||
|
state.refreshRouterViewKey = null;
|
||||||
|
nextTick(() => {
|
||||||
|
state.refreshRouterViewKey = fullPath;
|
||||||
|
state.keepAliveNameList = keepAliveNames.value;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I58U75
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I59RXK
|
||||||
|
nextTick(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (themeConfig.value.isCacheTagsView) cachedViews.value = Session.get('tagsViewList')?.map((item: any) => item.name);
|
||||||
|
}, 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
proxy.mittBus.off('onTagsViewRefreshRouterView', () => {});
|
||||||
|
});
|
||||||
|
// 监听路由变化,防止 tagsView 多标签时,切换动画消失
|
||||||
|
watch(
|
||||||
|
() => route.fullPath,
|
||||||
|
() => {
|
||||||
|
state.refreshRouterViewKey = decodeURI(route.fullPath);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
setTransitionName,
|
||||||
|
getKeepAliveNames,
|
||||||
|
...toRefs(state),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
24
web/src/main.ts
Normal file
24
web/src/main.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { createApp } from 'vue';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import App from './App.vue';
|
||||||
|
import router from './router';
|
||||||
|
import { directive } from '/@/utils/directive';
|
||||||
|
import { i18n } from '/@/i18n/index';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
|
||||||
|
import ElementPlus from 'element-plus';
|
||||||
|
import 'element-plus/dist/index.css';
|
||||||
|
import '/@/theme/index.scss';
|
||||||
|
import mitt from 'mitt';
|
||||||
|
import VueGridLayout from 'vue-grid-layout';
|
||||||
|
|
||||||
|
import fastCrud from './settings.ts'
|
||||||
|
|
||||||
|
const app = createApp(App);
|
||||||
|
|
||||||
|
directive(app);
|
||||||
|
other.elSvg(app);
|
||||||
|
|
||||||
|
app.use(pinia).use(router).use(ElementPlus, { i18n: i18n.global.t }).use(i18n).use(VueGridLayout).use(fastCrud).mount('#app');
|
||||||
|
|
||||||
|
app.config.globalProperties.mittBus = mitt();
|
||||||
157
web/src/router/backEnd.ts
Normal file
157
web/src/router/backEnd.ts
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
import { RouteRecordRaw } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { useRequestOldRoutes } from '/@/stores/requestOldRoutes';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
import { NextLoading } from '/@/utils/loading';
|
||||||
|
import { dynamicRoutes, notFoundAndNoPower } from '/@/router/route';
|
||||||
|
import { formatTwoStageRoutes, formatFlatteningRoutes, router } from '/@/router/index';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import { useMenuApi } from '/@/api/menu/index';
|
||||||
|
|
||||||
|
const menuApi = useMenuApi();
|
||||||
|
|
||||||
|
const layouModules: any = import.meta.glob('../layout/routerView/*.{vue,tsx}');
|
||||||
|
const viewsModules: any = import.meta.glob('../views/**/*.{vue,tsx}');
|
||||||
|
|
||||||
|
// 后端控制路由
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取目录下的 .vue、.tsx 全部文件
|
||||||
|
* @method import.meta.glob
|
||||||
|
* @link 参考:https://cn.vitejs.dev/guide/features.html#json
|
||||||
|
*/
|
||||||
|
const dynamicViewsModules: Record<string, Function> = Object.assign({}, { ...layouModules }, { ...viewsModules });
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端控制路由:初始化方法,防止刷新时路由丢失
|
||||||
|
* @method NextLoading 界面 loading 动画开始执行
|
||||||
|
* @method useUserInfo().setUserInfos() 触发初始化用户信息 pinia
|
||||||
|
* @method useRequestOldRoutes().setRequestOldRoutes() 存储接口原始路由(未处理component),根据需求选择使用
|
||||||
|
* @method setAddRoute 添加动态路由
|
||||||
|
* @method setFilterMenuAndCacheTagsViewRoutes 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
*/
|
||||||
|
export async function initBackEndControlRoutes() {
|
||||||
|
// 界面 loading 动画开始执行
|
||||||
|
if (window.nextLoading === undefined) NextLoading.start();
|
||||||
|
// 无 token 停止执行下一步
|
||||||
|
if (!Session.get('token')) return false;
|
||||||
|
// 触发初始化用户信息 pinia
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP
|
||||||
|
await useUserInfo().setUserInfos();
|
||||||
|
// 获取路由菜单数据
|
||||||
|
const res = await getBackEndControlRoutes();
|
||||||
|
// 存储接口原始路由(未处理component),根据需求选择使用
|
||||||
|
useRequestOldRoutes().setRequestOldRoutes(JSON.parse(JSON.stringify(res.data)));
|
||||||
|
// 处理路由(component),替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
|
||||||
|
dynamicRoutes[0].children = await backEndComponent(res.data);
|
||||||
|
// 添加动态路由
|
||||||
|
await setAddRoute();
|
||||||
|
// 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
await setFilterMenuAndCacheTagsViewRoutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
* @description 用于左侧菜单、横向菜单的显示
|
||||||
|
* @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
|
||||||
|
*/
|
||||||
|
export function setFilterMenuAndCacheTagsViewRoutes() {
|
||||||
|
const storesRoutesList = useRoutesList(pinia);
|
||||||
|
storesRoutesList.setRoutesList(dynamicRoutes[0].children as any);
|
||||||
|
setCacheTagsViewRoutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 缓存多级嵌套数组处理后的一维数组
|
||||||
|
* @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
|
||||||
|
*/
|
||||||
|
export function setCacheTagsViewRoutes() {
|
||||||
|
const storesTagsView = useTagsViewRoutes(pinia);
|
||||||
|
storesTagsView.setTagsViewRoutes(formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes))[0].children);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 处理路由格式及添加捕获所有路由或 404 Not found 路由
|
||||||
|
* @description 替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
|
||||||
|
* @returns 返回替换后的路由数组
|
||||||
|
*/
|
||||||
|
export function setFilterRouteEnd() {
|
||||||
|
let filterRouteEnd: any = formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes));
|
||||||
|
filterRouteEnd[0].children = [...filterRouteEnd[0].children, ...notFoundAndNoPower];
|
||||||
|
return filterRouteEnd;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 添加动态路由
|
||||||
|
* @method router.addRoute
|
||||||
|
* @description 此处循环为 dynamicRoutes(/@/router/route)第一个顶级 children 的路由一维数组,非多级嵌套
|
||||||
|
* @link 参考:https://next.router.vuejs.org/zh/api/#addroute
|
||||||
|
*/
|
||||||
|
export async function setAddRoute() {
|
||||||
|
await setFilterRouteEnd().forEach((route: RouteRecordRaw) => {
|
||||||
|
router.addRoute(route);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 请求后端路由菜单接口
|
||||||
|
* @description isRequestRoutes 为 true,则开启后端控制路由
|
||||||
|
* @returns 返回后端路由菜单数据
|
||||||
|
*/
|
||||||
|
export function getBackEndControlRoutes() {
|
||||||
|
// 模拟 admin 与 test
|
||||||
|
const stores = useUserInfo(pinia);
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
const auth = userInfos.value.roles[0];
|
||||||
|
// 管理员 admin
|
||||||
|
if (auth === 'admin') return menuApi.getMenuAdmin();
|
||||||
|
// 其它用户 test
|
||||||
|
else return menuApi.getMenuTest();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重新请求后端路由菜单接口
|
||||||
|
* @description 用于菜单管理界面刷新菜单(未进行测试)
|
||||||
|
* @description 路径:/src/views/system/menu/component/addMenu.vue
|
||||||
|
*/
|
||||||
|
export function setBackEndControlRefreshRoutes() {
|
||||||
|
getBackEndControlRoutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端路由 component 转换
|
||||||
|
* @param routes 后端返回的路由表数组
|
||||||
|
* @returns 返回处理成函数后的 component
|
||||||
|
*/
|
||||||
|
export function backEndComponent(routes: any) {
|
||||||
|
if (!routes) return;
|
||||||
|
return routes.map((item: any) => {
|
||||||
|
if (item.component) item.component = dynamicImport(dynamicViewsModules, item.component as string);
|
||||||
|
item.children && backEndComponent(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端路由 component 转换函数
|
||||||
|
* @param dynamicViewsModules 获取目录下的 .vue、.tsx 全部文件
|
||||||
|
* @param component 当前要处理项 component
|
||||||
|
* @returns 返回处理成函数后的 component
|
||||||
|
*/
|
||||||
|
export function dynamicImport(dynamicViewsModules: Record<string, Function>, component: string) {
|
||||||
|
const keys = Object.keys(dynamicViewsModules);
|
||||||
|
const matchKeys = keys.filter((key) => {
|
||||||
|
const k = key.replace(/..\/views|../, '');
|
||||||
|
return k.startsWith(`${component}`) || k.startsWith(`/${component}`);
|
||||||
|
});
|
||||||
|
if (matchKeys?.length === 1) {
|
||||||
|
const matchKey = matchKeys[0];
|
||||||
|
return dynamicViewsModules[matchKey];
|
||||||
|
}
|
||||||
|
if (matchKeys?.length > 1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
148
web/src/router/frontEnd.ts
Normal file
148
web/src/router/frontEnd.ts
Normal file
@@ -0,0 +1,148 @@
|
|||||||
|
import { RouteRecordRaw } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { formatTwoStageRoutes, formatFlatteningRoutes, router } from '/@/router/index';
|
||||||
|
import { dynamicRoutes, notFoundAndNoPower } from '/@/router/route';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { NextLoading } from '/@/utils/loading';
|
||||||
|
|
||||||
|
// 前端控制路由
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 前端控制路由:初始化方法,防止刷新时路由丢失
|
||||||
|
* @method NextLoading 界面 loading 动画开始执行
|
||||||
|
* @method useUserInfo(pinia).setUserInfos() 触发初始化用户信息 pinia
|
||||||
|
* @method setAddRoute 添加动态路由
|
||||||
|
* @method setFilterMenuAndCacheTagsViewRoutes 设置递归过滤有权限的路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
*/
|
||||||
|
export async function initFrontEndControlRoutes() {
|
||||||
|
// 界面 loading 动画开始执行
|
||||||
|
if (window.nextLoading === undefined) NextLoading.start();
|
||||||
|
// 无 token 停止执行下一步
|
||||||
|
if (!Session.get('token')) return false;
|
||||||
|
// 触发初始化用户信息 pinia
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP
|
||||||
|
await useUserInfo(pinia).setUserInfos();
|
||||||
|
// 添加动态路由
|
||||||
|
await setAddRoute();
|
||||||
|
// 设置递归过滤有权限的路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
await setFilterMenuAndCacheTagsViewRoutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 添加动态路由
|
||||||
|
* @method router.addRoute
|
||||||
|
* @description 此处循环为 dynamicRoutes(/@/router/route)第一个顶级 children 的路由一维数组,非多级嵌套
|
||||||
|
* @link 参考:https://next.router.vuejs.org/zh/api/#addroute
|
||||||
|
*/
|
||||||
|
export async function setAddRoute() {
|
||||||
|
await setFilterRouteEnd().forEach((route: RouteRecordRaw) => {
|
||||||
|
router.addRoute(route);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除/重置路由
|
||||||
|
* @method router.removeRoute
|
||||||
|
* @description 此处循环为 dynamicRoutes(/@/router/route)第一个顶级 children 的路由一维数组,非多级嵌套
|
||||||
|
* @link 参考:https://next.router.vuejs.org/zh/api/#push
|
||||||
|
*/
|
||||||
|
export async function frontEndsResetRoute() {
|
||||||
|
await setFilterRouteEnd().forEach((route: RouteRecordRaw) => {
|
||||||
|
const routeName: any = route.name;
|
||||||
|
router.hasRoute(routeName) && router.removeRoute(routeName);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取有当前用户权限标识的路由数组,进行对原路由的替换
|
||||||
|
* @description 替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
|
||||||
|
* @returns 返回替换后的路由数组
|
||||||
|
*/
|
||||||
|
export function setFilterRouteEnd() {
|
||||||
|
let filterRouteEnd: any = formatTwoStageRoutes(formatFlatteningRoutes(dynamicRoutes));
|
||||||
|
filterRouteEnd[0].children = [...setFilterRoute(filterRouteEnd[0].children), ...notFoundAndNoPower];
|
||||||
|
return filterRouteEnd;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取当前用户权限标识去比对路由表(未处理成多级嵌套路由)
|
||||||
|
* @description 这里主要用于动态路由的添加,router.addRoute
|
||||||
|
* @link 参考:https://next.router.vuejs.org/zh/api/#addroute
|
||||||
|
* @param chil dynamicRoutes(/@/router/route)第一个顶级 children 的下路由集合
|
||||||
|
* @returns 返回有当前用户权限标识的路由数组
|
||||||
|
*/
|
||||||
|
export function setFilterRoute(chil: any) {
|
||||||
|
const stores = useUserInfo(pinia);
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
let filterRoute: any = [];
|
||||||
|
chil.forEach((route: any) => {
|
||||||
|
if (route.meta.roles) {
|
||||||
|
route.meta.roles.forEach((metaRoles: any) => {
|
||||||
|
userInfos.value.roles.forEach((roles: any) => {
|
||||||
|
if (metaRoles === roles) filterRoute.push({ ...route });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return filterRoute;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 缓存多级嵌套数组处理后的一维数组
|
||||||
|
* @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
|
||||||
|
*/
|
||||||
|
export function setCacheTagsViewRoutes() {
|
||||||
|
// 获取有权限的路由,否则 tagsView、菜单搜索中无权限的路由也将显示
|
||||||
|
const stores = useUserInfo(pinia);
|
||||||
|
const storesTagsView = useTagsViewRoutes(pinia);
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
let rolesRoutes = setFilterHasRolesMenu(dynamicRoutes, userInfos.value.roles);
|
||||||
|
// 添加到 pinia setTagsViewRoutes 中
|
||||||
|
storesTagsView.setTagsViewRoutes(formatTwoStageRoutes(formatFlatteningRoutes(rolesRoutes))[0].children);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置递归过滤有权限的路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
|
* @description 用于左侧菜单、横向菜单的显示
|
||||||
|
* @description 用于 tagsView、菜单搜索中:未过滤隐藏的(isHide)
|
||||||
|
*/
|
||||||
|
export function setFilterMenuAndCacheTagsViewRoutes() {
|
||||||
|
const stores = useUserInfo(pinia);
|
||||||
|
const storesRoutesList = useRoutesList(pinia);
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
storesRoutesList.setRoutesList(setFilterHasRolesMenu(dynamicRoutes[0].children, userInfos.value.roles));
|
||||||
|
setCacheTagsViewRoutes();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 判断路由 `meta.roles` 中是否包含当前登录用户权限字段
|
||||||
|
* @param roles 用户权限标识,在 userInfos(用户信息)的 roles(登录页登录时缓存到浏览器)数组
|
||||||
|
* @param route 当前循环时的路由项
|
||||||
|
* @returns 返回对比后有权限的路由项
|
||||||
|
*/
|
||||||
|
export function hasRoles(roles: any, route: any) {
|
||||||
|
if (route.meta && route.meta.roles) return roles.some((role: any) => route.meta.roles.includes(role));
|
||||||
|
else return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取当前用户权限标识去比对路由表,设置递归过滤有权限的路由
|
||||||
|
* @param routes 当前路由 children
|
||||||
|
* @param roles 用户权限标识,在 userInfos(用户信息)的 roles(登录页登录时缓存到浏览器)数组
|
||||||
|
* @returns 返回有权限的路由数组 `meta.roles` 中控制
|
||||||
|
*/
|
||||||
|
export function setFilterHasRolesMenu(routes: any, roles: any) {
|
||||||
|
const menu: any = [];
|
||||||
|
routes.forEach((route: any) => {
|
||||||
|
const item = { ...route };
|
||||||
|
if (hasRoles(roles, item)) {
|
||||||
|
if (item.children) item.children = setFilterHasRolesMenu(item.children, roles);
|
||||||
|
menu.push(item);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return menu;
|
||||||
|
}
|
||||||
131
web/src/router/index.ts
Normal file
131
web/src/router/index.ts
Normal file
@@ -0,0 +1,131 @@
|
|||||||
|
import { createRouter, createWebHashHistory } from 'vue-router';
|
||||||
|
import NProgress from 'nprogress';
|
||||||
|
import 'nprogress/nprogress.css';
|
||||||
|
import pinia from '/@/stores/index';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useKeepALiveNames } from '/@/stores/keepAliveNames';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
import { staticRoutes } from '/@/router/route';
|
||||||
|
import { initFrontEndControlRoutes } from '/@/router/frontEnd';
|
||||||
|
import { initBackEndControlRoutes } from '/@/router/backEnd';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 1、前端控制路由时:isRequestRoutes 为 false,需要写 roles,需要走 setFilterRoute 方法。
|
||||||
|
* 2、后端控制路由时:isRequestRoutes 为 true,不需要写 roles,不需要走 setFilterRoute 方法),
|
||||||
|
* 相关方法已拆解到对应的 `backEnd.ts` 与 `frontEnd.ts`(他们互不影响,不需要同时改 2 个文件)。
|
||||||
|
* 特别说明:
|
||||||
|
* 1、前端控制:路由菜单由前端去写(无菜单管理界面,有角色管理界面),角色管理中有 roles 属性,需返回到 userInfo 中。
|
||||||
|
* 2、后端控制:路由菜单由后端返回(有菜单管理界面、有角色管理界面)
|
||||||
|
*/
|
||||||
|
|
||||||
|
// 读取 `/src/stores/themeConfig.ts` 是否开启后端控制路由配置
|
||||||
|
const storesThemeConfig = useThemeConfig(pinia);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { isRequestRoutes } = themeConfig.value;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建一个可以被 Vue 应用程序使用的路由实例
|
||||||
|
* @method createRouter(options: RouterOptions): Router
|
||||||
|
* @link 参考:https://next.router.vuejs.org/zh/api/#createrouter
|
||||||
|
*/
|
||||||
|
export const router = createRouter({
|
||||||
|
history: createWebHashHistory(),
|
||||||
|
routes: staticRoutes,
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 路由多级嵌套数组处理成一维数组
|
||||||
|
* @param arr 传入路由菜单数据数组
|
||||||
|
* @returns 返回处理后的一维路由菜单数组
|
||||||
|
*/
|
||||||
|
export function formatFlatteningRoutes(arr: any) {
|
||||||
|
if (arr.length <= 0) return false;
|
||||||
|
for (let i = 0; i < arr.length; i++) {
|
||||||
|
if (arr[i].children) {
|
||||||
|
arr = arr.slice(0, i + 1).concat(arr[i].children, arr.slice(i + 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return arr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 一维数组处理成多级嵌套数组(只保留二级:也就是二级以上全部处理成只有二级,keep-alive 支持二级缓存)
|
||||||
|
* @description isKeepAlive 处理 `name` 值,进行缓存。顶级关闭,全部不缓存
|
||||||
|
* @link 参考:https://v3.cn.vuejs.org/api/built-in-components.html#keep-alive
|
||||||
|
* @param arr 处理后的一维路由菜单数组
|
||||||
|
* @returns 返回将一维数组重新处理成 `定义动态路由(dynamicRoutes)` 的格式
|
||||||
|
*/
|
||||||
|
export function formatTwoStageRoutes(arr: any) {
|
||||||
|
if (arr.length <= 0) return false;
|
||||||
|
const newArr: any = [];
|
||||||
|
const cacheList: Array<string> = [];
|
||||||
|
arr.forEach((v: any) => {
|
||||||
|
if (v.path === '/') {
|
||||||
|
newArr.push({ component: v.component, name: v.name, path: v.path, redirect: v.redirect, meta: v.meta, children: [] });
|
||||||
|
} else {
|
||||||
|
// 判断是否是动态路由(xx/:id/:name),用于 tagsView 等中使用
|
||||||
|
// 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I3YX6G
|
||||||
|
if (v.path.indexOf('/:') > -1) {
|
||||||
|
v.meta['isDynamic'] = true;
|
||||||
|
v.meta['isDynamicPath'] = v.path;
|
||||||
|
}
|
||||||
|
newArr[0].children.push({ ...v });
|
||||||
|
// 存 name 值,keep-alive 中 include 使用,实现路由的缓存
|
||||||
|
// 路径:/@/layout/routerView/parent.vue
|
||||||
|
if (newArr[0].meta.isKeepAlive && v.meta.isKeepAlive) {
|
||||||
|
cacheList.push(v.name);
|
||||||
|
const stores = useKeepALiveNames(pinia);
|
||||||
|
stores.setCacheKeepAlive(cacheList);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return newArr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 路由加载前
|
||||||
|
router.beforeEach(async (to, from, next) => {
|
||||||
|
NProgress.configure({ showSpinner: false });
|
||||||
|
if (to.meta.title) NProgress.start();
|
||||||
|
const token = Session.get('token');
|
||||||
|
if (to.path === '/login' && !token) {
|
||||||
|
next();
|
||||||
|
NProgress.done();
|
||||||
|
} else {
|
||||||
|
if (!token) {
|
||||||
|
next(`/login?redirect=${to.path}¶ms=${JSON.stringify(to.query ? to.query : to.params)}`);
|
||||||
|
Session.clear();
|
||||||
|
NProgress.done();
|
||||||
|
} else if (token && to.path === '/login') {
|
||||||
|
next('/home');
|
||||||
|
NProgress.done();
|
||||||
|
} else {
|
||||||
|
const storesRoutesList = useRoutesList(pinia);
|
||||||
|
const { routesList } = storeToRefs(storesRoutesList);
|
||||||
|
if (routesList.value.length === 0) {
|
||||||
|
if (isRequestRoutes) {
|
||||||
|
// 后端控制路由:路由数据初始化,防止刷新时丢失
|
||||||
|
await initBackEndControlRoutes();
|
||||||
|
// 动态添加路由:防止非首页刷新时跳转回首页的问题
|
||||||
|
// 确保 addRoute() 时动态添加的路由已经被完全加载上去
|
||||||
|
next({ ...to, replace: true });
|
||||||
|
} else {
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP
|
||||||
|
await initFrontEndControlRoutes();
|
||||||
|
next({ ...to, replace: true });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 路由加载后
|
||||||
|
router.afterEach(() => {
|
||||||
|
NProgress.done();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 导出路由
|
||||||
|
export default router;
|
||||||
1210
web/src/router/route.ts
Normal file
1210
web/src/router/route.ts
Normal file
File diff suppressed because it is too large
Load Diff
42
web/src/settings.ts
Normal file
42
web/src/settings.ts
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
// 引入fast-crud
|
||||||
|
import {FastCrud} from "@fast-crud/fast-crud";
|
||||||
|
import "@fast-crud/fast-crud/dist/style.css";
|
||||||
|
|
||||||
|
// element
|
||||||
|
import ui from "@fast-crud/ui-element";
|
||||||
|
import {request} from "/@/utils/service.ts";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
async install(app: any, options: any) {
|
||||||
|
// 先安装ui
|
||||||
|
app.use(ui);
|
||||||
|
// 然后安装FastCrud
|
||||||
|
app.use(FastCrud, {
|
||||||
|
//i18n, //i18n配置,可选,默认使用中文,具体用法请看demo里的 src/i18n/index.js 文件
|
||||||
|
// 此处配置公共的dictRequest(字典请求)
|
||||||
|
async dictRequest({dict}:any) {
|
||||||
|
return await request({url: dict.url}); //根据dict的url,异步返回一个字典数组
|
||||||
|
},
|
||||||
|
//公共crud配置
|
||||||
|
commonOptions() {
|
||||||
|
return {
|
||||||
|
request: {
|
||||||
|
//接口请求配置
|
||||||
|
//你项目后台接口大概率与fast-crud所需要的返回结构不一致,所以需要配置此项
|
||||||
|
//请参考文档http://fast-crud.docmirror.cn/api/crud-options/request.html
|
||||||
|
transformQuery: ({page, form, sort}:any) => {
|
||||||
|
//转换为你pageRequest所需要的请求参数结构
|
||||||
|
return {page, form, sort};
|
||||||
|
},
|
||||||
|
transformRes: ({res}:any) => {
|
||||||
|
//将pageRequest的返回数据,转换为fast-crud所需要的格式
|
||||||
|
//return {records,currentPage,pageSize,total};
|
||||||
|
return {...res}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
8
web/src/stores/index.ts
Normal file
8
web/src/stores/index.ts
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
// https://pinia.vuejs.org/
|
||||||
|
import { createPinia } from 'pinia';
|
||||||
|
|
||||||
|
// 创建
|
||||||
|
const pinia = createPinia();
|
||||||
|
|
||||||
|
// 导出
|
||||||
|
export default pinia;
|
||||||
90
web/src/stores/interface/index.ts
Normal file
90
web/src/stores/interface/index.ts
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
/**
|
||||||
|
* 定义接口来定义对象的类型
|
||||||
|
* `stores` 全部类型定义在这里
|
||||||
|
*/
|
||||||
|
|
||||||
|
// 用户信息
|
||||||
|
export interface UserInfosState {
|
||||||
|
authBtnList: string[];
|
||||||
|
photo: string;
|
||||||
|
roles: string[];
|
||||||
|
time: number;
|
||||||
|
userName: string;
|
||||||
|
}
|
||||||
|
export interface UserInfosStates {
|
||||||
|
userInfos: UserInfosState;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 路由缓存列表
|
||||||
|
export interface KeepAliveNamesState {
|
||||||
|
keepAliveNames: string[];
|
||||||
|
cachedViews: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 后端返回原始路由(未处理时)
|
||||||
|
export interface RequestOldRoutesState {
|
||||||
|
requestOldRoutes: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
// TagsView 路由列表
|
||||||
|
export interface TagsViewRoutesState {
|
||||||
|
tagsViewRoutes: string[];
|
||||||
|
isTagsViewCurrenFull: Boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 路由列表
|
||||||
|
export interface RoutesListState {
|
||||||
|
routesList: string[];
|
||||||
|
isColumnsMenuHover: Boolean;
|
||||||
|
isColumnsNavHover: Boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 布局配置
|
||||||
|
export interface ThemeConfigState {
|
||||||
|
isDrawer: boolean;
|
||||||
|
primary: string;
|
||||||
|
topBar: string;
|
||||||
|
topBarColor: string;
|
||||||
|
isTopBarColorGradual: boolean;
|
||||||
|
menuBar: string;
|
||||||
|
menuBarColor: string;
|
||||||
|
isMenuBarColorGradual: boolean;
|
||||||
|
columnsMenuBar: string;
|
||||||
|
columnsMenuBarColor: string;
|
||||||
|
isColumnsMenuBarColorGradual: boolean;
|
||||||
|
isCollapse: boolean;
|
||||||
|
isUniqueOpened: boolean;
|
||||||
|
isFixedHeader: boolean;
|
||||||
|
isFixedHeaderChange: boolean;
|
||||||
|
isClassicSplitMenu: boolean;
|
||||||
|
isLockScreen: boolean;
|
||||||
|
lockScreenTime: number;
|
||||||
|
isShowLogo: boolean;
|
||||||
|
isShowLogoChange: boolean;
|
||||||
|
isBreadcrumb: boolean;
|
||||||
|
isTagsview: boolean;
|
||||||
|
isBreadcrumbIcon: boolean;
|
||||||
|
isTagsviewIcon: boolean;
|
||||||
|
isCacheTagsView: boolean;
|
||||||
|
isSortableTagsView: boolean;
|
||||||
|
isShareTagsView: boolean;
|
||||||
|
isFooter: boolean;
|
||||||
|
isGrayscale: boolean;
|
||||||
|
isInvert: boolean;
|
||||||
|
isIsDark: boolean;
|
||||||
|
isWartermark: boolean;
|
||||||
|
wartermarkText: string;
|
||||||
|
tagsStyle: string;
|
||||||
|
animation: string;
|
||||||
|
columnsAsideStyle: string;
|
||||||
|
columnsAsideLayout: string;
|
||||||
|
layout: string;
|
||||||
|
isRequestRoutes: boolean;
|
||||||
|
globalTitle: string;
|
||||||
|
globalViceTitle: string;
|
||||||
|
globalI18n: string;
|
||||||
|
globalComponentSize: string;
|
||||||
|
}
|
||||||
|
export interface ThemeConfigStates {
|
||||||
|
themeConfig: ThemeConfigState;
|
||||||
|
}
|
||||||
37
web/src/stores/keepAliveNames.ts
Normal file
37
web/src/stores/keepAliveNames.ts
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import { KeepAliveNamesState } from './interface';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 路由缓存列表
|
||||||
|
* @methods setCacheKeepAlive 设置要缓存的路由 names(开启 Tagsview)
|
||||||
|
* @methods addCachedView 添加要缓存的路由 names(关闭 Tagsview)
|
||||||
|
* @methods delCachedView 删除要缓存的路由 names(关闭 Tagsview)
|
||||||
|
* @methods delOthersCachedViews 右键菜单`关闭其它`,删除要缓存的路由 names(关闭 Tagsview)
|
||||||
|
* @methods delAllCachedViews 右键菜单`全部关闭`,删除要缓存的路由 names(关闭 Tagsview)
|
||||||
|
*/
|
||||||
|
export const useKeepALiveNames = defineStore('keepALiveNames', {
|
||||||
|
state: (): KeepAliveNamesState => ({
|
||||||
|
keepAliveNames: [],
|
||||||
|
cachedViews: [],
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
async setCacheKeepAlive(data: Array<string>) {
|
||||||
|
this.keepAliveNames = data;
|
||||||
|
},
|
||||||
|
async addCachedView(view: any) {
|
||||||
|
if (this.cachedViews.includes(view.name)) return;
|
||||||
|
if (view.meta.isKeepAlive) this.cachedViews.push(view.name);
|
||||||
|
},
|
||||||
|
async delCachedView(view: any) {
|
||||||
|
const index = this.cachedViews.indexOf(view.name);
|
||||||
|
index > -1 && this.cachedViews.splice(index, 1);
|
||||||
|
},
|
||||||
|
async delOthersCachedViews(view: any) {
|
||||||
|
if (view.meta.isKeepAlive) this.cachedViews = [view.name];
|
||||||
|
else this.cachedViews = [];
|
||||||
|
},
|
||||||
|
async delAllCachedViews() {
|
||||||
|
this.cachedViews = [];
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
17
web/src/stores/requestOldRoutes.ts
Normal file
17
web/src/stores/requestOldRoutes.ts
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import { RequestOldRoutesState } from './interface';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端返回原始路由(未处理时)
|
||||||
|
* @methods setCacheKeepAlive 设置接口原始路由数据
|
||||||
|
*/
|
||||||
|
export const useRequestOldRoutes = defineStore('requestOldRoutes', {
|
||||||
|
state: (): RequestOldRoutesState => ({
|
||||||
|
requestOldRoutes: [],
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
async setRequestOldRoutes(routes: Array<string>) {
|
||||||
|
this.requestOldRoutes = routes;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
27
web/src/stores/routesList.ts
Normal file
27
web/src/stores/routesList.ts
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import { RoutesListState } from './interface';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 路由列表
|
||||||
|
* @methods setRoutesList 设置路由数据
|
||||||
|
* @methods setColumnsMenuHover 设置分栏布局菜单鼠标移入 boolean
|
||||||
|
* @methods setColumnsNavHover 设置分栏布局最左侧导航鼠标移入 boolean
|
||||||
|
*/
|
||||||
|
export const useRoutesList = defineStore('routesList', {
|
||||||
|
state: (): RoutesListState => ({
|
||||||
|
routesList: [],
|
||||||
|
isColumnsMenuHover: false,
|
||||||
|
isColumnsNavHover: false,
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
async setRoutesList(data: Array<string>) {
|
||||||
|
this.routesList = data;
|
||||||
|
},
|
||||||
|
async setColumnsMenuHover(bool: Boolean) {
|
||||||
|
this.isColumnsMenuHover = bool;
|
||||||
|
},
|
||||||
|
async setColumnsNavHover(bool: Boolean) {
|
||||||
|
this.isColumnsNavHover = bool;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
24
web/src/stores/tagsViewRoutes.ts
Normal file
24
web/src/stores/tagsViewRoutes.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import { TagsViewRoutesState } from './interface';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TagsView 路由列表
|
||||||
|
* @methods setTagsViewRoutes 设置 TagsView 路由列表
|
||||||
|
* @methods setCurrenFullscreen 设置开启/关闭全屏时的 boolean 状态
|
||||||
|
*/
|
||||||
|
export const useTagsViewRoutes = defineStore('tagsViewRoutes', {
|
||||||
|
state: (): TagsViewRoutesState => ({
|
||||||
|
tagsViewRoutes: [],
|
||||||
|
isTagsViewCurrenFull: false,
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
async setTagsViewRoutes(data: Array<string>) {
|
||||||
|
this.tagsViewRoutes = data;
|
||||||
|
},
|
||||||
|
setCurrenFullscreen(bool: Boolean) {
|
||||||
|
Session.set('isTagsViewCurrenFull', bool);
|
||||||
|
this.isTagsViewCurrenFull = bool;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
146
web/src/stores/themeConfig.ts
Normal file
146
web/src/stores/themeConfig.ts
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import { ThemeConfigStates, ThemeConfigState } from './interface';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 布局配置
|
||||||
|
* 修复:https://gitee.com/lyt-top/vue-next-admin/issues/I567R1,感谢@lanbao123
|
||||||
|
* 2020.05.28 by lyt 优化。开发时配置不生效问题
|
||||||
|
* 修改配置时:
|
||||||
|
* 1、需要每次都清理 `window.localStorage` 浏览器永久缓存
|
||||||
|
* 2、或者点击布局配置最底部 `一键恢复默认` 按钮即可看到效果
|
||||||
|
*/
|
||||||
|
export const useThemeConfig = defineStore('themeConfig', {
|
||||||
|
state: (): ThemeConfigStates => ({
|
||||||
|
themeConfig: {
|
||||||
|
// 是否开启布局配置抽屉
|
||||||
|
isDrawer: false,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 全局主题
|
||||||
|
*/
|
||||||
|
// 默认 primary 主题颜色
|
||||||
|
primary: '#409eff',
|
||||||
|
// 是否开启深色模式
|
||||||
|
isIsDark: false,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 菜单 / 顶栏
|
||||||
|
* 注意:v1.0.17 版本去除设置布局切换,重置主题样式(initSetLayoutChange),
|
||||||
|
* 切换布局需手动设置样式,设置的样式自动同步各布局,
|
||||||
|
* 代码位置:/@/layout/navBars/breadcrumb/setings.vue
|
||||||
|
*/
|
||||||
|
// 默认顶栏导航背景颜色
|
||||||
|
topBar: '#ffffff',
|
||||||
|
// 默认顶栏导航字体颜色
|
||||||
|
topBarColor: '#606266',
|
||||||
|
// 是否开启顶栏背景颜色渐变
|
||||||
|
isTopBarColorGradual: false,
|
||||||
|
// 默认菜单导航背景颜色
|
||||||
|
menuBar: '#545c64',
|
||||||
|
// 默认菜单导航字体颜色
|
||||||
|
menuBarColor: '#eaeaea',
|
||||||
|
// 是否开启菜单背景颜色渐变
|
||||||
|
isMenuBarColorGradual: false,
|
||||||
|
// 默认分栏菜单背景颜色
|
||||||
|
columnsMenuBar: '#545c64',
|
||||||
|
// 默认分栏菜单字体颜色
|
||||||
|
columnsMenuBarColor: '#e6e6e6',
|
||||||
|
// 是否开启分栏菜单背景颜色渐变
|
||||||
|
isColumnsMenuBarColorGradual: false,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 界面设置
|
||||||
|
*/
|
||||||
|
// 是否开启菜单水平折叠效果
|
||||||
|
isCollapse: false,
|
||||||
|
// 是否开启菜单手风琴效果
|
||||||
|
isUniqueOpened: false,
|
||||||
|
// 是否开启固定 Header
|
||||||
|
isFixedHeader: false,
|
||||||
|
// 初始化变量,用于更新菜单 el-scrollbar 的高度,请勿删除
|
||||||
|
isFixedHeaderChange: false,
|
||||||
|
// 是否开启经典布局分割菜单(仅经典布局生效)
|
||||||
|
isClassicSplitMenu: false,
|
||||||
|
// 是否开启自动锁屏
|
||||||
|
isLockScreen: false,
|
||||||
|
// 开启自动锁屏倒计时(s/秒)
|
||||||
|
lockScreenTime: 30,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 界面显示
|
||||||
|
*/
|
||||||
|
// 是否开启侧边栏 Logo
|
||||||
|
isShowLogo: false,
|
||||||
|
// 初始化变量,用于 el-scrollbar 的高度更新,请勿删除
|
||||||
|
isShowLogoChange: false,
|
||||||
|
// 是否开启 Breadcrumb,强制经典、横向布局不显示
|
||||||
|
isBreadcrumb: true,
|
||||||
|
// 是否开启 Tagsview
|
||||||
|
isTagsview: true,
|
||||||
|
// 是否开启 Breadcrumb 图标
|
||||||
|
isBreadcrumbIcon: false,
|
||||||
|
// 是否开启 Tagsview 图标
|
||||||
|
isTagsviewIcon: false,
|
||||||
|
// 是否开启 TagsView 缓存
|
||||||
|
isCacheTagsView: false,
|
||||||
|
// 是否开启 TagsView 拖拽
|
||||||
|
isSortableTagsView: true,
|
||||||
|
// 是否开启 TagsView 共用
|
||||||
|
isShareTagsView: false,
|
||||||
|
// 是否开启 Footer 底部版权信息
|
||||||
|
isFooter: false,
|
||||||
|
// 是否开启灰色模式
|
||||||
|
isGrayscale: false,
|
||||||
|
// 是否开启色弱模式
|
||||||
|
isInvert: false,
|
||||||
|
// 是否开启水印
|
||||||
|
isWartermark: false,
|
||||||
|
// 水印文案
|
||||||
|
wartermarkText: 'small@小柒',
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 其它设置
|
||||||
|
*/
|
||||||
|
// Tagsview 风格:可选值"<tags-style-one|tags-style-four|tags-style-five>",默认 tags-style-five
|
||||||
|
// 定义的值与 `/src/layout/navBars/tagsView/tagsView.vue` 中的 class 同名
|
||||||
|
tagsStyle: 'tags-style-five',
|
||||||
|
// 主页面切换动画:可选值"<slide-right|slide-left|opacitys>",默认 slide-right
|
||||||
|
animation: 'slide-right',
|
||||||
|
// 分栏高亮风格:可选值"<columns-round|columns-card>",默认 columns-round
|
||||||
|
columnsAsideStyle: 'columns-round',
|
||||||
|
// 分栏布局风格:可选值"<columns-horizontal|columns-vertical>",默认 columns-horizontal
|
||||||
|
columnsAsideLayout: 'columns-vertical',
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 布局切换
|
||||||
|
* 注意:为了演示,切换布局时,颜色会被还原成默认,代码位置:/@/layout/navBars/breadcrumb/setings.vue
|
||||||
|
* 中的 `initSetLayoutChange(设置布局切换,重置主题样式)` 方法
|
||||||
|
*/
|
||||||
|
// 布局切换:可选值"<defaults|classic|transverse|columns>",默认 defaults
|
||||||
|
layout: 'defaults',
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后端控制路由
|
||||||
|
*/
|
||||||
|
// 是否开启后端控制路由
|
||||||
|
isRequestRoutes: false,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 全局网站标题 / 副标题
|
||||||
|
*/
|
||||||
|
// 网站主标题(菜单导航、浏览器当前网页标题)
|
||||||
|
globalTitle: 'vue-next-admin',
|
||||||
|
// 网站副标题(登录页顶部文字)
|
||||||
|
globalViceTitle: 'vueNextAdmin',
|
||||||
|
// 默认初始语言,可选值"<zh-cn|en|zh-tw>",默认 zh-cn
|
||||||
|
globalI18n: 'zh-cn',
|
||||||
|
// 默认全局组件大小,可选值"<large|'default'|small>",默认 'large'
|
||||||
|
globalComponentSize: 'large',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
setThemeConfig(data: ThemeConfigState) {
|
||||||
|
this.themeConfig = data;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
72
web/src/stores/userInfo.ts
Normal file
72
web/src/stores/userInfo.ts
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
import { defineStore } from 'pinia';
|
||||||
|
import Cookies from 'js-cookie';
|
||||||
|
import { UserInfosStates } from './interface';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 用户信息
|
||||||
|
* @methods setUserInfos 设置用户信息
|
||||||
|
*/
|
||||||
|
export const useUserInfo = defineStore('userInfo', {
|
||||||
|
state: (): UserInfosStates => ({
|
||||||
|
userInfos: {
|
||||||
|
userName: '',
|
||||||
|
photo: '',
|
||||||
|
time: 0,
|
||||||
|
roles: [],
|
||||||
|
authBtnList: [],
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
actions: {
|
||||||
|
async setUserInfos() {
|
||||||
|
// 存储用户信息到浏览器缓存
|
||||||
|
if (Session.get('userInfo')) {
|
||||||
|
this.userInfos = Session.get('userInfo');
|
||||||
|
} else {
|
||||||
|
const userInfos: any = await this.getApiUserInfo();
|
||||||
|
this.userInfos = userInfos;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 模拟接口数据
|
||||||
|
// https://gitee.com/lyt-top/vue-next-admin/issues/I5F1HP
|
||||||
|
async getApiUserInfo() {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
// 模拟数据,请求接口时,记得删除多余代码及对应依赖的引入
|
||||||
|
const userName = Cookies.get('userName');
|
||||||
|
// 模拟数据
|
||||||
|
let defaultRoles: Array<string> = [];
|
||||||
|
let defaultAuthBtnList: Array<string> = [];
|
||||||
|
// admin 页面权限标识,对应路由 meta.roles,用于控制路由的显示/隐藏
|
||||||
|
let adminRoles: Array<string> = ['admin'];
|
||||||
|
// admin 按钮权限标识
|
||||||
|
let adminAuthBtnList: Array<string> = ['btn.add', 'btn.del', 'btn.edit', 'btn.link'];
|
||||||
|
// test 页面权限标识,对应路由 meta.roles,用于控制路由的显示/隐藏
|
||||||
|
let testRoles: Array<string> = ['common'];
|
||||||
|
// test 按钮权限标识
|
||||||
|
let testAuthBtnList: Array<string> = ['btn.add', 'btn.link'];
|
||||||
|
// 不同用户模拟不同的用户权限
|
||||||
|
if (userName === 'admin') {
|
||||||
|
defaultRoles = adminRoles;
|
||||||
|
defaultAuthBtnList = adminAuthBtnList;
|
||||||
|
} else {
|
||||||
|
defaultRoles = testRoles;
|
||||||
|
defaultAuthBtnList = testAuthBtnList;
|
||||||
|
}
|
||||||
|
// 用户信息模拟数据
|
||||||
|
const userInfos = {
|
||||||
|
userName: userName,
|
||||||
|
photo:
|
||||||
|
userName === 'admin'
|
||||||
|
? 'https://img2.baidu.com/it/u=1978192862,2048448374&fm=253&fmt=auto&app=138&f=JPEG?w=504&h=500'
|
||||||
|
: 'https://img2.baidu.com/it/u=2370931438,70387529&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=500',
|
||||||
|
time: new Date().getTime(),
|
||||||
|
roles: defaultRoles,
|
||||||
|
authBtnList: defaultAuthBtnList,
|
||||||
|
};
|
||||||
|
resolve(userInfos);
|
||||||
|
}, 3000);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
281
web/src/theme/app.scss
Normal file
281
web/src/theme/app.scss
Normal file
@@ -0,0 +1,281 @@
|
|||||||
|
/* 初始化样式
|
||||||
|
------------------------------- */
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
outline: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--next-color-white: #ffffff;
|
||||||
|
--next-bg-main-color: #f8f8f8;
|
||||||
|
--next-bg-color: #f5f5ff;
|
||||||
|
--next-border-color-light: #f1f2f3;
|
||||||
|
--next-color-primary-lighter: #ecf5ff;
|
||||||
|
--next-color-success-lighter: #f0f9eb;
|
||||||
|
--next-color-warning-lighter: #fdf6ec;
|
||||||
|
--next-color-danger-lighter: #fef0f0;
|
||||||
|
--next-color-dark-hover: #0000001a;
|
||||||
|
--next-color-menu-hover: rgba(0, 0, 0, 0.2);
|
||||||
|
--next-color-user-hover: rgba(0, 0, 0, 0.04);
|
||||||
|
--next-color-seting-main: #e9eef3;
|
||||||
|
--next-color-seting-aside: #d3dce6;
|
||||||
|
--next-color-seting-header: #b3c0d1;
|
||||||
|
}
|
||||||
|
|
||||||
|
html,
|
||||||
|
body,
|
||||||
|
#app {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
font-family: Helvetica Neue, Helvetica, PingFang SC, Hiragino Sans GB, Microsoft YaHei, SimSun, sans-serif;
|
||||||
|
font-weight: 400;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
-webkit-tap-highlight-color: transparent;
|
||||||
|
background-color: var(--next-bg-main-color);
|
||||||
|
font-size: 14px;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 主布局样式
|
||||||
|
------------------------------- */
|
||||||
|
.layout-container {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
.layout-aside {
|
||||||
|
background: var(--next-bg-menuBar);
|
||||||
|
box-shadow: 2px 0 6px rgb(0 21 41 / 1%);
|
||||||
|
height: inherit;
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
overflow-x: hidden !important;
|
||||||
|
.el-scrollbar__view {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-header {
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
.layout-main {
|
||||||
|
padding: 0 !important;
|
||||||
|
overflow: hidden;
|
||||||
|
width: 100%;
|
||||||
|
background-color: var(--next-bg-main-color);
|
||||||
|
}
|
||||||
|
.el-scrollbar {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
// 此字段多次用到,建议不删除,如需修改,请重写覆盖样式
|
||||||
|
.layout-view-bg-white {
|
||||||
|
background: var(--el-color-white);
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 1px solid var(--el-border-color-light, #ebeef5);
|
||||||
|
}
|
||||||
|
.layout-el-aside-br-color {
|
||||||
|
border-right: 1px solid var(--el-border-color-light, #ebeef5);
|
||||||
|
}
|
||||||
|
// pc端左侧导航样式
|
||||||
|
.layout-aside-pc-220 {
|
||||||
|
width: 220px !important;
|
||||||
|
transition: width 0.3s ease;
|
||||||
|
}
|
||||||
|
.layout-aside-pc-64 {
|
||||||
|
width: 64px !important;
|
||||||
|
transition: width 0.3s ease;
|
||||||
|
}
|
||||||
|
.layout-aside-pc-1 {
|
||||||
|
width: 1px !important;
|
||||||
|
transition: width 0.3s ease;
|
||||||
|
}
|
||||||
|
// 手机端左侧导航样式
|
||||||
|
.layout-aside-mobile {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: -220px;
|
||||||
|
width: 220px;
|
||||||
|
z-index: 9999999;
|
||||||
|
}
|
||||||
|
.layout-aside-mobile-close {
|
||||||
|
left: -220px;
|
||||||
|
transition: all 0.3s cubic-bezier(0.39, 0.58, 0.57, 1);
|
||||||
|
}
|
||||||
|
.layout-aside-mobile-open {
|
||||||
|
left: 0;
|
||||||
|
transition: all 0.3s cubic-bezier(0.22, 0.61, 0.36, 1);
|
||||||
|
}
|
||||||
|
.layout-aside-mobile-mode {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
z-index: 9999998;
|
||||||
|
animation: error-img 0.3s;
|
||||||
|
}
|
||||||
|
.layout-scrollbar {
|
||||||
|
@extend .el-scrollbar;
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
.layout-mian-height-50 {
|
||||||
|
height: calc(100vh - 50px);
|
||||||
|
}
|
||||||
|
.layout-columns-warp {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.layout-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* element plus 全局样式
|
||||||
|
------------------------------- */
|
||||||
|
.layout-breadcrumb-seting {
|
||||||
|
.el-divider {
|
||||||
|
background-color: rgb(230, 230, 230);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* nprogress 进度条跟随主题颜色
|
||||||
|
------------------------------- */
|
||||||
|
#nprogress {
|
||||||
|
.bar {
|
||||||
|
background: var(--el-color-primary) !important;
|
||||||
|
z-index: 9999999 !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* flex 弹性布局
|
||||||
|
------------------------------- */
|
||||||
|
.flex {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
.flex-auto {
|
||||||
|
flex: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.flex-center {
|
||||||
|
@extend .flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.flex-margin {
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
.flex-warp {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-content: flex-start;
|
||||||
|
margin: 0 -5px;
|
||||||
|
.flex-warp-item {
|
||||||
|
padding: 5px;
|
||||||
|
.flex-warp-item-box {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* cursor 鼠标形状
|
||||||
|
------------------------------- */
|
||||||
|
// 默认
|
||||||
|
.cursor-default {
|
||||||
|
cursor: default !important;
|
||||||
|
}
|
||||||
|
// 帮助
|
||||||
|
.cursor-help {
|
||||||
|
cursor: help !important;
|
||||||
|
}
|
||||||
|
// 手指
|
||||||
|
.cursor-pointer {
|
||||||
|
cursor: pointer !important;
|
||||||
|
}
|
||||||
|
// 移动
|
||||||
|
.cursor-move {
|
||||||
|
cursor: move !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 宽高 100%
|
||||||
|
------------------------------- */
|
||||||
|
.w100 {
|
||||||
|
width: 100% !important;
|
||||||
|
}
|
||||||
|
.h100 {
|
||||||
|
height: 100% !important;
|
||||||
|
}
|
||||||
|
.vh100 {
|
||||||
|
height: 100vh !important;
|
||||||
|
}
|
||||||
|
.max100vh {
|
||||||
|
max-height: 100vh !important;
|
||||||
|
}
|
||||||
|
.min100vh {
|
||||||
|
min-height: 100vh !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 颜色值
|
||||||
|
------------------------------- */
|
||||||
|
.color-primary {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
.color-success {
|
||||||
|
color: var(--el-color-success);
|
||||||
|
}
|
||||||
|
.color-warning {
|
||||||
|
color: var(--el-color-warning);
|
||||||
|
}
|
||||||
|
.color-danger {
|
||||||
|
color: var(--el-color-danger);
|
||||||
|
}
|
||||||
|
.color-info {
|
||||||
|
color: var(--el-color-info);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 字体大小全局样式
|
||||||
|
------------------------------- */
|
||||||
|
@for $i from 10 through 32 {
|
||||||
|
.font#{$i} {
|
||||||
|
font-size: #{$i}px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 外边距、内边距全局样式
|
||||||
|
------------------------------- */
|
||||||
|
@for $i from 1 through 35 {
|
||||||
|
.mt#{$i} {
|
||||||
|
margin-top: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.mr#{$i} {
|
||||||
|
margin-right: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.mb#{$i} {
|
||||||
|
margin-bottom: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.ml#{$i} {
|
||||||
|
margin-left: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.pt#{$i} {
|
||||||
|
padding-top: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.pr#{$i} {
|
||||||
|
padding-right: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.pb#{$i} {
|
||||||
|
padding-bottom: #{$i}px !important;
|
||||||
|
}
|
||||||
|
.pl#{$i} {
|
||||||
|
padding-left: #{$i}px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
94
web/src/theme/common/transition.scss
Normal file
94
web/src/theme/common/transition.scss
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
/* 页面切换动画
|
||||||
|
------------------------------- */
|
||||||
|
.slide-right-enter-active,
|
||||||
|
.slide-right-leave-active,
|
||||||
|
.slide-left-enter-active,
|
||||||
|
.slide-left-leave-active {
|
||||||
|
will-change: transform;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
// slide-right
|
||||||
|
.slide-right-enter-from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(-20px);
|
||||||
|
}
|
||||||
|
.slide-right-leave-to {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(20px);
|
||||||
|
}
|
||||||
|
// slide-left
|
||||||
|
.slide-left-enter-from {
|
||||||
|
@extend .slide-right-leave-to;
|
||||||
|
}
|
||||||
|
.slide-left-leave-to {
|
||||||
|
@extend .slide-right-enter-from;
|
||||||
|
}
|
||||||
|
// opacitys
|
||||||
|
.opacitys-enter-active,
|
||||||
|
.opacitys-leave-active {
|
||||||
|
will-change: transform;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
.opacitys-enter-from,
|
||||||
|
.opacitys-leave-to {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Breadcrumb 面包屑过渡动画
|
||||||
|
------------------------------- */
|
||||||
|
.breadcrumb-enter-active,
|
||||||
|
.breadcrumb-leave-active {
|
||||||
|
transition: all 0.5s ease;
|
||||||
|
}
|
||||||
|
.breadcrumb-enter-from,
|
||||||
|
.breadcrumb-leave-active {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(20px);
|
||||||
|
}
|
||||||
|
.breadcrumb-leave-active {
|
||||||
|
position: absolute;
|
||||||
|
z-index: -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* logo 过渡动画
|
||||||
|
------------------------------- */
|
||||||
|
@keyframes logoAnimation {
|
||||||
|
0% {
|
||||||
|
transform: scale(0);
|
||||||
|
}
|
||||||
|
80% {
|
||||||
|
transform: scale(1.2);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: scale(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 404、401 过渡动画
|
||||||
|
------------------------------- */
|
||||||
|
@keyframes error-num {
|
||||||
|
0% {
|
||||||
|
transform: translateY(60px);
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: translateY(0);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@keyframes error-img {
|
||||||
|
0% {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@keyframes error-img-two {
|
||||||
|
0% {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
236
web/src/theme/dark.scss
Normal file
236
web/src/theme/dark.scss
Normal file
@@ -0,0 +1,236 @@
|
|||||||
|
/* 深色模式样式
|
||||||
|
------------------------------- */
|
||||||
|
[data-theme='dark'] {
|
||||||
|
// 变量(自定义时,只需修改这里的值)
|
||||||
|
--next-bg-main: #1f1f1f;
|
||||||
|
--next-color-white: #ffffff;
|
||||||
|
--next-color-disabled: #191919;
|
||||||
|
--next-color-bar: #dadada;
|
||||||
|
--next-color-primary: #303030;
|
||||||
|
--next-border-color: #424242;
|
||||||
|
--next-border-black: #333333;
|
||||||
|
--next-border-columns: #2a2a2a;
|
||||||
|
--next-color-seting: #505050;
|
||||||
|
--next-text-color-regular: #9b9da1;
|
||||||
|
--next-text-color-placeholder: #7a7a7a;
|
||||||
|
--next-color-hover: #3c3c3c;
|
||||||
|
--next-color-hover-rgba: rgba(0, 0, 0, 0.3);
|
||||||
|
|
||||||
|
// root
|
||||||
|
--next-bg-main-color: var(--next-bg-main) !important;
|
||||||
|
--next-bg-topBar: var(--next-color-disabled) !important;
|
||||||
|
--next-bg-topBarColor: var(--next-color-bar) !important;
|
||||||
|
--next-bg-menuBar: var(--next-color-disabled) !important;
|
||||||
|
--next-bg-menuBarColor: var(--next-color-bar) !important;
|
||||||
|
--next-bg-columnsMenuBar: var(--next-color-disabled) !important;
|
||||||
|
--next-bg-columnsMenuBarColor: var(--next-color-bar) !important;
|
||||||
|
--next-border-color-light: var(--next-border-black) !important;
|
||||||
|
--next-color-primary-lighter: var(--next-color-primary) !important;
|
||||||
|
--next-color-success-lighter: var(--next-color-primary) !important;
|
||||||
|
--next-color-warning-lighter: var(--next-color-primary) !important;
|
||||||
|
--next-color-danger-lighter: var(--next-color-primary) !important;
|
||||||
|
--next-bg-color: var(--next-color-primary) !important;
|
||||||
|
--next-color-dark-hover: var(--next-color-hover) !important;
|
||||||
|
--next-color-menu-hover: var(--next-color-hover-rgba) !important;
|
||||||
|
--next-color-user-hover: var(--next-color-hover-rgba) !important;
|
||||||
|
--next-color-seting-main: var(--next-color-seting) !important;
|
||||||
|
--next-color-seting-aside: var(--next-color-hover) !important;
|
||||||
|
--next-color-seting-header: var(--next-color-primary) !important;
|
||||||
|
|
||||||
|
// element plus
|
||||||
|
--el-color-white: var(--next-color-disabled) !important;
|
||||||
|
--el-text-color-primary: var(--next-color-bar) !important;
|
||||||
|
--el-border-color: var(--next-border-black) !important;
|
||||||
|
--el-border-color-light: var(--next-border-black) !important;
|
||||||
|
--el-border-color-lighter: var(--next-border-black) !important;
|
||||||
|
--el-border-color-extra-light: var(--el-color-primary-light-8) !important;
|
||||||
|
--el-text-color-regular: var(--next-text-color-regular) !important;
|
||||||
|
--el-bg-color: var(--next-color-disabled) !important;
|
||||||
|
--el-color-primary-light-9: var(--next-color-hover) !important;
|
||||||
|
--el-text-color-disabled: var(--next-text-color-placeholder) !important;
|
||||||
|
--el-text-color-disabled-base: var(--el-color-primary) !important;
|
||||||
|
--el-text-color-placeholder: var(--next-text-color-placeholder) !important;
|
||||||
|
--el-disabled-bg-color: var(--next-color-disabled) !important;
|
||||||
|
--el-fill-base: var(--next-color-white) !important;
|
||||||
|
--el-fill-colo: var(--next-color-hover-rgba) !important;
|
||||||
|
--el-fill-color: var(--next-color-hover-rgba) !important;
|
||||||
|
--el-fill-color-blank: var(--next-color-disabled) !important;
|
||||||
|
--el-fill-color-light: var(--next-color-hover-rgba) !important;
|
||||||
|
--el-bg-color-overlay: var(--el-color-primary-light-9) !important;
|
||||||
|
--el-mask-color: rgb(42 42 42 / 80%);
|
||||||
|
|
||||||
|
// button
|
||||||
|
.el-button {
|
||||||
|
&:hover {
|
||||||
|
border-color: var(--next-border-color) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-button--primary,
|
||||||
|
.el-button--info,
|
||||||
|
.el-button--danger,
|
||||||
|
.el-button--success,
|
||||||
|
.el-button--warning {
|
||||||
|
--el-button-text-color: var(--next-color-white) !important;
|
||||||
|
--el-button-hover-text-color: var(--next-color-white) !important;
|
||||||
|
--el-button-disabled-text-color: var(--next-color-white) !important;
|
||||||
|
&:hover {
|
||||||
|
border-color: var(--el-button-hover-border-color, var(--el-button-hover-bg-color)) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// drawer
|
||||||
|
.el-divider__text {
|
||||||
|
background-color: var(--el-color-white) !important;
|
||||||
|
}
|
||||||
|
.el-drawer {
|
||||||
|
border-left: 1px solid var(--next-border-color-light) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// tabs
|
||||||
|
.el-tabs--border-card {
|
||||||
|
background-color: var(--el-color-white) !important;
|
||||||
|
}
|
||||||
|
.el-tabs--border-card > .el-tabs__header .el-tabs__item.is-active {
|
||||||
|
background: var(--next-color-primary-lighter);
|
||||||
|
}
|
||||||
|
|
||||||
|
// alert / notice-bar
|
||||||
|
.home-card-item {
|
||||||
|
border: 1px solid var(--next-border-color-light) !important;
|
||||||
|
}
|
||||||
|
.el-alert,
|
||||||
|
.notice-bar {
|
||||||
|
border: 1px solid var(--next-border-color) !important;
|
||||||
|
background-color: var(--next-color-disabled) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// menu
|
||||||
|
.layout-aside {
|
||||||
|
border-right: 1px solid var(--next-border-color-light) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// colorPicker
|
||||||
|
.el-color-picker__mask {
|
||||||
|
background: unset !important;
|
||||||
|
}
|
||||||
|
.el-color-picker__trigger {
|
||||||
|
border: 1px solid var(--next-border-color-light) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// popper / dropdown
|
||||||
|
.el-popper {
|
||||||
|
border: 1px solid var(--next-border-color) !important;
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
.el-popper__arrow:before {
|
||||||
|
background: var(--el-color-white) !important;
|
||||||
|
border: 1px solid var(--next-border-color);
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-popper,
|
||||||
|
.el-dropdown-menu {
|
||||||
|
background: var(--el-color-white) !important;
|
||||||
|
}
|
||||||
|
.el-dropdown-menu__item:hover:not(.is-disabled) {
|
||||||
|
background: var(--el-bg-color) !important;
|
||||||
|
}
|
||||||
|
.el-dropdown-menu__item.is-disabled {
|
||||||
|
font-weight: 700 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// input
|
||||||
|
.el-input-group__append,
|
||||||
|
.el-input-group__prepend {
|
||||||
|
border: var(--el-input-border) !important;
|
||||||
|
border-right: none !important;
|
||||||
|
background: var(--next-color-disabled) !important;
|
||||||
|
border-left: 0 !important;
|
||||||
|
}
|
||||||
|
.el-input-number__decrease,
|
||||||
|
.el-input-number__increase {
|
||||||
|
background: var(--next-color-disabled) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// tag
|
||||||
|
.el-select .el-select__tags .el-tag {
|
||||||
|
background-color: var(--next-bg-color) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// pagination
|
||||||
|
.el-pagination.is-background .el-pager li:not(.disabled).active {
|
||||||
|
color: var(--next-color-white) !important;
|
||||||
|
}
|
||||||
|
.el-pagination.is-background .btn-next,
|
||||||
|
.el-pagination.is-background .btn-prev,
|
||||||
|
.el-pagination.is-background .el-pager li {
|
||||||
|
background-color: var(--next-bg-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
// radio
|
||||||
|
.el-radio-button:not(.is-active) .el-radio-button__inner {
|
||||||
|
border: 1px solid var(--next-border-color-light) !important;
|
||||||
|
border-left: 0 !important;
|
||||||
|
}
|
||||||
|
.el-radio-button.is-active .el-radio-button__inner {
|
||||||
|
color: var(--next-color-white) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// countup
|
||||||
|
.countup-card-item-flex {
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// editor
|
||||||
|
.editor-container {
|
||||||
|
.w-e-toolbar {
|
||||||
|
background: var(--el-color-white) !important;
|
||||||
|
border: 1px solid var(--next-border-color-light) !important;
|
||||||
|
.w-e-menu:hover {
|
||||||
|
background: var(--next-color-user-hover) !important;
|
||||||
|
i {
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.w-e-text-container {
|
||||||
|
border: 1px solid var(--next-border-color-light) !important;
|
||||||
|
border-top: none !important;
|
||||||
|
.w-e-text {
|
||||||
|
background: var(--el-color-white) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// date-picker
|
||||||
|
.el-picker-panel {
|
||||||
|
background: var(--el-color-white) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
// dialog
|
||||||
|
.el-dialog {
|
||||||
|
border: 1px solid var(--el-border-color-lighter);
|
||||||
|
.el-dialog__header {
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// columns
|
||||||
|
.layout-columns-aside ul .layout-columns-active {
|
||||||
|
color: var(--next-color-white) !important;
|
||||||
|
}
|
||||||
|
.layout-columns-aside {
|
||||||
|
border-right: 1px solid var(--next-border-columns);
|
||||||
|
}
|
||||||
|
|
||||||
|
// tagsView
|
||||||
|
.tags-style-one {
|
||||||
|
.is-active {
|
||||||
|
color: var(--el-text-color-primary) !important;
|
||||||
|
}
|
||||||
|
.layout-navbars-tagsview-ul-li:hover {
|
||||||
|
border-color: var(--el-border-color-lighter) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
293
web/src/theme/element.scss
Normal file
293
web/src/theme/element.scss
Normal file
@@ -0,0 +1,293 @@
|
|||||||
|
@import 'mixins/index.scss';
|
||||||
|
|
||||||
|
/* Button 按钮
|
||||||
|
------------------------------- */
|
||||||
|
// 第三方字体图标大小
|
||||||
|
.el-button i.el-icon,
|
||||||
|
.el-button i.iconfont,
|
||||||
|
.el-button i.fa,
|
||||||
|
.el-button--default i.iconfont,
|
||||||
|
.el-button--default i.fa {
|
||||||
|
font-size: 14px !important;
|
||||||
|
//margin-right: 5px;
|
||||||
|
}
|
||||||
|
.el-button--small i.iconfont,
|
||||||
|
.el-button--small i.fa {
|
||||||
|
font-size: 12px !important;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Input 输入框、InputNumber 计数器
|
||||||
|
------------------------------- */
|
||||||
|
.el-input {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
// 菜单搜索
|
||||||
|
.el-autocomplete-suggestion__wrap {
|
||||||
|
max-height: 280px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Form 表单
|
||||||
|
------------------------------- */
|
||||||
|
.el-form {
|
||||||
|
// 用于修改弹窗时表单内容间隔太大问题,如系统设置的新增菜单弹窗里的表单内容
|
||||||
|
.el-form-item:last-of-type {
|
||||||
|
margin-bottom: 0 !important;
|
||||||
|
}
|
||||||
|
// 修复行内表单最后一个 el-form-item 位置下移问题
|
||||||
|
&.el-form--inline {
|
||||||
|
.el-form-item--large.el-form-item:last-of-type {
|
||||||
|
margin-bottom: 22px !important;
|
||||||
|
}
|
||||||
|
.el-form-item--default.el-form-item:last-of-type,
|
||||||
|
.el-form-item--small.el-form-item:last-of-type {
|
||||||
|
margin-bottom: 18px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Alert 警告
|
||||||
|
------------------------------- */
|
||||||
|
.el-alert {
|
||||||
|
border: 1px solid;
|
||||||
|
}
|
||||||
|
.el-alert__title {
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Message 消息提示
|
||||||
|
------------------------------- */
|
||||||
|
.el-message {
|
||||||
|
min-width: unset !important;
|
||||||
|
padding: 15px !important;
|
||||||
|
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.02);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* NavMenu 导航菜单
|
||||||
|
------------------------------- */
|
||||||
|
// 鼠标 hover 时颜色
|
||||||
|
.el-menu-hover-bg-color {
|
||||||
|
background-color: var(--next-color-menu-hover) !important;
|
||||||
|
}
|
||||||
|
// 默认样式修改
|
||||||
|
.el-menu {
|
||||||
|
border-right: none !important;
|
||||||
|
width: 220px;
|
||||||
|
}
|
||||||
|
.el-menu-item {
|
||||||
|
height: 56px !important;
|
||||||
|
line-height: 56px !important;
|
||||||
|
}
|
||||||
|
.el-menu-item,
|
||||||
|
.el-sub-menu__title {
|
||||||
|
color: var(--next-bg-menuBarColor);
|
||||||
|
}
|
||||||
|
// 修复点击左侧菜单折叠再展开时,宽度不跟随问题
|
||||||
|
.el-menu--collapse {
|
||||||
|
width: 64px !important;
|
||||||
|
}
|
||||||
|
// 外部链接时
|
||||||
|
.el-menu-item a,
|
||||||
|
.el-menu-item a:hover,
|
||||||
|
.el-menu-item i,
|
||||||
|
.el-sub-menu__title i {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
// 第三方图标字体间距/大小设置
|
||||||
|
.el-menu-item .iconfont,
|
||||||
|
.el-sub-menu .iconfont,
|
||||||
|
.el-menu-item .fa,
|
||||||
|
.el-sub-menu .fa {
|
||||||
|
@include generalIcon;
|
||||||
|
}
|
||||||
|
// 水平菜单、横向菜单高亮 背景色,鼠标 hover 时,有子级菜单的背景色
|
||||||
|
.el-menu-item.is-active,
|
||||||
|
.el-sub-menu.is-active .el-sub-menu__title,
|
||||||
|
.el-sub-menu:not(.is-opened):hover .el-sub-menu__title {
|
||||||
|
@extend .el-menu-hover-bg-color;
|
||||||
|
}
|
||||||
|
.el-sub-menu.is-active.is-opened .el-sub-menu__title {
|
||||||
|
background-color: unset !important;
|
||||||
|
}
|
||||||
|
// 子级菜单背景颜色
|
||||||
|
// .el-menu--inline {
|
||||||
|
// background: var(--next-bg-menuBar-light-1);
|
||||||
|
// }
|
||||||
|
// 水平菜单、横向菜单折叠 a 标签
|
||||||
|
.el-popper.is-dark a {
|
||||||
|
color: var(--el-color-white) !important;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
// 水平菜单、横向菜单折叠背景色
|
||||||
|
.el-popper.is-pure.is-light {
|
||||||
|
// 水平菜单
|
||||||
|
.el-menu--vertical {
|
||||||
|
background: var(--next-bg-menuBar);
|
||||||
|
.el-sub-menu.is-active .el-sub-menu__title {
|
||||||
|
color: var(--el-menu-active-color);
|
||||||
|
}
|
||||||
|
.el-popper.is-pure.is-light {
|
||||||
|
.el-menu--vertical {
|
||||||
|
.el-sub-menu .el-sub-menu__title {
|
||||||
|
background-color: unset !important;
|
||||||
|
color: var(--next-bg-menuBarColor);
|
||||||
|
}
|
||||||
|
.el-sub-menu.is-active .el-sub-menu__title {
|
||||||
|
color: var(--el-menu-active-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 横向菜单
|
||||||
|
.el-menu--horizontal {
|
||||||
|
background: var(--next-bg-topBar);
|
||||||
|
.el-menu-item,
|
||||||
|
.el-sub-menu {
|
||||||
|
height: 50px !important;
|
||||||
|
line-height: 50px !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
.el-sub-menu__title {
|
||||||
|
height: 50px !important;
|
||||||
|
line-height: 50px !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
.el-popper.is-pure.is-light {
|
||||||
|
.el-menu--horizontal {
|
||||||
|
.el-sub-menu .el-sub-menu__title {
|
||||||
|
background-color: unset !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
.el-sub-menu.is-active .el-sub-menu__title {
|
||||||
|
color: var(--el-menu-active-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-menu-item.is-active,
|
||||||
|
.el-sub-menu.is-active .el-sub-menu__title {
|
||||||
|
color: var(--el-menu-active-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 横向菜单(经典、横向)布局
|
||||||
|
.el-menu.el-menu--horizontal {
|
||||||
|
border-bottom: none !important;
|
||||||
|
width: 100% !important;
|
||||||
|
.el-menu-item,
|
||||||
|
.el-sub-menu__title {
|
||||||
|
height: 50px !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
.el-menu-item:not(.is-active):hover,
|
||||||
|
.el-sub-menu:not(.is-active):hover .el-sub-menu__title {
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tabs 标签页
|
||||||
|
------------------------------- */
|
||||||
|
.el-tabs__nav-wrap::after {
|
||||||
|
height: 1px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Dropdown 下拉菜单
|
||||||
|
------------------------------- */
|
||||||
|
.el-dropdown-menu {
|
||||||
|
list-style: none !important; /*修复 Dropdown 下拉菜单样式问题 2022.03.04*/
|
||||||
|
}
|
||||||
|
.el-dropdown-menu .el-dropdown-menu__item {
|
||||||
|
white-space: nowrap;
|
||||||
|
&:not(.is-disabled):hover {
|
||||||
|
background-color: var(--el-dropdown-menuItem-hover-fill);
|
||||||
|
color: var(--el-dropdown-menuItem-hover-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Steps 步骤条
|
||||||
|
------------------------------- */
|
||||||
|
.el-step__icon-inner {
|
||||||
|
font-size: 30px !important;
|
||||||
|
font-weight: 400 !important;
|
||||||
|
}
|
||||||
|
.el-step__title {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Dialog 对话框
|
||||||
|
------------------------------- */
|
||||||
|
.el-overlay {
|
||||||
|
overflow: hidden;
|
||||||
|
.el-overlay-dialog {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
position: unset !important;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
.el-dialog {
|
||||||
|
margin: 0 auto !important;
|
||||||
|
position: absolute;
|
||||||
|
.el-dialog__body {
|
||||||
|
padding: 20px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-dialog__body {
|
||||||
|
max-height: calc(90vh - 111px) !important;
|
||||||
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Card 卡片
|
||||||
|
------------------------------- */
|
||||||
|
.el-card__header {
|
||||||
|
padding: 15px 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Table 表格 element plus 2.2.0 版本
|
||||||
|
------------------------------- */
|
||||||
|
.el-table {
|
||||||
|
.el-button.is-text {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* scrollbar
|
||||||
|
------------------------------- */
|
||||||
|
.el-scrollbar__bar {
|
||||||
|
z-index: 4;
|
||||||
|
}
|
||||||
|
.el-scrollbar__wrap {
|
||||||
|
max-height: 100%; /*防止页面切换时,滚动条高度不变的问题(滚动条高度非滚动条滚动高度)*/
|
||||||
|
}
|
||||||
|
.el-select-dropdown .el-scrollbar__wrap {
|
||||||
|
overflow-x: scroll !important;
|
||||||
|
}
|
||||||
|
.el-select-dropdown__wrap {
|
||||||
|
max-height: 274px !important; /*修复Select 选择器高度问题*/
|
||||||
|
}
|
||||||
|
.el-cascader-menu__wrap.el-scrollbar__wrap {
|
||||||
|
height: 204px !important; /*修复Cascader 级联选择器高度问题*/
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Drawer 抽屉
|
||||||
|
------------------------------- */
|
||||||
|
.el-drawer {
|
||||||
|
--el-drawer-padding-primary: unset !important;
|
||||||
|
.el-drawer__header {
|
||||||
|
padding: 0 15px !important;
|
||||||
|
height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 0 !important;
|
||||||
|
border-bottom: 1px solid var(--el-border-color);
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
}
|
||||||
|
.el-drawer__body {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
5
web/src/theme/fastCrud.scss
Normal file
5
web/src/theme/fastCrud.scss
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
.fs-page {
|
||||||
|
background-color: rgb(255, 255, 255);
|
||||||
|
border-radius: 10px;
|
||||||
|
height: 88vh !important;
|
||||||
|
}
|
||||||
70
web/src/theme/iconSelector.scss
Normal file
70
web/src/theme/iconSelector.scss
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
/* Popover 弹出框(图标选择器)
|
||||||
|
------------------------------- */
|
||||||
|
.icon-selector-popper {
|
||||||
|
padding: 0 !important;
|
||||||
|
.icon-selector-warp {
|
||||||
|
height: 260px;
|
||||||
|
overflow: hidden;
|
||||||
|
.icon-selector-warp-title {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
padding: 0 15px;
|
||||||
|
.icon-selector-warp-title-tab {
|
||||||
|
span {
|
||||||
|
cursor: pointer;
|
||||||
|
&:hover {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.span-active {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.icon-selector-warp-row {
|
||||||
|
height: 230px;
|
||||||
|
overflow: hidden;
|
||||||
|
border-top: 1px solid var(--el-border-color);
|
||||||
|
.el-row {
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
.el-scrollbar__bar.is-horizontal {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.icon-selector-warp-item {
|
||||||
|
display: flex;
|
||||||
|
border: 1px solid var(--el-border-color);
|
||||||
|
padding: 5px;
|
||||||
|
border-radius: 5px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
.icon-selector-warp-item-value {
|
||||||
|
i {
|
||||||
|
font-size: 20px;
|
||||||
|
color: var(--el-text-color-regular);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
background-color: var(--el-color-primary-light-9);
|
||||||
|
border: 1px solid var(--el-color-primary-light-5);
|
||||||
|
.icon-selector-warp-item-value {
|
||||||
|
i {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.icon-selector-active {
|
||||||
|
background-color: var(--el-color-primary-light-9);
|
||||||
|
border: 1px solid var(--el-color-primary-light-5);
|
||||||
|
.icon-selector-warp-item-value {
|
||||||
|
i {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
9
web/src/theme/index.scss
Normal file
9
web/src/theme/index.scss
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
@import './app.scss';
|
||||||
|
@import 'common/transition.scss';
|
||||||
|
@import './other.scss';
|
||||||
|
@import './element.scss';
|
||||||
|
@import './iconSelector.scss';
|
||||||
|
@import './media/media.scss';
|
||||||
|
@import './waves.scss';
|
||||||
|
@import './dark.scss';
|
||||||
|
@import './fastCrud.scss';
|
||||||
51
web/src/theme/loading.scss
Normal file
51
web/src/theme/loading.scss
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
.loading-next {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp {
|
||||||
|
width: 80px;
|
||||||
|
height: 80px;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item {
|
||||||
|
width: 33.333333%;
|
||||||
|
height: 33.333333%;
|
||||||
|
background: var(--el-color-primary);
|
||||||
|
float: left;
|
||||||
|
animation: loading-next-animation 1.2s infinite ease;
|
||||||
|
border-radius: 1px;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(7) {
|
||||||
|
animation-delay: 0s;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(4),
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(8) {
|
||||||
|
animation-delay: 0.1s;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(1),
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(5),
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(9) {
|
||||||
|
animation-delay: 0.2s;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(2),
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(6) {
|
||||||
|
animation-delay: 0.3s;
|
||||||
|
}
|
||||||
|
.loading-next .loading-next-box-warp .loading-next-box-item:nth-child(3) {
|
||||||
|
animation-delay: 0.4s;
|
||||||
|
}
|
||||||
|
@keyframes loading-next-animation {
|
||||||
|
0%,
|
||||||
|
70%,
|
||||||
|
100% {
|
||||||
|
transform: scale3D(1, 1, 1);
|
||||||
|
}
|
||||||
|
35% {
|
||||||
|
transform: scale3D(0, 0, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
94
web/src/theme/media/chart.scss
Normal file
94
web/src/theme/media/chart.scss
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于768px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $sm) {
|
||||||
|
.big-data-down-left {
|
||||||
|
width: 100% !important;
|
||||||
|
flex-direction: unset !important;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
.flex-warp-item {
|
||||||
|
min-height: 196.24px;
|
||||||
|
padding: 0 7.5px 15px 15px !important;
|
||||||
|
.flex-warp-item-box {
|
||||||
|
border: none !important;
|
||||||
|
border-bottom: 1px solid #ebeef5 !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.big-data-down-center {
|
||||||
|
width: 100% !important;
|
||||||
|
.big-data-down-center-one,
|
||||||
|
.big-data-down-center-two {
|
||||||
|
min-height: 196.24px;
|
||||||
|
padding-left: 15px !important;
|
||||||
|
.big-data-down-center-one-content {
|
||||||
|
border: none !important;
|
||||||
|
border-bottom: 1px solid #ebeef5 !important;
|
||||||
|
}
|
||||||
|
.flex-warp-item-box {
|
||||||
|
@extend .big-data-down-center-one-content;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.big-data-down-right {
|
||||||
|
.flex-warp-item {
|
||||||
|
.flex-warp-item-box {
|
||||||
|
border: none !important;
|
||||||
|
border-bottom: 1px solid #ebeef5 !important;
|
||||||
|
}
|
||||||
|
&:nth-of-type(2) {
|
||||||
|
padding-left: 15px !important;
|
||||||
|
}
|
||||||
|
&:last-of-type {
|
||||||
|
.flex-warp-item-box {
|
||||||
|
border: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度大于768px小于1200px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (min-width: $sm) and (max-width: $lg) {
|
||||||
|
.chart-warp-bottom {
|
||||||
|
.big-data-down-left {
|
||||||
|
width: 50% !important;
|
||||||
|
}
|
||||||
|
.big-data-down-center {
|
||||||
|
width: 50% !important;
|
||||||
|
}
|
||||||
|
.big-data-down-right {
|
||||||
|
.flex-warp-item {
|
||||||
|
width: 50% !important;
|
||||||
|
&:nth-of-type(2) {
|
||||||
|
padding-left: 7.5px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度小于1200px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $lg) {
|
||||||
|
.chart-warp-top {
|
||||||
|
.up-left {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.chart-warp-bottom {
|
||||||
|
overflow-y: auto !important;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
.big-data-down-right {
|
||||||
|
width: 100% !important;
|
||||||
|
flex-direction: unset !important;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
.flex-warp-item {
|
||||||
|
min-height: 196.24px;
|
||||||
|
padding: 0 7.5px 15px 15px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
10
web/src/theme/media/cityLinkage.scss
Normal file
10
web/src/theme/media/cityLinkage.scss
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于576px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $xs) {
|
||||||
|
.el-cascader__dropdown.el-popper {
|
||||||
|
overflow: auto;
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
25
web/src/theme/media/date.scss
Normal file
25
web/src/theme/media/date.scss
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于768px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $sm) {
|
||||||
|
// 时间选择器适配
|
||||||
|
.el-date-range-picker {
|
||||||
|
width: 100vw;
|
||||||
|
.el-picker-panel__body {
|
||||||
|
min-width: 100%;
|
||||||
|
.el-date-range-picker__content {
|
||||||
|
.el-date-range-picker__header div {
|
||||||
|
margin-left: 22px;
|
||||||
|
margin-right: 0px;
|
||||||
|
}
|
||||||
|
& + .el-date-range-picker__content {
|
||||||
|
.el-date-range-picker__header div {
|
||||||
|
margin-left: 0px;
|
||||||
|
margin-right: 22px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
12
web/src/theme/media/dialog.scss
Normal file
12
web/src/theme/media/dialog.scss
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于800px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: 800px) {
|
||||||
|
.el-dialog {
|
||||||
|
width: 90% !important;
|
||||||
|
}
|
||||||
|
.el-dialog.is-fullscreen {
|
||||||
|
width: 100% !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
45
web/src/theme/media/error.scss
Normal file
45
web/src/theme/media/error.scss
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于768px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $sm) {
|
||||||
|
.error {
|
||||||
|
.error-flex {
|
||||||
|
flex-direction: column-reverse !important;
|
||||||
|
height: auto !important;
|
||||||
|
width: 100% !important;
|
||||||
|
}
|
||||||
|
.right,
|
||||||
|
.left {
|
||||||
|
flex: unset !important;
|
||||||
|
display: flex !important;
|
||||||
|
}
|
||||||
|
.left-item {
|
||||||
|
margin: auto !important;
|
||||||
|
}
|
||||||
|
.right img {
|
||||||
|
max-width: 450px !important;
|
||||||
|
@extend .left-item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度大于768px小于992px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (min-width: $sm) and (max-width: $md) {
|
||||||
|
.error {
|
||||||
|
.error-flex {
|
||||||
|
padding-left: 30px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度小于1200px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $lg) {
|
||||||
|
.error {
|
||||||
|
.error-flex {
|
||||||
|
padding: 0 30px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
18
web/src/theme/media/form.scss
Normal file
18
web/src/theme/media/form.scss
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于576px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $xs) {
|
||||||
|
.el-form-item__label {
|
||||||
|
width: 100% !important;
|
||||||
|
text-align: left !important;
|
||||||
|
// 移动端 label 右对齐问题
|
||||||
|
justify-content: flex-start !important;
|
||||||
|
}
|
||||||
|
.el-form-item__content {
|
||||||
|
margin-left: 0 !important;
|
||||||
|
}
|
||||||
|
.el-form-item {
|
||||||
|
display: unset !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
23
web/src/theme/media/home.scss
Normal file
23
web/src/theme/media/home.scss
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于768px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $sm) {
|
||||||
|
.home-media,
|
||||||
|
.home-media-sm {
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度小于1200px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $lg) {
|
||||||
|
.home-media-lg {
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
.home-monitor {
|
||||||
|
.flex-warp-item {
|
||||||
|
width: 33.33% !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
15
web/src/theme/media/index.scss
Normal file
15
web/src/theme/media/index.scss
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
/* 栅格布局(媒体查询变量)
|
||||||
|
* https://developer.mozilla.org/zh-CN/docs/Learn/CSS/CSS_layout/Media_queries
|
||||||
|
* $us ≥376px 响应式栅格
|
||||||
|
* $xs ≥576px 响应式栅格
|
||||||
|
* $sm ≥768px 响应式栅格
|
||||||
|
* $md ≥992px 响应式栅格
|
||||||
|
* $lg ≥1200px 响应式栅格
|
||||||
|
* $xl ≥1920px 响应式栅格
|
||||||
|
------------------------------- */
|
||||||
|
$us: 376px;
|
||||||
|
$xs: 576px;
|
||||||
|
$sm: 768px;
|
||||||
|
$md: 992px;
|
||||||
|
$lg: 1200px;
|
||||||
|
$xl: 1920px;
|
||||||
55
web/src/theme/media/layout.scss
Normal file
55
web/src/theme/media/layout.scss
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
@import './index.scss';
|
||||||
|
|
||||||
|
/* 页面宽度小于576px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $xs) {
|
||||||
|
// MessageBox 弹框
|
||||||
|
.el-message-box {
|
||||||
|
width: 80% !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度小于768px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: $sm) {
|
||||||
|
// Breadcrumb 面包屑
|
||||||
|
.layout-navbars-breadcrumb-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
// 外链视图
|
||||||
|
.layout-view-link {
|
||||||
|
a {
|
||||||
|
max-width: 80%;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 菜单搜索
|
||||||
|
.layout-search-dialog {
|
||||||
|
.el-autocomplete {
|
||||||
|
width: 80% !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面宽度小于1000px
|
||||||
|
------------------------------- */
|
||||||
|
@media screen and (max-width: 1000px) {
|
||||||
|
// 布局配置
|
||||||
|
.layout-drawer-content-flex {
|
||||||
|
position: relative;
|
||||||
|
&::after {
|
||||||
|
content: '手机版不支持切换布局';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
z-index: 1;
|
||||||
|
text-align: center;
|
||||||
|
height: 140px;
|
||||||
|
line-height: 140px;
|
||||||
|
background: rgba(255, 255, 255, 0.9);
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user