mirror of
https://github.com/iczer/vue-antd-admin
synced 2025-04-06 04:00:06 +08:00
Compare commits
57 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
10a7613e6e | ||
|
3c79f89416 | ||
|
35f0e431b8 | ||
|
60be8cf4ec | ||
|
17e41ce1fc | ||
|
baae56f715 | ||
|
9a7493c99c | ||
|
cc60ff1a27 | ||
|
c0e5f2ccec | ||
|
3fa8995690 | ||
|
a4d4b28ce3 | ||
|
d7a530db46 | ||
|
7f5fbb7426 | ||
|
6000da4220 | ||
|
c0aec854af | ||
|
8062905b17 | ||
|
56635a948b | ||
|
2d0c6c2a2b | ||
|
5d4bd75b22 | ||
|
6d43e9af42 | ||
|
c2d2c2c686 | ||
|
a9b3da4d19 | ||
|
baf063f9ea | ||
|
816d19f7da | ||
|
ce83564335 | ||
|
1345a02cd0 | ||
|
449fd99f9d | ||
|
97a1417112 | ||
|
ba89880736 | ||
|
aa4e3d93d5 | ||
|
d730f6d783 | ||
|
8d42b936e5 | ||
|
da1dafda54 | ||
|
8127121ab6 | ||
|
d29a14936a | ||
|
80e3ad42bc | ||
|
bb7fa9abb6 | ||
|
990daf2d27 | ||
|
939f8640d3 | ||
|
a1ae7d1e3f | ||
|
8ddc7c167c | ||
|
ce536b95c2 | ||
|
4dbbb852a9 | ||
|
f0d60a8242 | ||
|
9ddd117d5e | ||
|
2fc5b9d594 | ||
|
83c6381a4b | ||
|
0c41878174 | ||
|
a5c34a8514 | ||
|
df076bda24 | ||
|
62b57a97cb | ||
|
867377a6d2 | ||
|
a2e5370ae8 | ||
|
39b64d0704 | ||
|
d2b2631fb1 | ||
|
345b46bf6f | ||
|
b0fc3a943e |
1
.env
1
.env
@ -1,3 +1,4 @@
|
||||
VUE_APP_PUBLIC_PATH=/
|
||||
VUE_APP_NAME=Admin
|
||||
VUE_APP_ROUTES_KEY=admin.routes
|
||||
VUE_APP_PERMISSIONS_KEY=admin.permissions
|
||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -17,3 +17,4 @@ selenium-debug.log
|
||||
*.njsproj
|
||||
*.sln
|
||||
package-lock.json
|
||||
.env.production.local
|
||||
|
@ -16,9 +16,9 @@ Multiple theme modes available:
|
||||

|
||||
</div>
|
||||
|
||||
- Preview:https://iczer.gitee.io/vue-antd-admin
|
||||
- Documentation:https://iczer.gitee.io/vue-antd-admin-docs
|
||||
- FAQ:https://iczer.gitee.io/vue-antd-admin-docs/start/faq.html
|
||||
- Preview:https://vue-antd-admin.pages.dev
|
||||
- Documentation:https://doc.vue-antd-admin.pages.dev
|
||||
- FAQ:https://doc.vue-antd-admin.pages.dev/start/faq.html
|
||||
- Mirror Repo in China:https://gitee.com/iczer/vue-antd-admin
|
||||
|
||||
## Browsers support
|
||||
@ -50,4 +50,4 @@ Any type of contribution is welcome, here are some examples of how you may contr
|
||||
- Use Vue Antd Admin in your daily work.
|
||||
- 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.
|
||||
- Join the community and share your experiences with us. QQ Group: 812277510、610090280(已满)
|
||||
- Join the community and share your experiences with us. QQ Group:942083829、812277510(full)、610090280(full)
|
||||
|
23
README.md
23
README.md
@ -1,3 +1,5 @@
|
||||
|
||||
|
||||
简体中文 | [English](./README.en-US.md)
|
||||
<h1 align="center">Vue Antd Admin</h1>
|
||||
|
||||
@ -6,6 +8,12 @@
|
||||
[Ant Design Pro](https://github.com/ant-design/ant-design-pro) 的 Vue 实现版本
|
||||
开箱即用的中后台前端/设计解决方案
|
||||
|
||||
:star::star::star:
|
||||
vue3 版本现已推出,更名为
|
||||
[stepin-template](https://github.com/stepui/stepin-template),欢迎体验,
|
||||
[立即前往](https://github.com/stepui/stepin-template)
|
||||
--
|
||||
|
||||
[](https://github.com/iczer/vue-antd-admin/blob/master/LICENSE)
|
||||
[](https://david-dm.org/iczer/vue-antd-admin)
|
||||
[](https://david-dm.org/iczer/vue-antd-admin?type=dev)
|
||||
@ -16,9 +24,9 @@
|
||||

|
||||
</div>
|
||||
|
||||
- 预览地址:https://iczer.gitee.io/vue-antd-admin
|
||||
- 使用文档:https://iczer.gitee.io/vue-antd-admin-docs
|
||||
- 常见问题:https://iczer.gitee.io/vue-antd-admin-docs/start/faq.html
|
||||
- 预览地址:https://vue-antd-admin.pages.dev
|
||||
- 使用文档:https://doc.vue-antd-admin.pages.dev
|
||||
- 常见问题:https://doc.vue-antd-admin.pages.dev/start/faq.html
|
||||
- 国内镜像:https://gitee.com/iczer/vue-antd-admin
|
||||
|
||||
## 浏览器支持
|
||||
@ -50,4 +58,11 @@ $ npm run serve
|
||||
- 在你的公司或个人项目中使用 Vue Antd Admin。
|
||||
- 通过 [Issue](https://github.com/iczer/vue-antd-admin/issues) 报告:bug:或进行咨询。
|
||||
- 提交 [Pull Request](https://github.com/iczer/vue-antd-admin/pulls) 改进 Admin 的代码。
|
||||
- 加入社群,与小伙伴们一同交流心得。QQ群:812277510、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>
|
||||
|
@ -1,7 +1,7 @@
|
||||
module.exports = {
|
||||
title: 'Vue Antd Admin',
|
||||
description: 'Vue Antd Admin',
|
||||
base: '/vue-antd-admin-docs/',
|
||||
base: '/',
|
||||
head: [
|
||||
['link', { rel: 'icon', href: '/favicon.ico' }]
|
||||
],
|
||||
|
@ -31,7 +31,7 @@ permission = {
|
||||
operation: ['add', 'delete', 'edit', 'close'] //权限下的操作权限
|
||||
}
|
||||
```
|
||||
你也可以设置 role 的值为字符串,比如 permission = 'form', 它等同于:
|
||||
你也可以设置 permission 的值为字符串,比如 permission = 'form', 它等同于:
|
||||
```js
|
||||
permission = {
|
||||
id: 'form'
|
||||
|
@ -44,7 +44,7 @@ const tokenCheck = {
|
||||
* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。
|
||||
|
||||
### onRejected
|
||||
我们会为 onFulfilled 钩子函数注入 error 和 options 两个参数:
|
||||
我们会为 onRejected 钩子函数注入 error 和 options 两个参数:
|
||||
* `error: Error`: axios 请求错误对象
|
||||
* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。
|
||||
|
||||
@ -128,4 +128,4 @@ export default {
|
||||
response: [resp401, resp403] // 响应拦截
|
||||
}
|
||||
```
|
||||
:::
|
||||
:::
|
||||
|
17
package.json
17
package.json
@ -1,17 +1,18 @@
|
||||
{
|
||||
"name": "vue-antd-admin",
|
||||
"version": "0.7.2",
|
||||
"version": "0.7.4",
|
||||
"homepage": "https://iczer.github.io/vue-antd-admin",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"serve": "vue-cli-service serve",
|
||||
"build": "vue-cli-service build",
|
||||
"serve": "export NODE_OPTIONS=--openssl-legacy-provider && vue-cli-service serve",
|
||||
"build": "export NODE_OPTIONS=--openssl-legacy-provider && vue-cli-service build",
|
||||
"lint": "vue-cli-service lint",
|
||||
"predeploy": "yarn build",
|
||||
"deploy": "gh-pages -d dist -b pages -r https://gitee.com/iczer/vue-antd-admin.git",
|
||||
"docs:dev": "vuepress dev docs",
|
||||
"docs:build": "vuepress build docs",
|
||||
"docs:deploy": "vuepress build docs && gh-pages -d docs/.vuepress/dist -b master -r https://gitee.com/iczer/vue-antd-admin-docs.git"
|
||||
"deploy": "gh-pages -d dist -b pages -r https://github.com/iczer/vue-antd-admin.git",
|
||||
"docs:dev": "export NODE_OPTIONS=--openssl-legacy-provider && vuepress dev docs",
|
||||
"docs:build": "export NODE_OPTIONS=--openssl-legacy-provider && vuepress build docs",
|
||||
"predocs:deploy": "yarn docs:build",
|
||||
"docs:deploy": "gh-pages -d docs/.vuepress/dist -b doc -r https://github.com/iczer/vue-antd-admin.git"
|
||||
},
|
||||
"dependencies": {
|
||||
"@antv/data-set": "^0.11.4",
|
||||
@ -53,7 +54,7 @@
|
||||
"vue-cli-plugin-style-resources-loader": "^0.1.4",
|
||||
"vue-template-compiler": "^2.6.11",
|
||||
"vuepress": "^1.5.2",
|
||||
"webpack-theme-color-replacer": "^1.3.12",
|
||||
"webpack-theme-color-replacer": "1.3.18",
|
||||
"whatwg-fetch": "^3.0.0"
|
||||
},
|
||||
"eslintConfig": {
|
||||
|
@ -43,10 +43,13 @@ export default {
|
||||
'theme.color': function(val) {
|
||||
let closeMessage = this.$message.loading(`您选择了主题色 ${val}, 正在切换...`)
|
||||
themeUtil.changeThemeColor(val, this.theme.mode).then(closeMessage)
|
||||
},
|
||||
'layout': function() {
|
||||
window.dispatchEvent(new Event('resize'))
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
...mapState('setting', ['theme', 'weekMode', 'lang'])
|
||||
...mapState('setting', ['layout', 'theme', 'weekMode', 'lang'])
|
||||
},
|
||||
methods: {
|
||||
...mapMutations('setting', ['setDevice']),
|
||||
|
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 |
@ -38,6 +38,26 @@ import {getI18nKey} from '@/utils/routerUtil'
|
||||
|
||||
const {Item, SubMenu} = Menu
|
||||
|
||||
const resolvePath = (path, params = {}) => {
|
||||
let _path = path
|
||||
Object.entries(params).forEach(([key, value]) => {
|
||||
_path = _path.replace(new RegExp(`:${key}`, 'g'), value)
|
||||
})
|
||||
return _path
|
||||
}
|
||||
|
||||
const toRoutesMap = (routes) => {
|
||||
const map = {}
|
||||
routes.forEach(route => {
|
||||
map[route.fullPath] = route
|
||||
if (route.children && route.children.length > 0) {
|
||||
const childrenMap = toRoutesMap(route.children)
|
||||
Object.assign(map, childrenMap)
|
||||
}
|
||||
})
|
||||
return map
|
||||
}
|
||||
|
||||
export default {
|
||||
name: 'IMenu',
|
||||
props: {
|
||||
@ -73,6 +93,9 @@ export default {
|
||||
computed: {
|
||||
menuTheme() {
|
||||
return this.theme == 'light' ? this.theme : 'dark'
|
||||
},
|
||||
routesMap() {
|
||||
return toRoutesMap(this.options)
|
||||
}
|
||||
},
|
||||
created () {
|
||||
@ -132,7 +155,8 @@ export default {
|
||||
},
|
||||
renderMenuItem: function (h, menu) {
|
||||
let tag = 'router-link'
|
||||
let config = {props: {to: menu.fullPath}, attrs: {style: 'overflow:hidden;white-space:normal;text-overflow:clip;'}}
|
||||
const path = resolvePath(menu.fullPath, menu.meta.params)
|
||||
let config = {props: {to: {path, query: menu.meta.query}, }, attrs: {style: 'overflow:hidden;white-space:normal;text-overflow:clip;'}}
|
||||
if (menu.meta && menu.meta.link) {
|
||||
tag = 'a'
|
||||
config = {attrs: {style: 'overflow:hidden;white-space:normal;text-overflow:clip;', href: menu.meta.link, target: '_blank'}}
|
||||
@ -200,16 +224,23 @@ export default {
|
||||
})
|
||||
},
|
||||
updateMenu () {
|
||||
const matchedRoutes = this.$route.matched.filter(item => item.path !== '')
|
||||
this.selectedKeys = this.getSelectedKey(this.$route)
|
||||
let openKeys = matchedRoutes.map(item => item.path)
|
||||
this.selectedKeys = this.getSelectedKeys()
|
||||
let openKeys = this.selectedKeys.filter(item => item !== '')
|
||||
openKeys = openKeys.slice(0, openKeys.length -1)
|
||||
if (!fastEqual(openKeys, this.sOpenKeys)) {
|
||||
this.collapsed || this.mode === 'horizontal' ? this.cachedOpenKeys = openKeys : this.sOpenKeys = openKeys
|
||||
}
|
||||
},
|
||||
getSelectedKey (route) {
|
||||
return route.matched.map(item => item.path)
|
||||
getSelectedKeys() {
|
||||
let matches = this.$route.matched
|
||||
const route = matches[matches.length - 1]
|
||||
let chose = this.routesMap[route.path]
|
||||
if (chose && chose.meta && chose.meta.highlight) {
|
||||
chose = this.routesMap[chose.meta.highlight]
|
||||
const resolve = this.$router.resolve({path: chose.fullPath})
|
||||
matches = (resolve.resolved && resolve.resolved.matched) || matches
|
||||
}
|
||||
return matches.map(item => item.path)
|
||||
}
|
||||
},
|
||||
render (h) {
|
||||
|
@ -23,7 +23,7 @@
|
||||
:expandedRowKeys="expandedRowKeys"
|
||||
:expandedRowRender="expandedRowRender"
|
||||
@change="onChange"
|
||||
:rowSelection="selectedRows ? {selectedRowKeys: selectedRowKeys, onChange: updateSelect} : undefined"
|
||||
:rowSelection="selectedRows ? {selectedRowKeys, onSelect, onSelectAll} : undefined"
|
||||
>
|
||||
<template slot-scope="text, record, index" :slot="slot" v-for="slot in Object.keys($scopedSlots).filter(key => key !== 'expandedRowRender') ">
|
||||
<slot :name="slot" v-bind="{text, record, index}"></slot>
|
||||
@ -64,22 +64,70 @@ export default {
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
updateSelect (selectedRowKeys, selectedRows) {
|
||||
this.$emit('update:selectedRows', selectedRows)
|
||||
this.$emit('selectedRowChange', selectedRowKeys, selectedRows)
|
||||
equals(record1, record2) {
|
||||
if (record1 === record2) {
|
||||
return true
|
||||
}
|
||||
const {rowKey} = this
|
||||
if (rowKey && typeof rowKey === 'string') {
|
||||
return record1[rowKey] === record2[rowKey]
|
||||
} else if (rowKey && typeof rowKey === 'function') {
|
||||
return rowKey(record1) === rowKey(record2)
|
||||
}
|
||||
return false
|
||||
},
|
||||
contains(arr, item) {
|
||||
if (!arr || arr.length === 0) {
|
||||
return false
|
||||
}
|
||||
const {equals} = this
|
||||
for (let i = 0; i < arr.length; i++) {
|
||||
if (equals(arr[i], item)) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
},
|
||||
onSelectAll(selected, rows) {
|
||||
const {getKey, contains} = this
|
||||
const unselected = this.dataSource.filter(item => !contains(rows, item, this.rowKey))
|
||||
const _selectedRows = this.selectedRows.filter(item => !contains(unselected, item, this.rowKey))
|
||||
const set = {}
|
||||
_selectedRows.forEach(item => set[getKey(item)] = item)
|
||||
rows.forEach(item => set[getKey(item)] = item)
|
||||
const _rows = Object.values(set)
|
||||
this.$emit('update:selectedRows', _rows)
|
||||
this.$emit('selectedRowChange', _rows.map(item => getKey(item)), _rows)
|
||||
},
|
||||
getKey(record) {
|
||||
const {rowKey} = this
|
||||
if (!rowKey || !record) {
|
||||
return undefined
|
||||
}
|
||||
if (typeof rowKey === 'string') {
|
||||
return record[rowKey]
|
||||
} else {
|
||||
return rowKey(record)
|
||||
}
|
||||
},
|
||||
onSelect(record, selected) {
|
||||
const {equals, selectedRows, getKey} = this
|
||||
const _selectedRows = selected ? [...selectedRows, record] : selectedRows.filter(row => !equals(row, record))
|
||||
this.$emit('update:selectedRows', _selectedRows)
|
||||
this.$emit('selectedRowChange', _selectedRows.map(item => getKey(item)), _selectedRows)
|
||||
},
|
||||
initTotalList (columns) {
|
||||
const totalList = columns.filter(item => item.needTotal)
|
||||
return columns.filter(item => item.needTotal)
|
||||
.map(item => {
|
||||
return {
|
||||
...item,
|
||||
total: 0
|
||||
}
|
||||
})
|
||||
return totalList
|
||||
},
|
||||
onClear() {
|
||||
this.updateSelect([], [])
|
||||
this.$emit('update:selectedRows', [])
|
||||
this.$emit('selectedRowChange', [], [])
|
||||
this.$emit('clear')
|
||||
},
|
||||
onChange(pagination, filters, sorter, {currentDataSource}) {
|
||||
@ -110,10 +158,8 @@ export default {
|
||||
},
|
||||
computed: {
|
||||
selectedRowKeys() {
|
||||
return this.selectedRows.map(record => {
|
||||
return (typeof this.rowKey === 'function') ? this.rowKey(record) : record[this.rowKey]
|
||||
})
|
||||
}
|
||||
return this.selectedRows.map(record => this.getKey(record))
|
||||
},
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
@ -76,7 +76,7 @@
|
||||
indentSize: Number,
|
||||
loading: Boolean,
|
||||
locale: Object,
|
||||
pagination: Object,
|
||||
pagination: [Object, Boolean],
|
||||
rowClassName: Function,
|
||||
rowKey: [String, Function],
|
||||
rowSelection: Object,
|
||||
|
@ -22,10 +22,11 @@ const ANTD = {
|
||||
'component-background': '#fff',
|
||||
'heading-color': 'rgba(0, 0, 0, 0.85)',
|
||||
'text-color': 'rgba(0, 0, 0, 0.65)',
|
||||
'text-color-inverse': '#fff',
|
||||
'text-color-inverse': '#fefefe',
|
||||
'text-color-secondary': 'rgba(0, 0, 0, 0.45)',
|
||||
'shadow-color': 'rgba(0, 0, 0, 0.15)',
|
||||
'border-color-split': '#f0f0f0',
|
||||
'border-color-base': '#d9d9d9',
|
||||
'background-color-light': '#fafafa',
|
||||
'background-color-base': '#f5f5f5',
|
||||
'table-selected-row-bg': '#fafafa',
|
||||
@ -34,8 +35,9 @@ const ANTD = {
|
||||
'disabled-color': 'rgba(0, 0, 0, 0.25)',
|
||||
'menu-dark-color': 'rgba(254, 254, 254, 0.65)',
|
||||
'menu-dark-highlight-color': '#fefefe',
|
||||
'menu-dark-selected-item-icon-color': '#fefefe',
|
||||
'menu-dark-arrow-color': '#fefefe',
|
||||
'btn-primary-color': '#fff',
|
||||
'btn-primary-color': '#fefefe',
|
||||
},
|
||||
light: {
|
||||
'layout-body-background': '#f0f2f5',
|
||||
@ -43,10 +45,11 @@ const ANTD = {
|
||||
'component-background': '#fff',
|
||||
'heading-color': 'rgba(0, 0, 0, 0.85)',
|
||||
'text-color': 'rgba(0, 0, 0, 0.65)',
|
||||
'text-color-inverse': '#fff',
|
||||
'text-color-inverse': '#fefefe',
|
||||
'text-color-secondary': 'rgba(0, 0, 0, 0.45)',
|
||||
'shadow-color': 'rgba(0, 0, 0, 0.15)',
|
||||
'border-color-split': '#f0f0f0',
|
||||
'border-color-base': '#d9d9d9',
|
||||
'background-color-light': '#fafafa',
|
||||
'background-color-base': '#f5f5f5',
|
||||
'table-selected-row-bg': '#fafafa',
|
||||
@ -55,8 +58,9 @@ const ANTD = {
|
||||
'disabled-color': 'rgba(0, 0, 0, 0.25)',
|
||||
'menu-dark-color': 'rgba(1, 1, 1, 0.65)',
|
||||
'menu-dark-highlight-color': '#fefefe',
|
||||
'menu-dark-selected-item-icon-color': '#fefefe',
|
||||
'menu-dark-arrow-color': '#fefefe',
|
||||
'btn-primary-color': '#fff',
|
||||
'btn-primary-color': '#fefefe',
|
||||
},
|
||||
night: {
|
||||
'layout-body-background': '#000',
|
||||
@ -64,10 +68,11 @@ const ANTD = {
|
||||
'component-background': '#141414',
|
||||
'heading-color': 'rgba(255, 255, 255, 0.85)',
|
||||
'text-color': 'rgba(255, 255, 255, 0.85)',
|
||||
'text-color-inverse': '#141414',
|
||||
'text-color-inverse': '#fefefe',
|
||||
'text-color-secondary': 'rgba(255, 255, 255, 0.45)',
|
||||
'shadow-color': 'rgba(255, 255, 255, 0.15)',
|
||||
'border-color-split': '#303030',
|
||||
'border-color-base': '#282828',
|
||||
'background-color-light': '#ffffff0a',
|
||||
'background-color-base': '#2a2a2a',
|
||||
'table-selected-row-bg': '#ffffff0a',
|
||||
@ -76,8 +81,9 @@ const ANTD = {
|
||||
'disabled-color': 'rgba(255, 255, 255, 0.25)',
|
||||
'menu-dark-color': 'rgba(254, 254, 254, 0.65)',
|
||||
'menu-dark-highlight-color': '#fefefe',
|
||||
'menu-dark-selected-item-icon-color': '#fefefe',
|
||||
'menu-dark-arrow-color': '#fefefe',
|
||||
'btn-primary-color': '#141414',
|
||||
'btn-primary-color': '#fefefe',
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -90,8 +90,10 @@ export default {
|
||||
...mapMutations('setting', ['correctPageMinHeight']),
|
||||
getRouteBreadcrumb() {
|
||||
let routes = this.$route.matched
|
||||
const path = this.$route.path
|
||||
let breadcrumb = []
|
||||
routes.forEach(route => {
|
||||
routes.filter(item => path.includes(item.path) || item.regex.test(path))
|
||||
.forEach(route => {
|
||||
const path = route.path.length === 0 ? '/home' : route.path
|
||||
breadcrumb.push(this.$t(getI18nKey(path)))
|
||||
})
|
||||
|
@ -25,13 +25,13 @@ export default {
|
||||
computed: {
|
||||
...mapState('setting', ['isMobile', 'multiPage', 'animate']),
|
||||
desc() {
|
||||
return this.page.desc
|
||||
return this.page?.desc
|
||||
},
|
||||
linkList() {
|
||||
return this.page.linkList
|
||||
return this.page?.linkList
|
||||
},
|
||||
extraImage() {
|
||||
return this.page.extraImage
|
||||
return this.page?.extraImage
|
||||
}
|
||||
},
|
||||
mounted () {
|
||||
|
@ -26,6 +26,9 @@ export default {
|
||||
.copyright{
|
||||
color: @text-color-second;
|
||||
font-size: 14px;
|
||||
i {
|
||||
margin: 0 4px;
|
||||
}
|
||||
}
|
||||
.links{
|
||||
margin-bottom: 8px;
|
||||
|
@ -56,9 +56,10 @@ export default {
|
||||
this.loading = false
|
||||
return
|
||||
}
|
||||
this.loadding = true
|
||||
if (this.show) return
|
||||
this.loading = true
|
||||
setTimeout(() => {
|
||||
this.loadding = false
|
||||
this.loading = false
|
||||
}, 1000)
|
||||
}
|
||||
}
|
||||
|
@ -14,11 +14,11 @@
|
||||
:type="fixedTabs ? 'lock' : 'unlock'"
|
||||
/>
|
||||
</a-tooltip>
|
||||
<a-tab-pane v-for="page in pageList" :key="page.fullPath">
|
||||
<div slot="tab" class="tab" @contextmenu="e => onContextmenu(page.fullPath, e)">
|
||||
<a-icon @click="onRefresh(page)" :class="['icon-sync', {'hide': page.fullPath !== active && !page.loading}]" :type="page.loading ? 'loading' : 'sync'" />
|
||||
<div class="title" @click="onTabClick(page.fullPath)" >{{pageName(page)}}</div>
|
||||
<a-icon v-if="!page.unclose" @click="onClose(page.fullPath)" class="icon-close" type="close"/>
|
||||
<a-tab-pane v-for="page in pageList" :key="page.path">
|
||||
<div slot="tab" class="tab" @contextmenu="e => onContextmenu(page.path, e)">
|
||||
<a-icon @click="onRefresh(page)" :class="['icon-sync', {'hide': page.path !== active && !page.loading}]" :type="page.loading ? 'loading' : 'sync'" />
|
||||
<div class="title" @click="onTabClick(page.path)" >{{pageName(page)}}</div>
|
||||
<a-icon v-if="!page.unclose" @click="onClose(page.path)" class="icon-close" type="close"/>
|
||||
</div>
|
||||
</a-tab-pane>
|
||||
</a-tabs>
|
||||
@ -89,13 +89,13 @@
|
||||
this.$emit('close', key)
|
||||
},
|
||||
onRefresh(page) {
|
||||
this.$emit('refresh', page.fullPath, page)
|
||||
this.$emit('refresh', page.path, page)
|
||||
},
|
||||
onContextmenu(pageKey, e) {
|
||||
this.$emit('contextmenu', pageKey, e)
|
||||
},
|
||||
pageName(page) {
|
||||
const custom = this.customTitles.find(item => item.path === page.fullPath)
|
||||
const custom = this.customTitles.find(item => item.path === page.path)
|
||||
return (custom && custom.title) || page.title || this.$t(getI18nKey(page.keyPath))
|
||||
}
|
||||
}
|
||||
@ -184,4 +184,4 @@
|
||||
.virtual-tabs{
|
||||
height: 48px;
|
||||
}
|
||||
</style>
|
||||
</style>
|
||||
|
@ -62,7 +62,7 @@ export default {
|
||||
this.loadCacheConfig(this.$router?.options?.routes)
|
||||
this.loadCachedTabs()
|
||||
const route = this.$route
|
||||
if (this.pageList.findIndex(item => item.fullPath === route.fullPath) === -1) {
|
||||
if (this.pageList.findIndex(item => item.path === route.fullPath) === -1) {
|
||||
this.pageList.push(this.createPage(route))
|
||||
}
|
||||
this.activePage = route.fullPath
|
||||
@ -87,9 +87,12 @@ export default {
|
||||
},
|
||||
'$route': function (newRoute) {
|
||||
this.activePage = newRoute.fullPath
|
||||
const page = this.pageList.find(item => item.path === newRoute.fullPath)
|
||||
if (!this.multiPage) {
|
||||
this.pageList = [this.createPage(newRoute)]
|
||||
} else if (this.pageList.findIndex(item => item.fullPath === newRoute.fullPath) === -1) {
|
||||
} else if (page) {
|
||||
page.fullPath = newRoute.fullPath
|
||||
} else if (!page) {
|
||||
this.pageList.push(this.createPage(newRoute))
|
||||
}
|
||||
if (this.multiPage) {
|
||||
@ -113,25 +116,26 @@ export default {
|
||||
methods: {
|
||||
changePage (key) {
|
||||
this.activePage = key
|
||||
this.$router.push(key)
|
||||
const page = this.pageList.find(item => item.path === key)
|
||||
this.$router.push(page.fullPath)
|
||||
},
|
||||
remove (key, next) {
|
||||
if (this.pageList.length === 1) {
|
||||
return this.$message.warning(this.$t('warn'))
|
||||
}
|
||||
//清除缓存
|
||||
let index = this.pageList.findIndex(item => item.fullPath === key)
|
||||
let index = this.pageList.findIndex(item => item.path === key)
|
||||
this.clearCaches = this.pageList.splice(index, 1).map(page => page.cachedKey)
|
||||
if (next) {
|
||||
this.$router.push(next)
|
||||
} else if (key === this.activePage) {
|
||||
index = index >= this.pageList.length ? this.pageList.length - 1 : index
|
||||
this.activePage = this.pageList[index].fullPath
|
||||
this.activePage = this.pageList[index].path
|
||||
this.$router.push(this.activePage)
|
||||
}
|
||||
},
|
||||
refresh (key, page) {
|
||||
page = page || this.pageList.find(item => item.fullPath === key)
|
||||
page = page || this.pageList.find(item => item.path === key)
|
||||
page.loading = true
|
||||
this.clearCache(page)
|
||||
if (key === this.activePage) {
|
||||
@ -159,7 +163,7 @@ export default {
|
||||
},
|
||||
closeOthers (pageKey) {
|
||||
// 清除缓存
|
||||
const clearPages = this.pageList.filter(item => item.fullPath !== pageKey && !item.unclose)
|
||||
const clearPages = this.pageList.filter(item => item.path !== pageKey && !item.unclose)
|
||||
this.clearCaches = clearPages.map(item => item.cachedKey)
|
||||
this.pageList = this.pageList.filter(item => !clearPages.includes(item))
|
||||
// 判断跳转
|
||||
@ -169,25 +173,25 @@ export default {
|
||||
}
|
||||
},
|
||||
closeLeft (pageKey) {
|
||||
const index = this.pageList.findIndex(item => item.fullPath === pageKey)
|
||||
const index = this.pageList.findIndex(item => item.path === pageKey)
|
||||
// 清除缓存
|
||||
const clearPages = this.pageList.filter((item, i) => i < index && !item.unclose)
|
||||
this.clearCaches = clearPages.map(item => item.cachedKey)
|
||||
this.pageList = this.pageList.filter(item => !clearPages.includes(item))
|
||||
// 判断跳转
|
||||
if (!this.pageList.find(item => item.fullPath === this.activePage)) {
|
||||
if (!this.pageList.find(item => item.path === this.activePage)) {
|
||||
this.activePage = pageKey
|
||||
this.$router.push(this.activePage)
|
||||
}
|
||||
},
|
||||
closeRight (pageKey) {
|
||||
// 清除缓存
|
||||
const index = this.pageList.findIndex(item => item.fullPath === pageKey)
|
||||
const index = this.pageList.findIndex(item => item.path === pageKey)
|
||||
const clearPages = this.pageList.filter((item, i) => i > index && !item.unclose)
|
||||
this.clearCaches = clearPages.map(item => item.cachedKey)
|
||||
this.pageList = this.pageList.filter(item => !clearPages.includes(item))
|
||||
// 判断跳转
|
||||
if (!this.pageList.find(item => item.fullPath === this.activePage)) {
|
||||
if (!this.pageList.find(item => item.path === this.activePage)) {
|
||||
this.activePage = pageKey
|
||||
this.$router.push(this.activePage)
|
||||
}
|
||||
@ -234,7 +238,8 @@ export default {
|
||||
closePageListener(event) {
|
||||
const {closeRoute, nextRoute} = event.detail
|
||||
const closePath = typeof closeRoute === 'string' ? closeRoute : closeRoute.path
|
||||
this.remove(closePath, nextRoute)
|
||||
const path = closePath && closePath.split('?')[0]
|
||||
this.remove(path, nextRoute)
|
||||
},
|
||||
/**
|
||||
* 页面刷新事件监听
|
||||
@ -242,7 +247,8 @@ export default {
|
||||
*/
|
||||
refreshPageListener(event) {
|
||||
const {pageKey} = event.detail
|
||||
this.refresh(pageKey)
|
||||
const path = pageKey && pageKey.split('?')[0]
|
||||
this.refresh(path)
|
||||
},
|
||||
/**
|
||||
* 页面 unload 事件监听器,添加页签到 session 缓存,用于刷新时保留页签
|
||||
@ -255,6 +261,7 @@ export default {
|
||||
return {
|
||||
keyPath: route.matched[route.matched.length - 1].path,
|
||||
fullPath: route.fullPath, loading: false,
|
||||
path: route.fullPath,
|
||||
title: route.meta && route.meta.page && route.meta.page.title,
|
||||
unclose: route.meta && route.meta.page && (route.meta.page.closable === false),
|
||||
}
|
||||
@ -264,7 +271,7 @@ export default {
|
||||
* @param route 页面对应的路由
|
||||
*/
|
||||
setCachedKey(route) {
|
||||
const page = this.pageList.find(item => item.fullPath === route.fullPath)
|
||||
const page = this.pageList.find(item => item.path === route.fullPath)
|
||||
page.unclose = route.meta && route.meta.page && (route.meta.page.closable === false)
|
||||
if (!page._init_) {
|
||||
const vnode = this.$refs.tabContent.$vnode
|
||||
@ -294,7 +301,7 @@ export default {
|
||||
routes.forEach(item => {
|
||||
const cacheAble = item.meta?.page?.cacheAble ?? pCache ?? true
|
||||
if (!cacheAble) {
|
||||
this.excludeKeys.push(new RegExp(`${item.fullPath}\\d+$`))
|
||||
this.excludeKeys.push(new RegExp(`${item.path.replace(/:[^/]*/g, '[^/]*')}(\\?.*)?\\d*$`))
|
||||
}
|
||||
if (item.children) {
|
||||
this.loadCacheConfig(item.children, cacheAble)
|
||||
|
@ -10,6 +10,7 @@ import 'animate.css/source/animate.css'
|
||||
import Plugins from '@/plugins'
|
||||
import {initI18n} from '@/utils/i18n'
|
||||
import bootstrap from '@/bootstrap'
|
||||
import 'moment/locale/zh-cn'
|
||||
|
||||
const router = initRouter(store.state.setting.asyncRoutes)
|
||||
const i18n = initI18n('CN', 'US')
|
||||
|
@ -5,6 +5,7 @@ import '@/mock/user/login'
|
||||
import '@/mock/workplace'
|
||||
import '@/mock/user/routes'
|
||||
import '@/mock/goods'
|
||||
import '@/mock/list'
|
||||
|
||||
// 设置全局延时
|
||||
Mock.setup({
|
||||
|
52
src/mock/list/index.js
Normal file
52
src/mock/list/index.js
Normal file
@ -0,0 +1,52 @@
|
||||
import Mock from 'mockjs'
|
||||
import '@/mock/extend'
|
||||
import {parseUrlParams} from '@/utils/request'
|
||||
|
||||
const current = new Date().getTime()
|
||||
|
||||
const source = Mock.mock({
|
||||
'list|100': [{
|
||||
'key|+1': 0,
|
||||
'no': `${current}-@integer(1,100)`,
|
||||
'description': '这是一段描述',
|
||||
'callNo|0-50': 5,
|
||||
'status|1-4': 1,
|
||||
'updatedAt': '@DATETIME',
|
||||
}]
|
||||
})
|
||||
|
||||
Mock.mock(RegExp(`${process.env.VUE_APP_API_BASE_URL}/list` + '.*'),'get', ({url}) => {
|
||||
const params = parseUrlParams(decodeURI(url))
|
||||
let {page, pageSize} = params
|
||||
page = eval(page) - 1 || 0
|
||||
pageSize = eval(pageSize) || 10
|
||||
|
||||
delete params.page
|
||||
delete params.pageSize
|
||||
|
||||
let result = source.list.filter(item => {
|
||||
for (let [key, value] of Object.entries(params)) {
|
||||
if (item[key] !== value) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
})
|
||||
const total = result.length
|
||||
if ((page) * pageSize > total) {
|
||||
result = []
|
||||
} else {
|
||||
result = result.slice(page * pageSize, (page + 1) * pageSize)
|
||||
}
|
||||
|
||||
return {
|
||||
code: 0,
|
||||
message: 'success',
|
||||
data: {
|
||||
page: page + 1,
|
||||
pageSize,
|
||||
total: 100,
|
||||
list: result
|
||||
}
|
||||
}
|
||||
})
|
@ -8,21 +8,32 @@ const user = Mock.mock({
|
||||
position: '@POSITION'
|
||||
})
|
||||
Mock.mock(`${process.env.VUE_APP_API_BASE_URL}/login`, 'post', ({body}) => {
|
||||
let result = {}
|
||||
let result = {data: {}}
|
||||
const {name, password} = JSON.parse(body)
|
||||
|
||||
if (name !== 'admin' || password !== '888888') {
|
||||
result.code = -1
|
||||
result.message = '账户名或密码错误(admin/888888)'
|
||||
let success = false
|
||||
|
||||
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 {
|
||||
success = false
|
||||
}
|
||||
|
||||
if (success) {
|
||||
result.code = 0
|
||||
result.message = Mock.mock('@TIMEFIX').CN + ',欢迎回来'
|
||||
result.data = {}
|
||||
result.data.user = user
|
||||
result.data.token = 'Authorization:' + Math.random()
|
||||
result.data.expireAt = new Date(new Date().getTime() + 30 * 60 * 1000)
|
||||
result.data.permissions = [{id: 'queryForm', operation: ['add', 'edit']}]
|
||||
result.data.roles = [{id: 'admin', operation: ['add', 'edit', 'delete']}]
|
||||
} else {
|
||||
result.code = -1
|
||||
result.message = '账户名或密码错误(admin/888888 or test/888888)'
|
||||
}
|
||||
return result
|
||||
})
|
||||
|
16
src/pages/Demo.vue
Normal file
16
src/pages/Demo.vue
Normal file
@ -0,0 +1,16 @@
|
||||
<template>
|
||||
<div>
|
||||
<p>query: {{$route.query}}</p>
|
||||
<p>params: {{$route.params}}</p>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
name: 'Demo'
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
@ -79,7 +79,7 @@
|
||||
</a-form>
|
||||
</div>
|
||||
<div>
|
||||
<div class="operator">
|
||||
<a-space class="operator">
|
||||
<a-button @click="addNew" type="primary">新建</a-button>
|
||||
<a-button >批量操作</a-button>
|
||||
<a-dropdown>
|
||||
@ -91,13 +91,14 @@
|
||||
更多操作 <a-icon type="down" />
|
||||
</a-button>
|
||||
</a-dropdown>
|
||||
</div>
|
||||
</a-space>
|
||||
<standard-table
|
||||
:columns="columns"
|
||||
:dataSource="dataSource"
|
||||
:selectedRows.sync="selectedRows"
|
||||
@clear="onClear"
|
||||
@change="onChange"
|
||||
:pagination="{...pagination, onChange: onPageChange}"
|
||||
@selectedRowChange="onSelectChange"
|
||||
>
|
||||
<div slot="description" slot-scope="{text}">
|
||||
@ -116,6 +117,7 @@
|
||||
<a @click="deleteRecord(record.key)" v-auth="`delete`">
|
||||
<a-icon type="delete" />删除2
|
||||
</a>
|
||||
<router-link :to="`/list/query/detail/${record.key}`" >详情</router-link>
|
||||
</div>
|
||||
<template slot="statusTitle">
|
||||
<a-icon @click.native="onStatusTitleClick" type="info-circle" />
|
||||
@ -127,6 +129,7 @@
|
||||
|
||||
<script>
|
||||
import StandardTable from '@/components/table/StandardTable'
|
||||
import {request} from '@/utils/request'
|
||||
const columns = [
|
||||
{
|
||||
title: '规则编号',
|
||||
@ -160,19 +163,6 @@ const columns = [
|
||||
}
|
||||
]
|
||||
|
||||
const dataSource = []
|
||||
|
||||
for (let i = 0; i < 100; i++) {
|
||||
dataSource.push({
|
||||
key: i,
|
||||
no: 'NO ' + i,
|
||||
description: '这是一段描述',
|
||||
callNo: Math.floor(Math.random() * 1000),
|
||||
status: Math.floor(Math.random() * 10) % 4,
|
||||
updatedAt: '2018-07-26'
|
||||
})
|
||||
}
|
||||
|
||||
export default {
|
||||
name: 'QueryList',
|
||||
components: {StandardTable},
|
||||
@ -180,14 +170,37 @@ export default {
|
||||
return {
|
||||
advanced: true,
|
||||
columns: columns,
|
||||
dataSource: dataSource,
|
||||
selectedRows: []
|
||||
dataSource: [],
|
||||
selectedRows: [],
|
||||
pagination: {
|
||||
current: 1,
|
||||
pageSize: 10,
|
||||
total: 0
|
||||
}
|
||||
}
|
||||
},
|
||||
authorize: {
|
||||
deleteRecord: 'delete'
|
||||
},
|
||||
mounted() {
|
||||
this.getData()
|
||||
},
|
||||
methods: {
|
||||
onPageChange(page, pageSize) {
|
||||
this.pagination.current = page
|
||||
this.pagination.pageSize = pageSize
|
||||
this.getData()
|
||||
},
|
||||
getData() {
|
||||
request(process.env.VUE_APP_API_BASE_URL + '/list', 'get', {page: this.pagination.current,
|
||||
pageSize: this.pagination.pageSize}).then(res => {
|
||||
const {list, page, pageSize, total} = res?.data?.data ?? {}
|
||||
this.dataSource = list
|
||||
this.pagination.current = page
|
||||
this.pagination.pageSize = pageSize
|
||||
this.pagination.total = total
|
||||
})
|
||||
},
|
||||
deleteRecord(key) {
|
||||
this.dataSource = this.dataSource.filter(item => item.key !== key)
|
||||
this.selectedRows = this.selectedRows.filter(item => item.key !== key)
|
||||
|
@ -39,9 +39,9 @@ const auth = function(authConfig, permission, role, permissions, roles) {
|
||||
if (type === 'permission') {
|
||||
return checkFromPermission(check, permission)
|
||||
} else if (type === 'role') {
|
||||
return checkFromRoles(check, role)
|
||||
return checkFromRoles(check, roles)
|
||||
} else {
|
||||
return checkFromPermission(check, permission) || checkFromRoles(check, role)
|
||||
return checkFromPermission(check, permission) || checkFromRoles(check, roles)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -17,8 +17,9 @@ const TabsPagePlugin = {
|
||||
},
|
||||
$setPageTitle(route, title) {
|
||||
if (title) {
|
||||
let path = typeof route === 'object' ? route.path : route
|
||||
path = path && path.split('?')[0]
|
||||
// let path = typeof route === 'object' ? route.path : route
|
||||
// path = path && path.split('?')[0]
|
||||
let path = typeof route === 'object' ? this.$router.resolve(route).route.fullPath : route
|
||||
this.$store.commit('setting/setCustomTitle', {path, title})
|
||||
}
|
||||
}
|
||||
@ -26,8 +27,8 @@ const TabsPagePlugin = {
|
||||
computed: {
|
||||
customTitle() {
|
||||
const customTitles = this.$store.state.setting.customTitles
|
||||
const path = this.$route.path.split('?')[0]
|
||||
const custom = customTitles.find(item => item.path === path)
|
||||
// const path = this.$route.path.split('?')[0]
|
||||
const custom = customTitles.find(item => item.path === this.$route.fullPath)
|
||||
return custom && custom.title
|
||||
}
|
||||
}
|
||||
|
@ -95,6 +95,15 @@ const options = {
|
||||
},
|
||||
component: () => import('@/pages/list/QueryList'),
|
||||
},
|
||||
{
|
||||
path: 'query/detail/:id',
|
||||
name: '查询详情',
|
||||
meta: {
|
||||
highlight: '/list/query',
|
||||
invisible: true
|
||||
},
|
||||
component: () => import('@/pages/Demo')
|
||||
},
|
||||
{
|
||||
path: 'primary',
|
||||
name: '标准列表',
|
||||
@ -230,6 +239,28 @@ const options = {
|
||||
},
|
||||
component: () => import('@/pages/form/basic')
|
||||
},
|
||||
{
|
||||
name: '带参菜单',
|
||||
path: 'router/query',
|
||||
meta: {
|
||||
icon: 'project',
|
||||
query: {
|
||||
name: '菜单默认参数'
|
||||
}
|
||||
},
|
||||
component: () => import('@/pages/Demo')
|
||||
},
|
||||
{
|
||||
name: '动态路由菜单',
|
||||
path: 'router/dynamic/:id',
|
||||
meta: {
|
||||
icon: 'project',
|
||||
params: {
|
||||
id: 123
|
||||
}
|
||||
},
|
||||
component: () => import('@/pages/Demo')
|
||||
},
|
||||
{
|
||||
name: 'Ant Design Vue',
|
||||
path: 'antdv',
|
||||
|
@ -3,6 +3,7 @@ import {ADMIN} from '@/config/default'
|
||||
import {formatFullPath} from '@/utils/i18n'
|
||||
import {filterMenu} from '@/utils/authority-utils'
|
||||
import {getLocalSetting} from '@/utils/themeUtil'
|
||||
import deepClone from 'lodash.clonedeep'
|
||||
|
||||
const localSetting = getLocalSetting(true)
|
||||
const customTitlesStr = sessionStorage.getItem(process.env.VUE_APP_TBAS_TITLES_KEY)
|
||||
@ -25,12 +26,12 @@ export default {
|
||||
menuData(state, getters, rootState) {
|
||||
if (state.filterMenu) {
|
||||
const {permissions, roles} = rootState.account
|
||||
filterMenu(state.menuData, permissions, roles)
|
||||
return filterMenu(deepClone(state.menuData), permissions, roles)
|
||||
}
|
||||
return state.menuData
|
||||
},
|
||||
firstMenu(state) {
|
||||
const {menuData} = state
|
||||
firstMenu(state, getters) {
|
||||
const {menuData} = getters
|
||||
if (menuData.length > 0 && !menuData[0].fullPath) {
|
||||
formatFullPath(menuData)
|
||||
}
|
||||
|
@ -8,10 +8,12 @@ function hasPermission(authority, permissions) {
|
||||
let required = '*'
|
||||
if (typeof authority === 'string') {
|
||||
required = authority
|
||||
} else if (Array.isArray(authority)) {
|
||||
required = authority
|
||||
} else if (typeof authority === 'object') {
|
||||
required = authority.permission
|
||||
}
|
||||
return required === '*' || (permissions && permissions.findIndex(item => item === required || item.id === required) !== -1)
|
||||
return required === '*' || hasAnyItem(required, permissions, (r, t) => !!(r === t || r === t.id))
|
||||
}
|
||||
|
||||
/**
|
||||
@ -24,25 +26,23 @@ function hasRole(authority, roles) {
|
||||
if (typeof authority === 'object') {
|
||||
required = authority.role
|
||||
}
|
||||
return authority === '*' || hasAnyRole(required, roles)
|
||||
return authority === '*' || hasAnyItem(required, roles, (r, t) => !!(r === t || r === t.id))
|
||||
}
|
||||
|
||||
/**
|
||||
* 判断是否有需要的任意一个角色
|
||||
* @param required {String | Array[String]} 需要的角色,可以是单个角色或者一个角色数组
|
||||
* @param roles 拥有的角色
|
||||
* 判断目标数组是否有所需元素
|
||||
* @param {String | String[]}required 所需元素,数组或单个元素
|
||||
* @param {String[]|Object[]} source 目标数组
|
||||
* @param {Function} filter 匹配条件
|
||||
* (r: String, s: String|Object) => boolean
|
||||
* @returns {boolean}
|
||||
*/
|
||||
function hasAnyRole(required, roles) {
|
||||
function hasAnyItem(required, source, filter) {
|
||||
if (!required) {
|
||||
return false
|
||||
} else if(Array.isArray(required)) {
|
||||
return roles.findIndex(role => {
|
||||
return required.findIndex(item => item === role || item === role.id) !== -1
|
||||
}) !== -1
|
||||
} else {
|
||||
return roles.findIndex(role => role === required || role.id === required) !== -1
|
||||
}
|
||||
let checkedList = Array.isArray(required) ? required : [required]
|
||||
return !!source.find(s => checkedList.find(r => filter(r, s)))
|
||||
}
|
||||
|
||||
/**
|
||||
@ -69,15 +69,16 @@ function hasAuthority(route, permissions, roles) {
|
||||
* @param roles
|
||||
*/
|
||||
function filterMenu(menuData, permissions, roles) {
|
||||
menuData.forEach(menu => {
|
||||
return menuData.filter(menu => {
|
||||
if (menu.meta && menu.meta.invisible === undefined) {
|
||||
if (!hasAuthority(menu, permissions, roles)) {
|
||||
menu.meta.invisible = true
|
||||
}
|
||||
if (menu.children && menu.children.length > 0) {
|
||||
filterMenu(menu.children, permissions, roles)
|
||||
return false
|
||||
}
|
||||
}
|
||||
if (menu.children && menu.children.length > 0) {
|
||||
menu.children = filterMenu(menu.children, permissions, roles)
|
||||
}
|
||||
return true
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -30,14 +30,14 @@ const METHOD = {
|
||||
* @param params 请求参数
|
||||
* @returns {Promise<AxiosResponse<T>>}
|
||||
*/
|
||||
async function request(url, method, params) {
|
||||
async function request(url, method, params, config) {
|
||||
switch (method) {
|
||||
case METHOD.GET:
|
||||
return axios.get(url, {params})
|
||||
return axios.get(url, {params, ...config})
|
||||
case METHOD.POST:
|
||||
return axios.post(url, params)
|
||||
return axios.post(url, params, config)
|
||||
default:
|
||||
return axios.get(url, {params})
|
||||
return axios.get(url, {params, ...config})
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -32,9 +32,9 @@ function parseRoutes(routesConfig, routerMap) {
|
||||
routesConfig.forEach(item => {
|
||||
// 获取注册在 routerMap 中的 router,初始化 routeCfg
|
||||
let router = undefined, routeCfg = {}
|
||||
if (typeof item === 'string' && routerMap[item]) {
|
||||
if (typeof item === 'string') {
|
||||
router = routerMap[item]
|
||||
routeCfg = {path: router.path || item, router: item}
|
||||
routeCfg = {path: (router && router.path) || item, router: item}
|
||||
} else if (typeof item === 'object') {
|
||||
router = routerMap[item.router]
|
||||
routeCfg = item
|
||||
@ -44,17 +44,39 @@ function parseRoutes(routesConfig, routerMap) {
|
||||
router = typeof item === 'string' ? {path: item, name: item} : item
|
||||
}
|
||||
// 从 router 和 routeCfg 解析路由
|
||||
const meta = {
|
||||
authority: router.authority,
|
||||
icon: router.icon,
|
||||
page: router.page,
|
||||
link: router.link,
|
||||
params: router.params,
|
||||
query: router.query,
|
||||
...router.meta
|
||||
}
|
||||
const cfgMeta = {
|
||||
authority: routeCfg.authority,
|
||||
icon: routeCfg.icon,
|
||||
page: routeCfg.page,
|
||||
link: routeCfg.link,
|
||||
params: routeCfg.params,
|
||||
query: routeCfg.query,
|
||||
...routeCfg.meta
|
||||
}
|
||||
Object.keys(cfgMeta).forEach(key => {
|
||||
if (cfgMeta[key] === undefined || cfgMeta[key] === null || cfgMeta[key] === '') {
|
||||
delete cfgMeta[key]
|
||||
}
|
||||
})
|
||||
Object.assign(meta, cfgMeta)
|
||||
const route = {
|
||||
path: routeCfg.path || router.path || routeCfg.router,
|
||||
name: routeCfg.name || router.name,
|
||||
component: router.component,
|
||||
redirect: routeCfg.redirect || router.redirect,
|
||||
meta: {
|
||||
authority: routeCfg.authority || router.authority || routeCfg.meta?.authority || router.meta?.authority || '*',
|
||||
icon: routeCfg.icon || router.icon || routeCfg.meta?.icon || router.meta?.icon,
|
||||
page: routeCfg.page || router.page || routeCfg.meta?.page || router.meta?.page,
|
||||
link: routeCfg.link || router.link || routeCfg.meta?.link || router.meta?.link
|
||||
}
|
||||
meta: {...meta, authority: meta.authority || '*'}
|
||||
}
|
||||
if (router.beforeEnter) {
|
||||
route.beforeEnter = router.beforeEnter
|
||||
}
|
||||
if (routeCfg.invisible || router.invisible) {
|
||||
route.meta.invisible = true
|
||||
@ -193,7 +215,7 @@ function formatAuthority(routes, pAuthorities = []) {
|
||||
let authority = {}
|
||||
if (!meta.authority) {
|
||||
authority = defaultAuthority
|
||||
}else if (typeof meta.authority === 'string') {
|
||||
}else if (typeof meta.authority === 'string' || Array.isArray(meta.authority)) {
|
||||
authority.permission = meta.authority
|
||||
} else if (typeof meta.authority === 'object') {
|
||||
authority = meta.authority
|
||||
|
@ -110,7 +110,7 @@ module.exports = {
|
||||
}
|
||||
}
|
||||
},
|
||||
publicPath: isProd ? '/vue-antd-admin/' : '/',
|
||||
publicPath: process.env.VUE_APP_PUBLIC_PATH,
|
||||
outputDir: 'dist',
|
||||
assetsDir: 'static',
|
||||
productionSourceMap: false
|
||||
|
21
yarn.lock
21
yarn.lock
@ -2735,15 +2735,10 @@ caniuse-api@^3.0.0:
|
||||
lodash.memoize "^4.1.2"
|
||||
lodash.uniq "^4.5.0"
|
||||
|
||||
caniuse-lite@^1.0.0, caniuse-lite@^1.0.30001043, caniuse-lite@^1.0.30001061:
|
||||
version "1.0.30001083"
|
||||
resolved "https://registry.npm.taobao.org/caniuse-lite/download/caniuse-lite-1.0.30001083.tgz?cache=0&sync_timestamp=1592075334738&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fcaniuse-lite%2Fdownload%2Fcaniuse-lite-1.0.30001083.tgz#52410c20c6f029f604f0d45eca0439a82e712442"
|
||||
integrity sha1-UkEMIMbwKfYE8NReygQ5qC5xJEI=
|
||||
|
||||
caniuse-lite@^1.0.30001087:
|
||||
version "1.0.30001088"
|
||||
resolved "https://registry.npm.taobao.org/caniuse-lite/download/caniuse-lite-1.0.30001088.tgz#23a6b9e192106107458528858f2c0e0dba0d9073"
|
||||
integrity sha1-I6a54ZIQYQdFhSiFjywODboNkHM=
|
||||
caniuse-lite@^1.0.0, caniuse-lite@^1.0.30001043, caniuse-lite@^1.0.30001061, caniuse-lite@^1.0.30001087:
|
||||
version "1.0.30001616"
|
||||
resolved "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001616.tgz"
|
||||
integrity sha512-RHVYKov7IcdNjVHJFNY/78RdG4oGVjbayxv8u5IO74Wv7Hlq4PnJE6mo/OjFijjVFNy5ijnCt6H3IIo4t+wfEw==
|
||||
|
||||
case-sensitive-paths-webpack-plugin@^2.3.0:
|
||||
version "2.3.0"
|
||||
@ -10423,10 +10418,10 @@ webpack-sources@*, webpack-sources@^1.0.1, webpack-sources@^1.1.0, webpack-sourc
|
||||
source-list-map "^2.0.0"
|
||||
source-map "~0.6.1"
|
||||
|
||||
webpack-theme-color-replacer@^1.3.12:
|
||||
version "1.3.12"
|
||||
resolved "https://registry.npm.taobao.org/webpack-theme-color-replacer/download/webpack-theme-color-replacer-1.3.12.tgz#0593a3149310c0e5b6b85afeccd61925b1b8e86b"
|
||||
integrity sha1-BZOjFJMQwOW2uFr+zNYZJbG46Gs=
|
||||
webpack-theme-color-replacer@1.3.18:
|
||||
version "1.3.18"
|
||||
resolved "https://registry.npmjs.org/webpack-theme-color-replacer/-/webpack-theme-color-replacer-1.3.18.tgz#98b70eab698e40b06ea3c56a3db8590f7ccef847"
|
||||
integrity sha512-z7qM3opvuSjAyJd0eLMOpZhH56r+fFctczWG6xnhUSeRsvbCg/EnFdsYoGL3xYJZNANvwLlggpJxnAcuFV5a6Q==
|
||||
dependencies:
|
||||
webpack-sources "*"
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user