mirror of
https://github.com/iczer/vue-antd-admin
synced 2025-04-06 04:00:06 +08:00
Merge remote-tracking branch 'origin/master'
This commit is contained in:
commit
8d42b936e5
2
.env
2
.env
@ -1,3 +1,4 @@
|
|||||||
|
VUE_APP_PUBLIC_PATH=/
|
||||||
VUE_APP_NAME=Admin
|
VUE_APP_NAME=Admin
|
||||||
VUE_APP_ROUTES_KEY=admin.routes
|
VUE_APP_ROUTES_KEY=admin.routes
|
||||||
VUE_APP_PERMISSIONS_KEY=admin.permissions
|
VUE_APP_PERMISSIONS_KEY=admin.permissions
|
||||||
@ -5,4 +6,5 @@ VUE_APP_ROLES_KEY=admin.roles
|
|||||||
VUE_APP_USER_KEY=admin.user
|
VUE_APP_USER_KEY=admin.user
|
||||||
VUE_APP_SETTING_KEY=admin.setting
|
VUE_APP_SETTING_KEY=admin.setting
|
||||||
VUE_APP_TBAS_KEY=admin.tabs
|
VUE_APP_TBAS_KEY=admin.tabs
|
||||||
|
VUE_APP_TBAS_TITLES_KEY=admin.tabs.titles
|
||||||
VUE_APP_API_BASE_URL=http://api.iczer.com
|
VUE_APP_API_BASE_URL=http://api.iczer.com
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -17,3 +17,4 @@ selenium-debug.log
|
|||||||
*.njsproj
|
*.njsproj
|
||||||
*.sln
|
*.sln
|
||||||
package-lock.json
|
package-lock.json
|
||||||
|
.env.production.local
|
||||||
|
@ -18,7 +18,7 @@ Multiple theme modes available:
|
|||||||
|
|
||||||
- Preview:https://iczer.gitee.io/vue-antd-admin
|
- Preview:https://iczer.gitee.io/vue-antd-admin
|
||||||
- Documentation:https://iczer.gitee.io/vue-antd-admin-docs
|
- Documentation:https://iczer.gitee.io/vue-antd-admin-docs
|
||||||
- FAQ:https://iczer.github.io/vue-antd-admin/start/faq.html
|
- FAQ:https://iczer.gitee.io/vue-antd-admin-docs/start/faq.html
|
||||||
- Mirror Repo in China:https://gitee.com/iczer/vue-antd-admin
|
- Mirror Repo in China:https://gitee.com/iczer/vue-antd-admin
|
||||||
|
|
||||||
## Browsers support
|
## Browsers support
|
||||||
@ -43,11 +43,11 @@ $ yarn serve
|
|||||||
$ npm install
|
$ npm install
|
||||||
$ npm run serve
|
$ npm run serve
|
||||||
```
|
```
|
||||||
More instructions at [documentation](https://iczer.github.io/vue-antd-admin).
|
More instructions at [documentation](https://iczer.gitee.io/vue-antd-admin-docs).
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
Any type of contribution is welcome, here are some examples of how you may contribute to this project: :star2::
|
Any type of contribution is welcome, here are some examples of how you may contribute to this project: :star2::
|
||||||
- Use Vue Antd Admin in your daily work.
|
- Use Vue Antd Admin in your daily work.
|
||||||
- Submit [Issue](https://github.com/iczer/vue-antd-admin/issues) to report :bug: or ask questions.
|
- Submit [Issue](https://github.com/iczer/vue-antd-admin/issues) to report :bug: or ask questions.
|
||||||
- Propose [Pull Request](https://github.com/iczer/vue-antd-admin/pulls) to improve our code.
|
- Propose [Pull Request](https://github.com/iczer/vue-antd-admin/pulls) to improve our code.
|
||||||
- Join the community and share your experiences with us. QQ Group: 610090280
|
- Join the community and share your experiences with us. QQ Group:942083829、812277510(full)、610090280(full)
|
||||||
|
13
README.md
13
README.md
@ -18,7 +18,7 @@
|
|||||||
|
|
||||||
- 预览地址:https://iczer.gitee.io/vue-antd-admin
|
- 预览地址:https://iczer.gitee.io/vue-antd-admin
|
||||||
- 使用文档:https://iczer.gitee.io/vue-antd-admin-docs
|
- 使用文档:https://iczer.gitee.io/vue-antd-admin-docs
|
||||||
- 常见问题:https://iczer.github.io/vue-antd-admin/start/faq.html
|
- 常见问题:https://iczer.gitee.io/vue-antd-admin-docs/start/faq.html
|
||||||
- 国内镜像:https://gitee.com/iczer/vue-antd-admin
|
- 国内镜像:https://gitee.com/iczer/vue-antd-admin
|
||||||
|
|
||||||
## 浏览器支持
|
## 浏览器支持
|
||||||
@ -43,11 +43,18 @@ $ yarn serve
|
|||||||
$ npm install
|
$ npm install
|
||||||
$ npm run serve
|
$ npm run serve
|
||||||
```
|
```
|
||||||
更多信息参考 [使用文档](https://iczer.github.io/vue-antd-admin)
|
更多信息参考 [使用文档](https://iczer.gitee.io/vue-antd-admin-docs)
|
||||||
|
|
||||||
## 参与贡献
|
## 参与贡献
|
||||||
我们非常欢迎你的贡献,你可以通过以下方式和我们一起共建 :star2::
|
我们非常欢迎你的贡献,你可以通过以下方式和我们一起共建 :star2::
|
||||||
- 在你的公司或个人项目中使用 Vue Antd Admin。
|
- 在你的公司或个人项目中使用 Vue Antd Admin。
|
||||||
- 通过 [Issue](https://github.com/iczer/vue-antd-admin/issues) 报告:bug:或进行咨询。
|
- 通过 [Issue](https://github.com/iczer/vue-antd-admin/issues) 报告:bug:或进行咨询。
|
||||||
- 提交 [Pull Request](https://github.com/iczer/vue-antd-admin/pulls) 改进 Admin 的代码。
|
- 提交 [Pull Request](https://github.com/iczer/vue-antd-admin/pulls) 改进 Admin 的代码。
|
||||||
- 加入社群,与小伙伴们一同交流心得。QQ群:610090280
|
- 加入社群,与小伙伴们一同交流心得。QQ群:942083829、 812277510(已满)、610090280(已满)
|
||||||
|
|
||||||
|
## 打赏
|
||||||
|
如果该项目对您有所帮助,可以请作者喝一杯咖啡。
|
||||||
|
<p>
|
||||||
|
<img src="./src/assets/img/alipay.png" width="320px" style="display: inline-block;" />
|
||||||
|
<img src="./src/assets/img/wechatpay.png" width="320px" style="display: inline-block; margin-left: 24px;" />
|
||||||
|
</p>
|
||||||
|
@ -36,7 +36,8 @@ module.exports = {
|
|||||||
title: '进阶',
|
title: '进阶',
|
||||||
collapsable: false,
|
collapsable: false,
|
||||||
children: [
|
children: [
|
||||||
'/advance/i18n', '/advance/async', '/advance/authority', '/advance/login', '/advance/guard', '/advance/interceptors'
|
'/advance/i18n', '/advance/async', '/advance/authority', '/advance/login', '/advance/guard', '/advance/interceptors',
|
||||||
|
'/advance/api'
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
42
docs/advance/api.md
Normal file
42
docs/advance/api.md
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
---
|
||||||
|
title: 全局API
|
||||||
|
lang: zn-CN
|
||||||
|
---
|
||||||
|
# 全局API
|
||||||
|
我们提供了一些全局Api,在日常功能开发中或许会有帮助,它们均被绑定到了页面组件或子组件实例上。
|
||||||
|
在组件内可以直接通过`this.$[apiName]`的方式调用。如下:
|
||||||
|
|
||||||
|
## 多页签
|
||||||
|
### $closePage(closeRoute, nextRoute)
|
||||||
|
该api用于关闭当前已打开的页签,接收两个参数:
|
||||||
|
* **closeRoute**
|
||||||
|
要关闭的页签对应的 route 对象,可简写为路由的 fullPath 字符串值。
|
||||||
|
* **nextRoute**
|
||||||
|
关闭页签要后跳转的 route 对象,可不传,不传则会自动选择打开页签(临近原则)。
|
||||||
|
|
||||||
|
### $refreshPage(route)
|
||||||
|
该api用于刷新路由对应的页签,接收一个参数:
|
||||||
|
* **route**
|
||||||
|
要刷新的页签对应的 route 对象,可简写为路由的 fullPath 字符串值。
|
||||||
|
|
||||||
|
### $openPage(route, title)
|
||||||
|
该api用于打开一个新页签,接收两个参数:
|
||||||
|
* **route**
|
||||||
|
要打开的页签对应的 route 对象,可简写为路由的 fullPath 字符串值。
|
||||||
|
* **title**
|
||||||
|
设置打开页签的标题,可不传。
|
||||||
|
|
||||||
|
### $setPageTitle(route, title)
|
||||||
|
该api用于设置页签的标题,接收两个参数:
|
||||||
|
* **route**
|
||||||
|
要设置的页签对应的 route 对象,可简写为路由的 fullPath 字符串值。
|
||||||
|
* **title**
|
||||||
|
页签的标题。
|
||||||
|
|
||||||
|
## 权限
|
||||||
|
### $auth(check, type)
|
||||||
|
该api可以用于操作权限校验,接收两个参数:
|
||||||
|
* **check**
|
||||||
|
需要要校验的操作权限
|
||||||
|
* **type**
|
||||||
|
操作权限校验类别,可选 `permission` 和 `role`,即通过权限校验还是角色进行校验,可不传(不传的话,会对两种类型都进行匹配,任意一种匹配成功即校验通过)。
|
@ -3,5 +3,127 @@ title: 国际化
|
|||||||
lang: zn-CN
|
lang: zn-CN
|
||||||
---
|
---
|
||||||
# 国际化
|
# 国际化
|
||||||
|
vue-antd-admin 采用 [vue-i18n](https://kazupon.github.io/vue-i18n/) 插件来实现国际化,该项目已经内置并且加载好了基础配置。可以直接上手使用。
|
||||||
|
|
||||||
### 作者还没来得及编辑该页面,如果你感兴趣,可以点击下方链接,帮助作者完善此页
|
> 如果你还没有看快速入门,请先移步查看: [页面 -> i18n国际化配置](../develop/page.html#i18n国际化配置)
|
||||||
|
|
||||||
|
|
||||||
|
## 菜单和路由
|
||||||
|
|
||||||
|
### 默认情况
|
||||||
|
如果你没有对菜单进行国际化配置,admin 默认会从路由数据中提取数据作为国际化配置。route.name 作为中文语言,route.path 作为英文语言。
|
||||||
|
国际化提取函数定义在 `@/utils/i18n.js` 文件中,会在路由加载时调用,如下:
|
||||||
|
```js
|
||||||
|
/**
|
||||||
|
* 从路由提取国际化数据
|
||||||
|
* @param i18n
|
||||||
|
* @param routes
|
||||||
|
*/
|
||||||
|
function mergeI18nFromRoutes(i18n, routes) {
|
||||||
|
formatFullPath(routes)
|
||||||
|
const CN = generateI18n(new Object(), routes, 'name')
|
||||||
|
const US = generateI18n(new Object(), routes, 'path')
|
||||||
|
i18n.mergeLocaleMessage('CN', CN)
|
||||||
|
i18n.mergeLocaleMessage('US', US)
|
||||||
|
const messages = routesI18n.messages
|
||||||
|
Object.keys(messages).forEach(lang => {
|
||||||
|
i18n.mergeLocaleMessage(lang, messages[lang])
|
||||||
|
})
|
||||||
|
}
|
||||||
|
```
|
||||||
|
### 自定义
|
||||||
|
如果你想自定义菜单国际化数据,可在 `@/router/i18n.js` 文件中配置。我们以路由的 path 作为 key(嵌套path 的写法也会被解析),name 作为 国际化语言的值。
|
||||||
|
假设你有一个路由的配置如下:
|
||||||
|
```js
|
||||||
|
[{
|
||||||
|
path: 'parent',
|
||||||
|
...
|
||||||
|
children: [{
|
||||||
|
path: 'self',
|
||||||
|
...
|
||||||
|
}]
|
||||||
|
}]
|
||||||
|
|
||||||
|
or
|
||||||
|
|
||||||
|
[{
|
||||||
|
path: 'other',
|
||||||
|
...
|
||||||
|
children: [{
|
||||||
|
path: '/parent/self', // 在国际化配置中 key 会解析为 parent.self
|
||||||
|
...
|
||||||
|
}]
|
||||||
|
}]
|
||||||
|
```
|
||||||
|
那么你需要在 `@/router/i18n.js` 中这样配置:
|
||||||
|
```jsx
|
||||||
|
messages: {
|
||||||
|
CN: {
|
||||||
|
parent: {
|
||||||
|
name: '父級菜單',
|
||||||
|
self: {name: '菜單名'},
|
||||||
|
},
|
||||||
|
US: {
|
||||||
|
parent: {
|
||||||
|
name: 'parent menu',
|
||||||
|
self: {name: 'menu name'},
|
||||||
|
},
|
||||||
|
HK: {
|
||||||
|
parent: {
|
||||||
|
name: '父級菜單',
|
||||||
|
self: {name: '菜單名'},
|
||||||
|
},
|
||||||
|
```
|
||||||
|
|
||||||
|
## 添加语言
|
||||||
|
|
||||||
|
首先在 `@/layouts/header/AdminHeader.vue` ,新增一门语言 (多个同理)。
|
||||||
|
|
||||||
|
```vue {15}
|
||||||
|
<template>
|
||||||
|
...
|
||||||
|
</template>
|
||||||
|
<script>
|
||||||
|
...
|
||||||
|
export default {
|
||||||
|
...
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
langList: [
|
||||||
|
{key: 'CN', name: '简体中文', alias: '简体'},
|
||||||
|
{key: 'HK', name: '繁體中文', alias: '繁體'},
|
||||||
|
{key: 'US', name: 'English', alias: 'English'},
|
||||||
|
// 新增一个语言选项, key是i18n的索引,name是菜单显示名称
|
||||||
|
{key: 'JP', name: 'Japanese', alias: 'Japanese'}
|
||||||
|
],
|
||||||
|
searchActive: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
```
|
||||||
|
|
||||||
|
> TIP: 后续开发建议把这里改成动态配置的方式!
|
||||||
|
|
||||||
|
然后开始往 `@/router/i18n.js` 和 `@/pages/你的页面/i18n.js` 里面分别添加上语言的翻译。
|
||||||
|
|
||||||
|
```vue {12,13,14}
|
||||||
|
module.exports = {
|
||||||
|
messages: {
|
||||||
|
CN: {
|
||||||
|
home: {name: '首页'},
|
||||||
|
},
|
||||||
|
US: {
|
||||||
|
home: {name: 'home'},
|
||||||
|
},
|
||||||
|
HK: {
|
||||||
|
home: {name: '首頁'},
|
||||||
|
},
|
||||||
|
JP: {
|
||||||
|
home: {name: '最初のページ'},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
> Notice: 更多用法请移步到 [vue-i18n](https://kazupon.github.io/vue-i18n/) 。
|
@ -5,4 +5,4 @@ lang: zh-CN
|
|||||||
# 社区
|
# 社区
|
||||||
|
|
||||||
## 交流学习
|
## 交流学习
|
||||||
### QQ群:610090280
|
### QQ群:812277510、610090280(已满)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "vue-antd-admin",
|
"name": "vue-antd-admin",
|
||||||
"version": "0.6.1",
|
"version": "0.7.2",
|
||||||
"homepage": "https://iczer.github.io/vue-antd-admin",
|
"homepage": "https://iczer.github.io/vue-antd-admin",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -16,7 +16,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@antv/data-set": "^0.11.4",
|
"@antv/data-set": "^0.11.4",
|
||||||
"animate.css": "^4.1.0",
|
"animate.css": "^4.1.0",
|
||||||
"ant-design-vue": "^1.6.2",
|
"ant-design-vue": "1.7.2",
|
||||||
"axios": "^0.19.2",
|
"axios": "^0.19.2",
|
||||||
"clipboard": "^2.0.6",
|
"clipboard": "^2.0.6",
|
||||||
"core-js": "^3.6.5",
|
"core-js": "^3.6.5",
|
||||||
|
@ -11,11 +11,13 @@
|
|||||||
<link rel="stylesheet" href="<%= htmlWebpackPlugin.options.cdn.css[i] %>" />
|
<link rel="stylesheet" href="<%= htmlWebpackPlugin.options.cdn.css[i] %>" />
|
||||||
<% } %>
|
<% } %>
|
||||||
</head>
|
</head>
|
||||||
<body class="beauty-scroll">
|
<body>
|
||||||
<noscript>
|
<noscript>
|
||||||
<strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
|
<strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
|
||||||
</noscript>
|
</noscript>
|
||||||
<div id="app"></div>
|
<div id="popContainer" class="beauty-scroll" style="height: 100vh; overflow-y: scroll">
|
||||||
|
<div id="app"></div>
|
||||||
|
</div>
|
||||||
<!-- require cdn assets js -->
|
<!-- require cdn assets js -->
|
||||||
<% for (var i in htmlWebpackPlugin.options.cdn && htmlWebpackPlugin.options.cdn.js) { %>
|
<% for (var i in htmlWebpackPlugin.options.cdn && htmlWebpackPlugin.options.cdn.js) { %>
|
||||||
<script type="text/javascript" src="<%= htmlWebpackPlugin.options.cdn.js[i] %>"></script>
|
<script type="text/javascript" src="<%= htmlWebpackPlugin.options.cdn.js[i] %>"></script>
|
||||||
|
11
src/App.vue
11
src/App.vue
@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<a-config-provider :locale="locale">
|
<a-config-provider :locale="locale" :get-popup-container="popContainer">
|
||||||
<router-view/>
|
<router-view/>
|
||||||
</a-config-provider>
|
</a-config-provider>
|
||||||
</template>
|
</template>
|
||||||
@ -31,6 +31,7 @@ export default {
|
|||||||
},
|
},
|
||||||
lang(val) {
|
lang(val) {
|
||||||
this.setLanguage(val)
|
this.setLanguage(val)
|
||||||
|
this.setHtmlTitle()
|
||||||
},
|
},
|
||||||
$route() {
|
$route() {
|
||||||
this.setHtmlTitle()
|
this.setHtmlTitle()
|
||||||
@ -42,10 +43,13 @@ export default {
|
|||||||
'theme.color': function(val) {
|
'theme.color': function(val) {
|
||||||
let closeMessage = this.$message.loading(`您选择了主题色 ${val}, 正在切换...`)
|
let closeMessage = this.$message.loading(`您选择了主题色 ${val}, 正在切换...`)
|
||||||
themeUtil.changeThemeColor(val, this.theme.mode).then(closeMessage)
|
themeUtil.changeThemeColor(val, this.theme.mode).then(closeMessage)
|
||||||
|
},
|
||||||
|
'layout': function() {
|
||||||
|
window.dispatchEvent(new Event('resize'))
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState('setting', ['theme', 'weekMode', 'lang'])
|
...mapState('setting', ['layout', 'theme', 'weekMode', 'lang'])
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
...mapMutations('setting', ['setDevice']),
|
...mapMutations('setting', ['setDevice']),
|
||||||
@ -76,6 +80,9 @@ export default {
|
|||||||
const key = route.path === '/' ? 'home.name' : getI18nKey(route.matched[route.matched.length - 1].path)
|
const key = route.path === '/' ? 'home.name' : getI18nKey(route.matched[route.matched.length - 1].path)
|
||||||
document.title = process.env.VUE_APP_NAME + ' | ' + this.$t(key)
|
document.title = process.env.VUE_APP_NAME + ' | ' + this.$t(key)
|
||||||
},
|
},
|
||||||
|
popContainer() {
|
||||||
|
return document.getElementById("popContainer")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
BIN
src/assets/img/alipay.png
Normal file
BIN
src/assets/img/alipay.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 150 KiB |
BIN
src/assets/img/wechatpay.png
Normal file
BIN
src/assets/img/wechatpay.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 119 KiB |
37
src/components/cache/AKeepAlive.js
vendored
37
src/components/cache/AKeepAlive.js
vendored
@ -4,7 +4,16 @@ const patternTypes = [String, RegExp, Array]
|
|||||||
|
|
||||||
function matches (pattern, name) {
|
function matches (pattern, name) {
|
||||||
if (Array.isArray(pattern)) {
|
if (Array.isArray(pattern)) {
|
||||||
return pattern.indexOf(name) > -1
|
if (pattern.indexOf(name) > -1) {
|
||||||
|
return true
|
||||||
|
} else {
|
||||||
|
for (let item of pattern) {
|
||||||
|
if (isRegExp(item) && item.test(name)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
} else if (typeof pattern === 'string') {
|
} else if (typeof pattern === 'string') {
|
||||||
return pattern.split(',').indexOf(name) > -1
|
return pattern.split(',').indexOf(name) > -1
|
||||||
} else if (isRegExp(pattern)) {
|
} else if (isRegExp(pattern)) {
|
||||||
@ -18,6 +27,13 @@ function getComponentName (opts) {
|
|||||||
return opts && (opts.Ctor.options.name || opts.tag)
|
return opts && (opts.Ctor.options.name || opts.tag)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getComponentKey (vnode) {
|
||||||
|
const {componentOptions, key} = vnode
|
||||||
|
return key == null
|
||||||
|
? componentOptions.Ctor.cid + (componentOptions.tag ? `::${componentOptions.tag}` : '')
|
||||||
|
: key + componentOptions.Ctor.cid
|
||||||
|
}
|
||||||
|
|
||||||
function getFirstComponentChild (children) {
|
function getFirstComponentChild (children) {
|
||||||
if (Array.isArray(children)) {
|
if (Array.isArray(children)) {
|
||||||
for (let i = 0; i < children.length; i++) {
|
for (let i = 0; i < children.length; i++) {
|
||||||
@ -35,7 +51,8 @@ function pruneCache (keepAliveInstance, filter) {
|
|||||||
const cachedNode = cache[key]
|
const cachedNode = cache[key]
|
||||||
if (cachedNode) {
|
if (cachedNode) {
|
||||||
const name = getComponentName(cachedNode.componentOptions)
|
const name = getComponentName(cachedNode.componentOptions)
|
||||||
if (name && !filter(name)) {
|
const componentKey = getComponentKey(cachedNode)
|
||||||
|
if (name && !filter(name, componentKey)) {
|
||||||
pruneCacheEntry(cache, key, keys, _vnode)
|
pruneCacheEntry(cache, key, keys, _vnode)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -70,6 +87,7 @@ export default {
|
|||||||
props: {
|
props: {
|
||||||
include: patternTypes,
|
include: patternTypes,
|
||||||
exclude: patternTypes,
|
exclude: patternTypes,
|
||||||
|
excludeKeys: patternTypes,
|
||||||
max: [String, Number],
|
max: [String, Number],
|
||||||
clearCaches: Array
|
clearCaches: Array
|
||||||
},
|
},
|
||||||
@ -98,10 +116,13 @@ export default {
|
|||||||
|
|
||||||
mounted () {
|
mounted () {
|
||||||
this.$watch('include', val => {
|
this.$watch('include', val => {
|
||||||
pruneCache(this, name => matches(val, name))
|
pruneCache(this, (name) => matches(val, name))
|
||||||
})
|
})
|
||||||
this.$watch('exclude', val => {
|
this.$watch('exclude', val => {
|
||||||
pruneCache(this, name => !matches(val, name))
|
pruneCache(this, (name) => !matches(val, name))
|
||||||
|
})
|
||||||
|
this.$watch('excludeKeys', val => {
|
||||||
|
pruneCache(this, (name, key) => !matches(val, key))
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -112,12 +133,14 @@ export default {
|
|||||||
if (componentOptions) {
|
if (componentOptions) {
|
||||||
// check pattern
|
// check pattern
|
||||||
const name = getComponentName(componentOptions)
|
const name = getComponentName(componentOptions)
|
||||||
const { include, exclude } = this
|
const componentKey = getComponentKey(vnode)
|
||||||
|
const { include, exclude, excludeKeys } = this
|
||||||
if (
|
if (
|
||||||
// not included
|
// not included
|
||||||
(include && (!name || !matches(include, name))) ||
|
(include && (!name || !matches(include, name))) ||
|
||||||
// excluded
|
// excluded
|
||||||
(exclude && name && matches(exclude, name))
|
(exclude && name && matches(exclude, name)) ||
|
||||||
|
(excludeKeys && componentKey && matches(excludeKeys, componentKey))
|
||||||
) {
|
) {
|
||||||
return vnode
|
return vnode
|
||||||
}
|
}
|
||||||
@ -127,7 +150,7 @@ export default {
|
|||||||
// same constructor may get registered as different local components
|
// same constructor may get registered as different local components
|
||||||
// so cid alone is not enough (#3269)
|
// so cid alone is not enough (#3269)
|
||||||
? componentOptions.Ctor.cid + (componentOptions.tag ? `::${componentOptions.tag}` : '')
|
? componentOptions.Ctor.cid + (componentOptions.tag ? `::${componentOptions.tag}` : '')
|
||||||
: vnode.key
|
: vnode.key + componentOptions.Ctor.cid
|
||||||
if (cache[key]) {
|
if (cache[key]) {
|
||||||
vnode.componentInstance = cache[key].componentInstance
|
vnode.componentInstance = cache[key].componentInstance
|
||||||
// make current key freshest
|
// make current key freshest
|
||||||
|
@ -200,9 +200,10 @@ export default {
|
|||||||
})
|
})
|
||||||
},
|
},
|
||||||
updateMenu () {
|
updateMenu () {
|
||||||
const menuRoutes = this.$route.matched.filter(item => item.path !== '')
|
const matchedRoutes = this.$route.matched.filter(item => item.path !== '')
|
||||||
this.selectedKeys = this.getSelectedKey(this.$route)
|
this.selectedKeys = this.getSelectedKey(this.$route)
|
||||||
let openKeys = menuRoutes.map(item => item.path)
|
let openKeys = matchedRoutes.map(item => item.path)
|
||||||
|
openKeys = openKeys.slice(0, openKeys.length -1)
|
||||||
if (!fastEqual(openKeys, this.sOpenKeys)) {
|
if (!fastEqual(openKeys, this.sOpenKeys)) {
|
||||||
this.collapsed || this.mode === 'horizontal' ? this.cachedOpenKeys = openKeys : this.sOpenKeys = openKeys
|
this.collapsed || this.mode === 'horizontal' ? this.cachedOpenKeys = openKeys : this.sOpenKeys = openKeys
|
||||||
}
|
}
|
||||||
|
@ -95,7 +95,14 @@ export default {
|
|||||||
return {
|
return {
|
||||||
...item,
|
...item,
|
||||||
total: selectedRows.reduce((sum, val) => {
|
total: selectedRows.reduce((sum, val) => {
|
||||||
return sum + val[item.dataIndex]
|
let v
|
||||||
|
try{
|
||||||
|
v = val[item.dataIndex] ? val[item.dataIndex] : eval(`val.${item.dataIndex}`);
|
||||||
|
}catch(_){
|
||||||
|
v = val[item.dataIndex];
|
||||||
|
}
|
||||||
|
v = !isNaN(parseFloat(v)) ? parseFloat(v) : 0;
|
||||||
|
return sum + v
|
||||||
}, 0)
|
}, 0)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -49,31 +49,23 @@
|
|||||||
checkedCounts(val) {
|
checkedCounts(val) {
|
||||||
this.checkAll = val === this.columns.length
|
this.checkAll = val === this.columns.length
|
||||||
this.indeterminate = val > 0 && val < this.columns.length
|
this.indeterminate = val > 0 && val < this.columns.length
|
||||||
|
},
|
||||||
|
columns(newVal, oldVal) {
|
||||||
|
if (newVal != oldVal) {
|
||||||
|
this.checkedCounts = newVal.length
|
||||||
|
this.formatColumns(newVal)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.$emit('update:visibleColumns', [...this.columns])
|
this.formatColumns(this.columns)
|
||||||
for (let col of this.columns) {
|
|
||||||
if (col.visible === undefined) {
|
|
||||||
this.$set(col, 'visible', true)
|
|
||||||
}
|
|
||||||
if (!col.visible) {
|
|
||||||
this.checkedCounts -= 1
|
|
||||||
this.$set(col, 'colSpan', 0)
|
|
||||||
this.$set(col, 'customCell', () => ({style: 'display: none;'}))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
onCheckChange(e, col) {
|
onCheckChange(e, col) {
|
||||||
if (!col.visible) {
|
if (!col.visible) {
|
||||||
this.checkedCounts -= 1
|
this.checkedCounts -= 1
|
||||||
this.$set(col, 'colSpan', 0)
|
|
||||||
this.$set(col, 'customCell', () => ({style: 'display: none;'}))
|
|
||||||
} else {
|
} else {
|
||||||
this.checkedCounts += 1
|
this.checkedCounts += 1
|
||||||
this.$set(col, 'colSpan', undefined)
|
|
||||||
this.$set(col, 'customCell', undefined)
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
fixColumn(fixed, col) {
|
fixColumn(fixed, col) {
|
||||||
@ -85,7 +77,6 @@
|
|||||||
},
|
},
|
||||||
setSearch(col) {
|
setSearch(col) {
|
||||||
this.$set(col, 'searchAble', !col.searchAble)
|
this.$set(col, 'searchAble', !col.searchAble)
|
||||||
console.log(col)
|
|
||||||
if (!col.searchAble && col.search) {
|
if (!col.searchAble && col.search) {
|
||||||
this.resetSearch(col)
|
this.resetSearch(col)
|
||||||
}
|
}
|
||||||
@ -101,13 +92,8 @@
|
|||||||
backColumns.forEach((back, index) => {
|
backColumns.forEach((back, index) => {
|
||||||
const column = columns[index]
|
const column = columns[index]
|
||||||
column.visible = back.visible === undefined || back.visible
|
column.visible = back.visible === undefined || back.visible
|
||||||
if (column.visible) {
|
if (!column.visible) {
|
||||||
this.$set(column, 'colSpan', undefined)
|
|
||||||
this.$set(column, 'customCell', undefined)
|
|
||||||
} else {
|
|
||||||
counts -= 1
|
counts -= 1
|
||||||
this.$set(column, 'colSpan', 0)
|
|
||||||
this.$set(column, 'customCell', () => ({style: 'display: none;'}))
|
|
||||||
}
|
}
|
||||||
if (back.fixed !== undefined) {
|
if (back.fixed !== undefined) {
|
||||||
column.fixed = back.fixed
|
column.fixed = back.fixed
|
||||||
@ -125,18 +111,10 @@
|
|||||||
onCheckAllChange(e) {
|
onCheckAllChange(e) {
|
||||||
if (e.target.checked) {
|
if (e.target.checked) {
|
||||||
this.checkedCounts = this.columns.length
|
this.checkedCounts = this.columns.length
|
||||||
this.columns.forEach(col => {
|
this.columns.forEach(col => col.visible = true)
|
||||||
col.visible = true
|
|
||||||
this.$set(col, 'colSpan', undefined)
|
|
||||||
this.$set(col, 'customCell', undefined)
|
|
||||||
})
|
|
||||||
} else {
|
} else {
|
||||||
this.checkedCounts = 0
|
this.checkedCounts = 0
|
||||||
this.columns.forEach(col => {
|
this.columns.forEach(col => col.visible = false)
|
||||||
col.visible = false
|
|
||||||
this.$set(col, 'colSpan', 0)
|
|
||||||
this.$set(col, 'customCell', () => ({style: 'display: none;'}))
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
getConditions(columns) {
|
getConditions(columns) {
|
||||||
@ -146,6 +124,16 @@
|
|||||||
conditions[col.dataIndex] = col.search.value
|
conditions[col.dataIndex] = col.search.value
|
||||||
})
|
})
|
||||||
return conditions
|
return conditions
|
||||||
|
},
|
||||||
|
formatColumns(columns) {
|
||||||
|
for (let col of columns) {
|
||||||
|
if (col.visible === undefined) {
|
||||||
|
this.$set(col, 'visible', true)
|
||||||
|
}
|
||||||
|
if (!col.visible) {
|
||||||
|
this.checkedCounts -= 1
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -32,7 +32,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<a-table
|
<a-table
|
||||||
v-bind="{...$options.propsData, title: undefined, loading: false}"
|
v-bind="{...$props, columns: visibleColumns, title: undefined, loading: false}"
|
||||||
:size="sSize"
|
:size="sSize"
|
||||||
@expandedRowsChange="onExpandedRowsChange"
|
@expandedRowsChange="onExpandedRowsChange"
|
||||||
@change="onChange"
|
@change="onChange"
|
||||||
@ -62,7 +62,7 @@
|
|||||||
props: {
|
props: {
|
||||||
tableLayout: String,
|
tableLayout: String,
|
||||||
bordered: Boolean,
|
bordered: Boolean,
|
||||||
childrenColumnName: Array[String],
|
childrenColumnName: {type: String, default: 'children'},
|
||||||
columns: Array,
|
columns: Array,
|
||||||
components: Object,
|
components: Object,
|
||||||
dataSource: Array,
|
dataSource: Array,
|
||||||
@ -76,12 +76,12 @@
|
|||||||
indentSize: Number,
|
indentSize: Number,
|
||||||
loading: Boolean,
|
loading: Boolean,
|
||||||
locale: Object,
|
locale: Object,
|
||||||
pagination: Object,
|
pagination: [Object, Boolean],
|
||||||
rowClassName: Function,
|
rowClassName: Function,
|
||||||
rowKey: [String, Function],
|
rowKey: [String, Function],
|
||||||
rowSelection: Object,
|
rowSelection: Object,
|
||||||
scroll: Object,
|
scroll: Object,
|
||||||
showHeader: Boolean,
|
showHeader: {type: Boolean, default: true},
|
||||||
size: String,
|
size: String,
|
||||||
title: String,
|
title: String,
|
||||||
customHeaderRow: Function,
|
customHeaderRow: Function,
|
||||||
@ -109,6 +109,9 @@
|
|||||||
},
|
},
|
||||||
scopedSlots() {
|
scopedSlots() {
|
||||||
return Object.keys(this.$scopedSlots).filter(slot => slot !== 'expandedRowRender' && slot !== 'title')
|
return Object.keys(this.$scopedSlots).filter(slot => slot !== 'expandedRowRender' && slot !== 'title')
|
||||||
|
},
|
||||||
|
visibleColumns(){
|
||||||
|
return this.columns.filter(col => col.visible)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
|
@ -7,7 +7,10 @@
|
|||||||
{{col.title}}:
|
{{col.title}}:
|
||||||
</template>
|
</template>
|
||||||
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot>
|
<slot v-else-if="col.slots && col.slots.title" :name="col.slots.title"></slot>
|
||||||
<a-switch @change="onSwitchChange(col)" class="switch" v-model="col.search.value" size="small" checked-children="是" un-checked-children="否" />
|
<a-switch @change="onSwitchChange(col)" class="switch" v-model="col.search.value" size="small"
|
||||||
|
:checked-children="(col.search.switchOptions && col.search.switchOptions.checkedText) || '是'"
|
||||||
|
:un-checked-children="(col.search.switchOptions && col.search.switchOptions.uncheckedText) || '否'"
|
||||||
|
/>
|
||||||
<a-icon v-if="col.search.value !== undefined" class="close" @click="e => onCloseClick(e, col)" type="close-circle" theme="filled" />
|
<a-icon v-if="col.search.value !== undefined" class="close" @click="e => onCloseClick(e, col)" type="close-circle" theme="filled" />
|
||||||
</div>
|
</div>
|
||||||
<div v-else-if="col.dataType === 'time'" :class="['title', {active: col.search.value}]">
|
<div v-else-if="col.dataType === 'time'" :class="['title', {active: col.search.value}]">
|
||||||
@ -70,12 +73,14 @@
|
|||||||
props: ['columns', 'formatConditions'],
|
props: ['columns', 'formatConditions'],
|
||||||
inject: ['table'],
|
inject: ['table'],
|
||||||
created() {
|
created() {
|
||||||
this.columns.forEach(item => {
|
this.formatColumns(this.columns)
|
||||||
this.$set(item, 'search', {...item.search, visible: false, value: undefined, format: this.getFormat(item)})
|
|
||||||
})
|
|
||||||
console.log(this.columns)
|
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
|
columns(newVal, oldVal) {
|
||||||
|
if (newVal != oldVal) {
|
||||||
|
this.formatColumns(newVal)
|
||||||
|
}
|
||||||
|
},
|
||||||
searchCols(newVal, oldVal) {
|
searchCols(newVal, oldVal) {
|
||||||
if (newVal.length != oldVal.length) {
|
if (newVal.length != oldVal.length) {
|
||||||
const newConditions = this.getConditions(newVal)
|
const newConditions = this.getConditions(newVal)
|
||||||
@ -218,6 +223,11 @@
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
|
},
|
||||||
|
formatColumns(columns) {
|
||||||
|
columns.forEach(item => {
|
||||||
|
this.$set(item, 'search', {...item.search, visible: false, value: undefined, format: this.getFormat(item)})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,7 @@ const Item = {
|
|||||||
return h(
|
return h(
|
||||||
'li',
|
'li',
|
||||||
{class: 'avatar-item'},
|
{class: 'avatar-item'},
|
||||||
[!this.$props.tips ? h(ATooltip, {props: {title: this.$props.tips}}, [avatar]) : avatar]
|
[this.$props.tips ? h(ATooltip, {props: {title: this.$props.tips}}, [avatar]) : avatar]
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@ module.exports = {
|
|||||||
mode: 'dark', //主题模式 可选 dark、 light 和 night
|
mode: 'dark', //主题模式 可选 dark、 light 和 night
|
||||||
success: '#52c41a', //成功色
|
success: '#52c41a', //成功色
|
||||||
warning: '#faad14', //警告色
|
warning: '#faad14', //警告色
|
||||||
error: '#f5222d', //错误色
|
error: '#f5222f', //错误色
|
||||||
},
|
},
|
||||||
layout: 'side', //导航布局,可选 side 和 head,分别为侧边导航和顶部导航
|
layout: 'side', //导航布局,可选 side 和 head,分别为侧边导航和顶部导航
|
||||||
fixedHeader: false, //固定头部状态栏,true:固定,false:不固定
|
fixedHeader: false, //固定头部状态栏,true:固定,false:不固定
|
||||||
@ -15,6 +15,7 @@ module.exports = {
|
|||||||
pageWidth: 'fixed', //内容区域宽度,fixed:固定宽度,fluid:流式宽度
|
pageWidth: 'fixed', //内容区域宽度,fixed:固定宽度,fluid:流式宽度
|
||||||
weekMode: false, //色弱模式,true:开启,false:不开启
|
weekMode: false, //色弱模式,true:开启,false:不开启
|
||||||
multiPage: false, //多页签模式,true:开启,false:不开启
|
multiPage: false, //多页签模式,true:开启,false:不开启
|
||||||
|
cachePage: true, //是否缓存页面数据,仅多页签模式下生效,true 缓存, false 不缓存
|
||||||
hideSetting: false, //隐藏设置抽屉,true:隐藏,false:不隐藏
|
hideSetting: false, //隐藏设置抽屉,true:隐藏,false:不隐藏
|
||||||
systemName: 'Vue Antd Admin', //系统名称
|
systemName: 'Vue Antd Admin', //系统名称
|
||||||
copyright: '2018 ICZER 工作室出品', //copyright
|
copyright: '2018 ICZER 工作室出品', //copyright
|
||||||
|
@ -19,12 +19,24 @@ const cssResolve = {
|
|||||||
return cssObj.toText()
|
return cssObj.toText()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
'.ant-tree-checkbox-checked .ant-tree-checkbox-inner::after': {
|
||||||
|
resolve(cssText, cssObj) {
|
||||||
|
cssObj.rules.push('border-top:0', 'border-left:0')
|
||||||
|
return cssObj.toText()
|
||||||
|
}
|
||||||
|
},
|
||||||
'.ant-checkbox-checked .ant-checkbox-inner:after': {
|
'.ant-checkbox-checked .ant-checkbox-inner:after': {
|
||||||
resolve(cssText, cssObj) {
|
resolve(cssText, cssObj) {
|
||||||
cssObj.rules.push('border-top:0', 'border-left:0')
|
cssObj.rules.push('border-top:0', 'border-left:0')
|
||||||
return cssObj.toText()
|
return cssObj.toText()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
'.ant-tree-checkbox-checked .ant-tree-checkbox-inner:after': {
|
||||||
|
resolve(cssText, cssObj) {
|
||||||
|
cssObj.rules.push('border-top:0', 'border-left:0')
|
||||||
|
return cssObj.toText()
|
||||||
|
}
|
||||||
|
},
|
||||||
'.ant-menu-dark .ant-menu-inline.ant-menu-sub': {
|
'.ant-menu-dark .ant-menu-inline.ant-menu-sub': {
|
||||||
resolve(cssText, cssObj) {
|
resolve(cssText, cssObj) {
|
||||||
cssObj.rules = cssObj.rules.filter(rule => rule.indexOf('box-shadow') == -1)
|
cssObj.rules = cssObj.rules.filter(rule => rule.indexOf('box-shadow') == -1)
|
||||||
|
@ -60,10 +60,10 @@ export default {
|
|||||||
this.updatePageHeight(0)
|
this.updatePageHeight(0)
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState('setting', ['layout', 'multiPage', 'pageMinHeight', 'pageWidth']),
|
...mapState('setting', ['layout', 'multiPage', 'pageMinHeight', 'pageWidth', 'customTitles']),
|
||||||
pageTitle() {
|
pageTitle() {
|
||||||
let pageTitle = this.page && this.page.title
|
let pageTitle = this.page && this.page.title
|
||||||
return pageTitle === undefined ? (this.title || this.routeName) : this.$t(pageTitle)
|
return this.customTitle || (pageTitle && this.$t(pageTitle)) || this.title || this.routeName
|
||||||
},
|
},
|
||||||
routeName() {
|
routeName() {
|
||||||
const route = this.$route
|
const route = this.$route
|
||||||
@ -90,11 +90,17 @@ export default {
|
|||||||
...mapMutations('setting', ['correctPageMinHeight']),
|
...mapMutations('setting', ['correctPageMinHeight']),
|
||||||
getRouteBreadcrumb() {
|
getRouteBreadcrumb() {
|
||||||
let routes = this.$route.matched
|
let routes = this.$route.matched
|
||||||
|
const path = this.$route.path
|
||||||
let breadcrumb = []
|
let breadcrumb = []
|
||||||
routes.forEach(route => {
|
routes.filter(item => path.includes(item.path))
|
||||||
|
.forEach(route => {
|
||||||
const path = route.path.length === 0 ? '/home' : route.path
|
const path = route.path.length === 0 ? '/home' : route.path
|
||||||
breadcrumb.push(this.$t(getI18nKey(path)))
|
breadcrumb.push(this.$t(getI18nKey(path)))
|
||||||
})
|
})
|
||||||
|
let pageTitle = this.page && this.page.title
|
||||||
|
if (this.customTitle || pageTitle) {
|
||||||
|
breadcrumb[breadcrumb.length - 1] = this.customTitle || pageTitle
|
||||||
|
}
|
||||||
return breadcrumb
|
return breadcrumb
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
|
@ -26,6 +26,9 @@ export default {
|
|||||||
.copyright{
|
.copyright{
|
||||||
color: @text-color-second;
|
color: @text-color-second;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
|
i {
|
||||||
|
margin: 0 4px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.links{
|
.links{
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
|
@ -63,7 +63,7 @@
|
|||||||
this.affixed = this.fixedTabs
|
this.affixed = this.fixedTabs
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState('setting', ['layout', 'pageWidth', 'fixedHeader', 'fixedTabs']),
|
...mapState('setting', ['layout', 'pageWidth', 'fixedHeader', 'fixedTabs', 'customTitles']),
|
||||||
lockTitle() {
|
lockTitle() {
|
||||||
return this.$t(this.fixedTabs ? 'unlock' : 'lock')
|
return this.$t(this.fixedTabs ? 'unlock' : 'lock')
|
||||||
}
|
}
|
||||||
@ -95,7 +95,9 @@
|
|||||||
this.$emit('contextmenu', pageKey, e)
|
this.$emit('contextmenu', pageKey, e)
|
||||||
},
|
},
|
||||||
pageName(page) {
|
pageName(page) {
|
||||||
return this.$t(getI18nKey(page.keyPath))
|
const pagePath = page.fullPath.split('?')[0]
|
||||||
|
const custom = this.customTitles.find(item => item.path === pagePath)
|
||||||
|
return (custom && custom.title) || page.title || this.$t(getI18nKey(page.keyPath))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -115,7 +117,7 @@
|
|||||||
.icon-close{
|
.icon-close{
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
margin-left: 6px;
|
margin-left: 6px;
|
||||||
margin-right: -4px;
|
margin-right: -4px !important;
|
||||||
color: @text-color-second;
|
color: @text-color-second;
|
||||||
&:hover{
|
&:hover{
|
||||||
color: @text-color;
|
color: @text-color;
|
||||||
|
@ -12,10 +12,10 @@
|
|||||||
/>
|
/>
|
||||||
<div :class="['tabs-view-content', layout, pageWidth]" :style="`margin-top: ${multiPage ? -24 : 0}px`">
|
<div :class="['tabs-view-content', layout, pageWidth]" :style="`margin-top: ${multiPage ? -24 : 0}px`">
|
||||||
<page-toggle-transition :disabled="animate.disabled" :animate="animate.name" :direction="animate.direction">
|
<page-toggle-transition :disabled="animate.disabled" :animate="animate.name" :direction="animate.direction">
|
||||||
<a-keep-alive v-if="multiPage" v-model="clearCaches">
|
<a-keep-alive :exclude-keys="excludeKeys" v-if="multiPage && cachePage" v-model="clearCaches">
|
||||||
<router-view v-if="!refreshing" ref="tabContent" :key="$route.fullPath" />
|
<router-view v-if="!refreshing" ref="tabContent" :key="$route.fullPath" />
|
||||||
</a-keep-alive>
|
</a-keep-alive>
|
||||||
<router-view v-else />
|
<router-view ref="tabContent" v-else-if="!refreshing" />
|
||||||
</page-toggle-transition>
|
</page-toggle-transition>
|
||||||
</div>
|
</div>
|
||||||
</admin-layout>
|
</admin-layout>
|
||||||
@ -40,11 +40,12 @@ export default {
|
|||||||
pageList: [],
|
pageList: [],
|
||||||
activePage: '',
|
activePage: '',
|
||||||
menuVisible: false,
|
menuVisible: false,
|
||||||
refreshing: false
|
refreshing: false,
|
||||||
|
excludeKeys: []
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState('setting', ['multiPage', 'animate', 'layout', 'pageWidth']),
|
...mapState('setting', ['multiPage', 'cachePage', 'animate', 'layout', 'pageWidth']),
|
||||||
menuItemList() {
|
menuItemList() {
|
||||||
return [
|
return [
|
||||||
{ key: '1', icon: 'vertical-right', text: this.$t('closeLeft') },
|
{ key: '1', icon: 'vertical-right', text: this.$t('closeLeft') },
|
||||||
@ -58,6 +59,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
created () {
|
created () {
|
||||||
|
this.loadCacheConfig(this.$router?.options?.routes)
|
||||||
this.loadCachedTabs()
|
this.loadCachedTabs()
|
||||||
const route = this.$route
|
const route = this.$route
|
||||||
if (this.pageList.findIndex(item => item.fullPath === route.fullPath) === -1) {
|
if (this.pageList.findIndex(item => item.fullPath === route.fullPath) === -1) {
|
||||||
@ -79,6 +81,10 @@ export default {
|
|||||||
this.correctPageMinHeight(this.tabsOffset)
|
this.correctPageMinHeight(this.tabsOffset)
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
|
'$router.options.routes': function (val) {
|
||||||
|
this.excludeKeys = []
|
||||||
|
this.loadCacheConfig(val)
|
||||||
|
},
|
||||||
'$route': function (newRoute) {
|
'$route': function (newRoute) {
|
||||||
this.activePage = newRoute.fullPath
|
this.activePage = newRoute.fullPath
|
||||||
if (!this.multiPage) {
|
if (!this.multiPage) {
|
||||||
@ -249,6 +255,7 @@ export default {
|
|||||||
return {
|
return {
|
||||||
keyPath: route.matched[route.matched.length - 1].path,
|
keyPath: route.matched[route.matched.length - 1].path,
|
||||||
fullPath: route.fullPath, loading: false,
|
fullPath: route.fullPath, loading: false,
|
||||||
|
title: route.meta && route.meta.page && route.meta.page.title,
|
||||||
unclose: route.meta && route.meta.page && (route.meta.page.closable === false),
|
unclose: route.meta && route.meta.page && (route.meta.page.closable === false),
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -260,7 +267,8 @@ export default {
|
|||||||
const page = this.pageList.find(item => item.fullPath === route.fullPath)
|
const page = this.pageList.find(item => item.fullPath === route.fullPath)
|
||||||
page.unclose = route.meta && route.meta.page && (route.meta.page.closable === false)
|
page.unclose = route.meta && route.meta.page && (route.meta.page.closable === false)
|
||||||
if (!page._init_) {
|
if (!page._init_) {
|
||||||
page.cachedKey = this.$refs.tabContent.$vnode.key
|
const vnode = this.$refs.tabContent.$vnode
|
||||||
|
page.cachedKey = vnode.key + vnode.componentOptions.Ctor.cid
|
||||||
page._init_ = true
|
page._init_ = true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -282,6 +290,17 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
loadCacheConfig(routes, pCache = true) {
|
||||||
|
routes.forEach(item => {
|
||||||
|
const cacheAble = item.meta?.page?.cacheAble ?? pCache ?? true
|
||||||
|
if (!cacheAble) {
|
||||||
|
this.excludeKeys.push(new RegExp(`${item.fullPath}\\d+$`))
|
||||||
|
}
|
||||||
|
if (item.children) {
|
||||||
|
this.loadCacheConfig(item.children, cacheAble)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
...mapMutations('setting', ['correctPageMinHeight'])
|
...mapMutations('setting', ['correctPageMinHeight'])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,6 +10,7 @@ import 'animate.css/source/animate.css'
|
|||||||
import Plugins from '@/plugins'
|
import Plugins from '@/plugins'
|
||||||
import {initI18n} from '@/utils/i18n'
|
import {initI18n} from '@/utils/i18n'
|
||||||
import bootstrap from '@/bootstrap'
|
import bootstrap from '@/bootstrap'
|
||||||
|
import 'moment/locale/zh-cn'
|
||||||
|
|
||||||
const router = initRouter(store.state.setting.asyncRoutes)
|
const router = initRouter(store.state.setting.asyncRoutes)
|
||||||
const i18n = initI18n('CN', 'US')
|
const i18n = initI18n('CN', 'US')
|
||||||
|
@ -49,3 +49,58 @@ Mock.mock(RegExp(`${process.env.VUE_APP_API_BASE_URL}/goods` + '.*'),'get', ({ur
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const columnsConfig = [
|
||||||
|
{
|
||||||
|
title: '商品名称',
|
||||||
|
dataIndex: 'name',
|
||||||
|
searchAble: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '订单号',
|
||||||
|
dataIndex: 'orderId'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
searchAble: true,
|
||||||
|
dataIndex: 'status',
|
||||||
|
dataType: 'select',
|
||||||
|
slots: {title: 'statusTitle'},
|
||||||
|
scopedSlots: {customRender: 'status'},
|
||||||
|
search: {
|
||||||
|
selectOptions: [
|
||||||
|
{title: '已下单', value: 1},
|
||||||
|
{title: '已付款', value: 2},
|
||||||
|
{title: '已审核', value: 3},
|
||||||
|
// {title: '已发货', value: 4}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '发货',
|
||||||
|
searchAble: true,
|
||||||
|
dataIndex: 'send',
|
||||||
|
dataType: 'boolean',
|
||||||
|
scopedSlots: {customRender: 'send'}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '发货时间',
|
||||||
|
dataIndex: 'sendTime',
|
||||||
|
dataType: 'datetime'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '下单日期',
|
||||||
|
searchAble: true,
|
||||||
|
dataIndex: 'orderDate',
|
||||||
|
dataType: 'date',
|
||||||
|
visible: false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '审核时间',
|
||||||
|
dataIndex: 'auditTime',
|
||||||
|
dataType: 'time',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
Mock.mock(`${process.env.VUE_APP_API_BASE_URL}/columns`, 'get', () => {
|
||||||
|
return columnsConfig
|
||||||
|
})
|
@ -8,21 +8,32 @@ const user = Mock.mock({
|
|||||||
position: '@POSITION'
|
position: '@POSITION'
|
||||||
})
|
})
|
||||||
Mock.mock(`${process.env.VUE_APP_API_BASE_URL}/login`, 'post', ({body}) => {
|
Mock.mock(`${process.env.VUE_APP_API_BASE_URL}/login`, 'post', ({body}) => {
|
||||||
let result = {}
|
let result = {data: {}}
|
||||||
const {name, password} = JSON.parse(body)
|
const {name, password} = JSON.parse(body)
|
||||||
|
|
||||||
if (name !== 'admin' || password !== '888888') {
|
let success = false
|
||||||
result.code = -1
|
|
||||||
result.message = '账户名或密码错误(admin/888888)'
|
if (name === 'admin' && password === '888888') {
|
||||||
|
success = true
|
||||||
|
result.data.permissions = [{id: 'queryForm', operation: ['add', 'edit']}]
|
||||||
|
result.data.roles = [{id: 'admin', operation: ['add', 'edit', 'delete']}]
|
||||||
|
} else if (name === 'test' || password === '888888') {
|
||||||
|
success = true
|
||||||
|
result.data.permissions = [{id: 'queryForm', operation: ['add', 'edit']}]
|
||||||
|
result.data.roles = [{id: 'test', operation: ['add', 'edit', 'delete']}]
|
||||||
} else {
|
} else {
|
||||||
|
success = false
|
||||||
|
}
|
||||||
|
|
||||||
|
if (success) {
|
||||||
result.code = 0
|
result.code = 0
|
||||||
result.message = Mock.mock('@TIMEFIX').CN + ',欢迎回来'
|
result.message = Mock.mock('@TIMEFIX').CN + ',欢迎回来'
|
||||||
result.data = {}
|
|
||||||
result.data.user = user
|
result.data.user = user
|
||||||
result.data.token = 'Authorization:' + Math.random()
|
result.data.token = 'Authorization:' + Math.random()
|
||||||
result.data.expireAt = new Date(new Date().getTime() + 30 * 60 * 1000)
|
result.data.expireAt = new Date(new Date().getTime() + 30 * 60 * 1000)
|
||||||
result.data.permissions = [{id: 'queryForm', operation: ['add', 'edit']}]
|
} else {
|
||||||
result.data.roles = [{id: 'admin', operation: ['add', 'edit', 'delete']}]
|
result.code = -1
|
||||||
|
result.message = '账户名或密码错误(admin/888888 or test/888888)'
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
})
|
})
|
||||||
|
@ -90,24 +90,19 @@
|
|||||||
searchAble: true,
|
searchAble: true,
|
||||||
dataIndex: 'send',
|
dataIndex: 'send',
|
||||||
dataType: 'boolean',
|
dataType: 'boolean',
|
||||||
scopedSlots: {customRender: 'send'}
|
scopedSlots: {customRender: 'send'},
|
||||||
},
|
search: {
|
||||||
{
|
switchOptions: {
|
||||||
title: '发货时间',
|
checkedText: '开',
|
||||||
dataIndex: 'sendTime',
|
uncheckedText: '关'
|
||||||
dataType: 'datetime'
|
}
|
||||||
},
|
}
|
||||||
{
|
|
||||||
title: '下单日期',
|
|
||||||
searchAble: true,
|
|
||||||
dataIndex: 'orderDate',
|
|
||||||
dataType: 'date'
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '审核时间',
|
title: '审核时间',
|
||||||
dataIndex: 'auditTime',
|
dataIndex: 'auditTime',
|
||||||
dataType: 'time',
|
dataType: 'time',
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
dataSource: [],
|
dataSource: [],
|
||||||
conditions: {}
|
conditions: {}
|
||||||
@ -115,6 +110,7 @@
|
|||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.getGoodList()
|
this.getGoodList()
|
||||||
|
this.getColumns()
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
getGoodList() {
|
getGoodList() {
|
||||||
@ -129,8 +125,12 @@
|
|||||||
this.loading = false
|
this.loading = false
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
getColumns() {
|
||||||
|
ds.goodsColumns().then(res => {
|
||||||
|
this.columns = res.data
|
||||||
|
})
|
||||||
|
},
|
||||||
onSearch(conditions, searchOptions) {
|
onSearch(conditions, searchOptions) {
|
||||||
console.log(conditions)
|
|
||||||
console.log(searchOptions)
|
console.log(searchOptions)
|
||||||
this.page = 1
|
this.page = 1
|
||||||
this.conditions = conditions
|
this.conditions = conditions
|
||||||
|
@ -79,7 +79,7 @@
|
|||||||
</a-form>
|
</a-form>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="operator">
|
<a-space class="operator">
|
||||||
<a-button @click="addNew" type="primary">新建</a-button>
|
<a-button @click="addNew" type="primary">新建</a-button>
|
||||||
<a-button >批量操作</a-button>
|
<a-button >批量操作</a-button>
|
||||||
<a-dropdown>
|
<a-dropdown>
|
||||||
@ -91,7 +91,7 @@
|
|||||||
更多操作 <a-icon type="down" />
|
更多操作 <a-icon type="down" />
|
||||||
</a-button>
|
</a-button>
|
||||||
</a-dropdown>
|
</a-dropdown>
|
||||||
</div>
|
</a-space>
|
||||||
<standard-table
|
<standard-table
|
||||||
:columns="columns"
|
:columns="columns"
|
||||||
:dataSource="dataSource"
|
:dataSource="dataSource"
|
||||||
|
@ -6,9 +6,29 @@ const TabsPagePlugin = {
|
|||||||
const event = new CustomEvent('page:close', {detail:{closeRoute, nextRoute}})
|
const event = new CustomEvent('page:close', {detail:{closeRoute, nextRoute}})
|
||||||
window.dispatchEvent(event)
|
window.dispatchEvent(event)
|
||||||
},
|
},
|
||||||
$refreshPage(pageKey) {
|
$refreshPage(route) {
|
||||||
const event = new CustomEvent('page:refresh', {detail:{pageKey}})
|
const path = typeof route === 'object' ? route.path : route
|
||||||
|
const event = new CustomEvent('page:refresh', {detail:{pageKey: path}})
|
||||||
window.dispatchEvent(event)
|
window.dispatchEvent(event)
|
||||||
|
},
|
||||||
|
$openPage(route, title) {
|
||||||
|
this.$setPageTitle(route, title)
|
||||||
|
this.$router.push(route)
|
||||||
|
},
|
||||||
|
$setPageTitle(route, title) {
|
||||||
|
if (title) {
|
||||||
|
let path = typeof route === 'object' ? route.path : route
|
||||||
|
path = path && path.split('?')[0]
|
||||||
|
this.$store.commit('setting/setCustomTitle', {path, title})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
customTitle() {
|
||||||
|
const customTitles = this.$store.state.setting.customTitles
|
||||||
|
const path = this.$route.path.split('?')[0]
|
||||||
|
const custom = customTitles.find(item => item.path === path)
|
||||||
|
return custom && custom.title
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -56,6 +56,9 @@ const options = {
|
|||||||
name: '表单页',
|
name: '表单页',
|
||||||
meta: {
|
meta: {
|
||||||
icon: 'form',
|
icon: 'form',
|
||||||
|
page: {
|
||||||
|
cacheAble: false
|
||||||
|
}
|
||||||
},
|
},
|
||||||
component: PageView,
|
component: PageView,
|
||||||
children: [
|
children: [
|
||||||
|
@ -6,4 +6,5 @@ module.exports = {
|
|||||||
LOGIN: `${BASE_URL}/login`,
|
LOGIN: `${BASE_URL}/login`,
|
||||||
ROUTES: `${BASE_URL}/routes`,
|
ROUTES: `${BASE_URL}/routes`,
|
||||||
GOODS: `${BASE_URL}/goods`,
|
GOODS: `${BASE_URL}/goods`,
|
||||||
|
GOODS_COLUMNS: `${BASE_URL}/columns`,
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,12 @@
|
|||||||
import {GOODS} from './api'
|
import {GOODS, GOODS_COLUMNS} from './api'
|
||||||
import {METHOD, request} from '@/utils/request'
|
import {METHOD, request} from '@/utils/request'
|
||||||
|
|
||||||
export async function goodsList(params) {
|
export async function goodsList(params) {
|
||||||
return request(GOODS, METHOD.GET, params)
|
return request(GOODS, METHOD.GET, params)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {goodsList}
|
export async function goodsColumns() {
|
||||||
|
return request(GOODS_COLUMNS, METHOD.GET)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {goodsList, goodsColumns}
|
@ -3,8 +3,11 @@ import {ADMIN} from '@/config/default'
|
|||||||
import {formatFullPath} from '@/utils/i18n'
|
import {formatFullPath} from '@/utils/i18n'
|
||||||
import {filterMenu} from '@/utils/authority-utils'
|
import {filterMenu} from '@/utils/authority-utils'
|
||||||
import {getLocalSetting} from '@/utils/themeUtil'
|
import {getLocalSetting} from '@/utils/themeUtil'
|
||||||
|
import deepClone from 'lodash.clonedeep'
|
||||||
|
|
||||||
const localSetting = getLocalSetting(true)
|
const localSetting = getLocalSetting(true)
|
||||||
|
const customTitlesStr = sessionStorage.getItem(process.env.VUE_APP_TBAS_TITLES_KEY)
|
||||||
|
const customTitles = (customTitlesStr && JSON.parse(customTitlesStr)) || []
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
namespaced: true,
|
namespaced: true,
|
||||||
@ -15,6 +18,7 @@ export default {
|
|||||||
pageMinHeight: 0,
|
pageMinHeight: 0,
|
||||||
menuData: [],
|
menuData: [],
|
||||||
activatedFirst: undefined,
|
activatedFirst: undefined,
|
||||||
|
customTitles,
|
||||||
...config,
|
...config,
|
||||||
...localSetting
|
...localSetting
|
||||||
},
|
},
|
||||||
@ -22,7 +26,7 @@ export default {
|
|||||||
menuData(state, getters, rootState) {
|
menuData(state, getters, rootState) {
|
||||||
if (state.filterMenu) {
|
if (state.filterMenu) {
|
||||||
const {permissions, roles} = rootState.account
|
const {permissions, roles} = rootState.account
|
||||||
filterMenu(state.menuData, permissions, roles)
|
return filterMenu(deepClone(state.menuData), permissions, roles)
|
||||||
}
|
}
|
||||||
return state.menuData
|
return state.menuData
|
||||||
},
|
},
|
||||||
@ -39,11 +43,11 @@ export default {
|
|||||||
},
|
},
|
||||||
subMenu(state) {
|
subMenu(state) {
|
||||||
const {menuData, activatedFirst} = state
|
const {menuData, activatedFirst} = state
|
||||||
if (!menuData[0].fullPath) {
|
if (menuData.length > 0 && !menuData[0].fullPath) {
|
||||||
formatFullPath(menuData)
|
formatFullPath(menuData)
|
||||||
}
|
}
|
||||||
const current = menuData.find(menu => menu.fullPath === activatedFirst)
|
const current = menuData.find(menu => menu.fullPath === activatedFirst)
|
||||||
return current && current.children ? current.children : []
|
return current && current.children || []
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mutations: {
|
mutations: {
|
||||||
@ -94,6 +98,17 @@ export default {
|
|||||||
},
|
},
|
||||||
setFixedTabs(state, fixedTabs) {
|
setFixedTabs(state, fixedTabs) {
|
||||||
state.fixedTabs = fixedTabs
|
state.fixedTabs = fixedTabs
|
||||||
|
},
|
||||||
|
setCustomTitle(state, {path, title}) {
|
||||||
|
if (title) {
|
||||||
|
const obj = state.customTitles.find(item => item.path === path)
|
||||||
|
if (obj) {
|
||||||
|
obj.title = title
|
||||||
|
} else {
|
||||||
|
state.customTitles.push({path, title})
|
||||||
|
}
|
||||||
|
sessionStorage.setItem(process.env.VUE_APP_TBAS_TITLES_KEY, JSON.stringify(state.customTitles))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -69,15 +69,16 @@ function hasAuthority(route, permissions, roles) {
|
|||||||
* @param roles
|
* @param roles
|
||||||
*/
|
*/
|
||||||
function filterMenu(menuData, permissions, roles) {
|
function filterMenu(menuData, permissions, roles) {
|
||||||
menuData.forEach(menu => {
|
return menuData.filter(menu => {
|
||||||
if (menu.meta && menu.meta.invisible === undefined) {
|
if (menu.meta && menu.meta.invisible === undefined) {
|
||||||
if (!hasAuthority(menu, permissions, roles)) {
|
if (!hasAuthority(menu, permissions, roles)) {
|
||||||
menu.meta.invisible = true
|
return false
|
||||||
}
|
|
||||||
if (menu.children && menu.children.length > 0) {
|
|
||||||
filterMenu(menu.children, permissions, roles)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (menu.children && menu.children.length > 0) {
|
||||||
|
menu.children = filterMenu(menu.children, permissions, roles)
|
||||||
|
}
|
||||||
|
return true
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,8 +9,8 @@ const resp401 = {
|
|||||||
*/
|
*/
|
||||||
onFulfilled(response, options) {
|
onFulfilled(response, options) {
|
||||||
const {message} = options
|
const {message} = options
|
||||||
if (response.status === 401) {
|
if (response.code === 401) {
|
||||||
message.error('无此接口权限')
|
message.error('无此权限')
|
||||||
}
|
}
|
||||||
return response
|
return response
|
||||||
},
|
},
|
||||||
@ -22,7 +22,10 @@ const resp401 = {
|
|||||||
*/
|
*/
|
||||||
onRejected(error, options) {
|
onRejected(error, options) {
|
||||||
const {message} = options
|
const {message} = options
|
||||||
message.error(error.message)
|
const {response} = error
|
||||||
|
if (response.status === 401) {
|
||||||
|
message.error('无此权限')
|
||||||
|
}
|
||||||
return Promise.reject(error)
|
return Promise.reject(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -30,10 +33,18 @@ const resp401 = {
|
|||||||
const resp403 = {
|
const resp403 = {
|
||||||
onFulfilled(response, options) {
|
onFulfilled(response, options) {
|
||||||
const {message} = options
|
const {message} = options
|
||||||
if (response.status === 403) {
|
if (response.code === 403) {
|
||||||
message.error(`请求被拒绝`)
|
message.error('请求被拒绝')
|
||||||
}
|
}
|
||||||
return response
|
return response
|
||||||
|
},
|
||||||
|
onRejected(error, options) {
|
||||||
|
const {message} = options
|
||||||
|
const {response} = error
|
||||||
|
if (response.status === 403) {
|
||||||
|
message.error('请求被拒绝')
|
||||||
|
}
|
||||||
|
return Promise.reject(error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,14 +30,14 @@ const METHOD = {
|
|||||||
* @param params 请求参数
|
* @param params 请求参数
|
||||||
* @returns {Promise<AxiosResponse<T>>}
|
* @returns {Promise<AxiosResponse<T>>}
|
||||||
*/
|
*/
|
||||||
async function request(url, method, params) {
|
async function request(url, method, params, config) {
|
||||||
switch (method) {
|
switch (method) {
|
||||||
case METHOD.GET:
|
case METHOD.GET:
|
||||||
return axios.get(url, {params})
|
return axios.get(url, {params, ...config})
|
||||||
case METHOD.POST:
|
case METHOD.POST:
|
||||||
return axios.post(url, params)
|
return axios.post(url, params, config)
|
||||||
default:
|
default:
|
||||||
return axios.get(url, {params})
|
return axios.get(url, {params, ...config})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -32,9 +32,9 @@ function parseRoutes(routesConfig, routerMap) {
|
|||||||
routesConfig.forEach(item => {
|
routesConfig.forEach(item => {
|
||||||
// 获取注册在 routerMap 中的 router,初始化 routeCfg
|
// 获取注册在 routerMap 中的 router,初始化 routeCfg
|
||||||
let router = undefined, routeCfg = {}
|
let router = undefined, routeCfg = {}
|
||||||
if (typeof item === 'string' && routerMap[item]) {
|
if (typeof item === 'string') {
|
||||||
router = routerMap[item]
|
router = routerMap[item]
|
||||||
routeCfg = {path: router.path || item, router: item}
|
routeCfg = {path: (router && router.path) || item, router: item}
|
||||||
} else if (typeof item === 'object') {
|
} else if (typeof item === 'object') {
|
||||||
router = routerMap[item.router]
|
router = routerMap[item.router]
|
||||||
routeCfg = item
|
routeCfg = item
|
||||||
@ -50,10 +50,10 @@ function parseRoutes(routesConfig, routerMap) {
|
|||||||
component: router.component,
|
component: router.component,
|
||||||
redirect: routeCfg.redirect || router.redirect,
|
redirect: routeCfg.redirect || router.redirect,
|
||||||
meta: {
|
meta: {
|
||||||
authority: routeCfg.authority || router.authority || '*',
|
authority: routeCfg.authority || router.authority || routeCfg.meta?.authority || router.meta?.authority || '*',
|
||||||
icon: routeCfg.icon || router.icon,
|
icon: routeCfg.icon || router.icon || routeCfg.meta?.icon || router.meta?.icon,
|
||||||
page: routeCfg.page || router.page,
|
page: routeCfg.page || router.page || routeCfg.meta?.page || router.meta?.page,
|
||||||
link: routeCfg.link || router.link
|
link: routeCfg.link || router.link || routeCfg.meta?.link || router.meta?.link
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (routeCfg.invisible || router.invisible) {
|
if (routeCfg.invisible || router.invisible) {
|
||||||
|
@ -110,7 +110,7 @@ module.exports = {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
publicPath: isProd ? '/vue-antd-admin/' : '/',
|
publicPath: process.env.VUE_APP_PUBLIC_PATH,
|
||||||
outputDir: 'dist',
|
outputDir: 'dist',
|
||||||
assetsDir: 'static',
|
assetsDir: 'static',
|
||||||
productionSourceMap: false
|
productionSourceMap: false
|
||||||
|
22
yarn.lock
22
yarn.lock
@ -1146,6 +1146,14 @@
|
|||||||
resolved "https://registry.npm.taobao.org/@nodelib/fs.stat/download/@nodelib/fs.stat-1.1.3.tgz#2b5a3ab3f918cca48a8c754c08168e3f03eba61b"
|
resolved "https://registry.npm.taobao.org/@nodelib/fs.stat/download/@nodelib/fs.stat-1.1.3.tgz#2b5a3ab3f918cca48a8c754c08168e3f03eba61b"
|
||||||
integrity sha1-K1o6s/kYzKSKjHVMCBaOPwPrphs=
|
integrity sha1-K1o6s/kYzKSKjHVMCBaOPwPrphs=
|
||||||
|
|
||||||
|
"@simonwep/pickr@~1.7.0":
|
||||||
|
version "1.7.4"
|
||||||
|
resolved "https://registry.npm.taobao.org/@simonwep/pickr/download/@simonwep/pickr-1.7.4.tgz#b14fcd945890388b870cd6db4d6c78d531f25141"
|
||||||
|
integrity sha1-sU/NlFiQOIuHDNbbTWx41THyUUE=
|
||||||
|
dependencies:
|
||||||
|
core-js "^3.6.5"
|
||||||
|
nanopop "^2.1.0"
|
||||||
|
|
||||||
"@sindresorhus/is@^0.14.0":
|
"@sindresorhus/is@^0.14.0":
|
||||||
version "0.14.0"
|
version "0.14.0"
|
||||||
resolved "https://registry.npm.taobao.org/@sindresorhus/is/download/@sindresorhus/is-0.14.0.tgz#9fb3a3cf3132328151f353de4632e01e52102bea"
|
resolved "https://registry.npm.taobao.org/@sindresorhus/is/download/@sindresorhus/is-0.14.0.tgz#9fb3a3cf3132328151f353de4632e01e52102bea"
|
||||||
@ -1959,13 +1967,14 @@ ansi-styles@^4.0.0, ansi-styles@^4.1.0:
|
|||||||
"@types/color-name" "^1.1.1"
|
"@types/color-name" "^1.1.1"
|
||||||
color-convert "^2.0.1"
|
color-convert "^2.0.1"
|
||||||
|
|
||||||
ant-design-vue@^1.6.2:
|
ant-design-vue@1.7.2:
|
||||||
version "1.6.2"
|
version "1.7.2"
|
||||||
resolved "https://registry.npm.taobao.org/ant-design-vue/download/ant-design-vue-1.6.2.tgz?cache=0&sync_timestamp=1591081225900&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fant-design-vue%2Fdownload%2Fant-design-vue-1.6.2.tgz#983634caac9cdaca0b3a095b540105b4f76da29f"
|
resolved "https://registry.npm.taobao.org/ant-design-vue/download/ant-design-vue-1.7.2.tgz#aac7ff802205711631c8698e2a0c7b4e61dfd73e"
|
||||||
integrity sha1-mDY0yqyc2soLOglbVAEFtPdtop8=
|
integrity sha1-qsf/gCIFcRYxyGmOKgx7TmHf1z4=
|
||||||
dependencies:
|
dependencies:
|
||||||
"@ant-design/icons" "^2.1.1"
|
"@ant-design/icons" "^2.1.1"
|
||||||
"@ant-design/icons-vue" "^2.0.0"
|
"@ant-design/icons-vue" "^2.0.0"
|
||||||
|
"@simonwep/pickr" "~1.7.0"
|
||||||
add-dom-event-listener "^1.0.2"
|
add-dom-event-listener "^1.0.2"
|
||||||
array-tree-filter "^2.1.0"
|
array-tree-filter "^2.1.0"
|
||||||
async-validator "^3.0.3"
|
async-validator "^3.0.3"
|
||||||
@ -6947,6 +6956,11 @@ nanomatch@^1.2.9:
|
|||||||
snapdragon "^0.8.1"
|
snapdragon "^0.8.1"
|
||||||
to-regex "^3.0.1"
|
to-regex "^3.0.1"
|
||||||
|
|
||||||
|
nanopop@^2.1.0:
|
||||||
|
version "2.1.0"
|
||||||
|
resolved "https://registry.npm.taobao.org/nanopop/download/nanopop-2.1.0.tgz#23476513cee2405888afd2e8a4b54066b70b9e60"
|
||||||
|
integrity sha1-I0dlE87iQFiIr9LopLVAZrcLnmA=
|
||||||
|
|
||||||
natural-compare@^1.4.0:
|
natural-compare@^1.4.0:
|
||||||
version "1.4.0"
|
version "1.4.0"
|
||||||
resolved "https://registry.npm.taobao.org/natural-compare/download/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7"
|
resolved "https://registry.npm.taobao.org/natural-compare/download/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user