使用addRoutes重构权限

This commit is contained in:
Pan 2017-05-17 16:26:33 +08:00
parent 624af00ba3
commit faea4d8bf6
10 changed files with 191 additions and 192 deletions

View file

@ -20,7 +20,6 @@ import Sticky from 'components/Sticky'; // 粘性header组件
import vueWaves from './directive/waves';// 水波纹指令 import vueWaves from './directive/waves';// 水波纹指令
import errLog from 'store/errLog';// error log组件 import errLog from 'store/errLog';// error log组件
import './mock/index.js'; // 该项目所有请求使用mockjs模拟 import './mock/index.js'; // 该项目所有请求使用mockjs模拟
import permission from 'store/permission'; // 权限控制
// register globally // register globally
Vue.component('multiselect', Multiselect); Vue.component('multiselect', Multiselect);
@ -36,6 +35,7 @@ Object.keys(filters).forEach(key => {
// permissiom judge // permissiom judge
function hasPermission(roles, permissionRoles) { function hasPermission(roles, permissionRoles) {
if (roles.indexOf('admin') >= 0) return true; // admin权限 直接通过 if (roles.indexOf('admin') >= 0) return true; // admin权限 直接通过
if (!permissionRoles) return true;
return roles.some(role => permissionRoles.indexOf(role) >= 0) return roles.some(role => permissionRoles.indexOf(role) >= 0)
} }
@ -47,42 +47,24 @@ router.beforeEach((to, from, next) => {
if (to.path === '/login') { if (to.path === '/login') {
next({ path: '/' }); next({ path: '/' });
} else { } else {
if (to.meta && to.meta.role) { // 判断即将进入的页面是否需要权限 if (store.getters.roles.length === 0) { // 判断当前用户是否已拉取完info信息
if (store.getters.roles.length !== 0) { // 判断当前用户是否已拉取完info信息 store.dispatch('GetInfo').then(res => { // 拉取info
if (hasPermission(store.getters.roles, to.meta.role)) { // 判断权限 const roles = res.data.role;
next(); // 有权限 store.dispatch('GenerateRoutes', { roles }).then(() => { // 生成可访问的路由表
} else { router.addRoutes(store.getters.addRouters) // 动态添加可访问路由表
next({ path: '/401', query: { noGoBack: true } }); // 无权限 next(to); // hack方法 确保addRoutes已完成
} })
} else { // 未拉取info信息 }).catch(err => {
store.dispatch('GetInfo').then(() => { // 拉取info console.log(err);
permission.init({ // 初始化权限 });
roles: store.getters.roles, } else {
router: router.options.routes // 没有动态改变权限的需求可直接next() 删除下方权限判断 ↓
}); if (hasPermission(store.getters.roles, to.meta.role)) {
if (hasPermission(store.getters.roles, to.meta.role)) { // 判断权限 next();//
next();// 有权限
} else {
next({ path: '/401', query: { noGoBack: true } }); // 无权限
}
}).catch(err => {
console.log(err);
});
}
} else { // 页面不需要权限 直接进入
if (store.getters.roles.length !== 0) {
next();
} else { } else {
store.dispatch('GetInfo').then(() => { next({ path: '/401', query: { noGoBack: true } });
permission.init({
roles: store.getters.roles,
router: router.options.routes
});
next();
}).catch(err => {
console.log(err);
});
} }
// 可删 ↑
} }
} }
} else { } else {
@ -95,6 +77,7 @@ router.beforeEach((to, from, next) => {
} }
}); });
router.afterEach(() => { router.afterEach(() => {
NProgress.done(); // 结束Progress NProgress.done(); // 结束Progress
}); });

View file

@ -70,49 +70,56 @@ Vue.use(Router);
* noDropdown : if noDropdown:true will not has submenu * noDropdown : if noDropdown:true will not has submenu
* meta : { role: ['admin'] } will control the page role * meta : { role: ['admin'] } will control the page role
*/ */
export default new Router({
// mode: 'history', //后端支持可开 export const constantRouterMap = [
scrollBehavior: () => ({ y: 0 }),
routes: [
{ path: '/login', component: Login, hidden: true }, { path: '/login', component: Login, hidden: true },
{ path: '/authredirect', component: authRedirect, hidden: true }, { path: '/authredirect', component: authRedirect, hidden: true },
{ path: '/sendpwd', component: sendPWD, hidden: true }, { path: '/sendpwd', component: sendPWD, hidden: true },
{ path: '/reset', component: reset, hidden: true }, { path: '/reset', component: reset, hidden: true },
{ path: '/404', component: Err404, hidden: true }, { path: '/404', component: Err404, hidden: true },
{ path: '/401', component: Err401, hidden: true }, { path: '/401', component: Err401, hidden: true },
{ {
path: '/', path: '/',
component: Layout, component: Layout,
redirect: '/dashboard', redirect: '/dashboard',
name: '首页', name: '首页',
hidden: true, hidden: true,
children: [{ path: 'dashboard', component: dashboard }] children: [{ path: 'dashboard', component: dashboard }]
}, },
{ {
path: '/introduction', path: '/introduction',
component: Layout, component: Layout,
redirect: '/introduction/index', redirect: '/introduction/index',
icon: 'xinrenzhinan', icon: 'xinrenzhinan',
noDropdown: true, noDropdown: true,
children: [{ path: 'index', component: Introduction, name: '简述' }] children: [{ path: 'index', component: Introduction, name: '简述' }]
}, }
{ ]
path: '/permission',
component: Layout, export default new Router({
redirect: '/permission/index', // mode: 'history', //后端支持可开
name: '权限测试', scrollBehavior: () => ({ y: 0 }),
icon: 'quanxian', routes: constantRouterMap
meta: { role: ['admin'] }, });
noDropdown: true,
children: [{ path: 'index', component: Permission, name: '权限测试页', meta: { role: ['admin'] } }] export const asyncRouterMap = [
}, {
{ path: '/permission',
path: '/components', component: Layout,
component: Layout, redirect: '/permission/index',
redirect: '/components/index', name: '权限测试',
name: '组件', icon: 'quanxian',
icon: 'zujian', meta: { role: ['admin'] },
children: [ noDropdown: true,
children: [{ path: 'index', component: Permission, name: '权限测试页', meta: { role: ['admin'] } }]
},
{
path: '/components',
component: Layout,
redirect: '/components/index',
name: '组件',
icon: 'zujian',
children: [
{ path: 'index', component: componentsIndex, name: '介绍 ' }, { path: 'index', component: componentsIndex, name: '介绍 ' },
{ path: 'tinymce', component: Tinymce, name: '富文本编辑器' }, { path: 'tinymce', component: Tinymce, name: '富文本编辑器' },
{ path: 'markdown', component: Markdown, name: 'Markdown' }, { path: 'markdown', component: Markdown, name: 'Markdown' },
@ -124,75 +131,73 @@ export default new Router({
{ path: 'sticky', component: Sticky, name: 'Sticky' }, { path: 'sticky', component: Sticky, name: 'Sticky' },
{ path: 'countto', component: CountTo, name: 'CountTo' }, { path: 'countto', component: CountTo, name: 'CountTo' },
{ path: 'mixin', component: Mixin, name: '小组件' } { path: 'mixin', component: Mixin, name: '小组件' }
] ]
}, },
{ {
path: '/charts', path: '/charts',
component: Layout, component: Layout,
redirect: '/charts/index', redirect: '/charts/index',
name: '图表', name: '图表',
icon: 'tubiaoleixingzhengchang', icon: 'tubiaoleixingzhengchang',
children: [ children: [
{ path: 'index', component: chartIndex, name: '介绍' }, { path: 'index', component: chartIndex, name: '介绍' },
{ path: 'keyboard', component: KeyboardChart, name: '键盘图表' }, { path: 'keyboard', component: KeyboardChart, name: '键盘图表' },
{ path: 'keyboard2', component: KeyboardChart2, name: '键盘图表2' }, { path: 'keyboard2', component: KeyboardChart2, name: '键盘图表2' },
{ path: 'line', component: LineMarker, name: '折线图' }, { path: 'line', component: LineMarker, name: '折线图' },
{ path: 'mixchart', component: MixChart, name: '混合图表' } { path: 'mixchart', component: MixChart, name: '混合图表' }
] ]
}, },
{ {
path: '/errorpage', path: '/errorpage',
component: Layout, component: Layout,
redirect: 'noredirect', redirect: 'noredirect',
name: '错误页面', name: '错误页面',
icon: '404', icon: '404',
children: [ children: [
{ path: '401', component: Err401, name: '401' }, { path: '401', component: Err401, name: '401' },
{ path: '404', component: Err404, name: '404' } { path: '404', component: Err404, name: '404' }
] ]
}, },
{ {
path: '/errlog', path: '/errlog',
component: Layout, component: Layout,
redirect: 'noredirect', redirect: 'noredirect',
name: 'errlog', name: 'errlog',
icon: 'bug', icon: 'bug',
noDropdown: true, noDropdown: true,
children: [{ path: 'log', component: ErrorLog, name: '错误日志' }] children: [{ path: 'log', component: ErrorLog, name: '错误日志' }]
}, },
{ {
path: '/excel', path: '/excel',
component: Layout, component: Layout,
redirect: 'noredirect', redirect: 'noredirect',
name: 'excel', name: 'excel',
icon: 'EXCEL', icon: 'EXCEL',
noDropdown: true, noDropdown: true,
children: [{ path: 'download', component: ExcelDownload, name: '导出excel' }] children: [{ path: 'download', component: ExcelDownload, name: '导出excel' }]
}, },
{ {
path: '/theme', path: '/theme',
component: Layout, component: Layout,
redirect: 'noredirect', redirect: 'noredirect',
name: 'theme', name: 'theme',
icon: 'theme', icon: 'theme',
noDropdown: true, noDropdown: true,
children: [{ path: 'index', component: Theme, name: '换肤' }] children: [{ path: 'index', component: Theme, name: '换肤' }]
}, },
{ {
path: '/example', path: '/example',
component: Layout, component: Layout,
redirect: 'noredirect', redirect: 'noredirect',
name: '综合实例', name: '综合实例',
icon: 'zonghe', icon: 'zonghe',
children: [ children: [
{ path: 'dynamictable', component: DynamicTable, name: '动态table' }, { path: 'dynamictable', component: DynamicTable, name: '动态table' },
{ path: 'dragtable', component: DragTable, name: '拖拽table' }, { path: 'dragtable', component: DragTable, name: '拖拽table' },
{ path: 'inline_edit_table', component: InlineEditTable, name: 'table内编辑' }, { path: 'inline_edit_table', component: InlineEditTable, name: 'table内编辑' },
{ path: 'table', component: Table, name: '综合table' }, { path: 'table', component: Table, name: '综合table' },
{ path: 'form1', component: Form1, name: '综合form1' } { path: 'form1', component: Form1, name: '综合form1' }
] ]
}, },
{ path: '*', redirect: '/404', hidden: true }
{ path: '*', redirect: '/404', hidden: true } ];
]
});

View file

@ -1,6 +1,5 @@
const getters = { const getters = {
sidebar: state => state.app.sidebar, sidebar: state => state.app.sidebar,
livenewsChannels: state => state.app.livenewsChannels,
token: state => state.user.token, token: state => state.user.token,
avatar: state => state.user.avatar, avatar: state => state.user.avatar,
name: state => state.user.name, name: state => state.user.name,
@ -10,6 +9,8 @@ const getters = {
auth_type: state => state.user.auth_type, auth_type: state => state.user.auth_type,
status: state => state.user.status, status: state => state.user.status,
roles: state => state.user.roles, roles: state => state.user.roles,
setting: state => state.user.setting setting: state => state.user.setting,
permission_routers: state => state.permission.routers,
addRouters: state => state.permission.addRouters
}; };
export default getters export default getters

View file

@ -2,6 +2,7 @@ import Vue from 'vue';
import Vuex from 'vuex'; import Vuex from 'vuex';
import app from './modules/app'; import app from './modules/app';
import user from './modules/user'; import user from './modules/user';
import permission from './modules/permission';
import getters from './getters'; import getters from './getters';
Vue.use(Vuex); Vue.use(Vuex);
@ -9,7 +10,8 @@ Vue.use(Vuex);
const store = new Vuex.Store({ const store = new Vuex.Store({
modules: { modules: {
app, app,
user user,
permission
}, },
getters getters
}); });

View file

@ -16,21 +16,11 @@ const app = {
Cookies.set('sidebarStatus', 0); Cookies.set('sidebarStatus', 0);
} }
state.sidebar.opened = !state.sidebar.opened; state.sidebar.opened = !state.sidebar.opened;
},
SET_LIVENEWS_CHANNELS: (status, channels) => {
status.livenewsChannels = JSON.stringify(channels);
Cookies.set('livenewsChannels', JSON.stringify(channels));
} }
}, },
actions: { actions: {
ToggleSideBar: ({ commit }) => { ToggleSideBar: ({ commit }) => {
commit('TOGGLE_SIDEBAR') commit('TOGGLE_SIDEBAR')
},
setTheme: ({ commit }, theme) => {
commit('SET_THEME', theme)
},
setlivenewsChannels: ({ commit }, channels) => {
commit('SET_LIVENEWS_CHANNELS', channels)
} }
} }
}; };

View file

@ -0,0 +1,51 @@
import { asyncRouterMap, constantRouterMap } from 'src/router';
function hasPermission(roles, route) {
if (route.meta && route.meta.role) {
return roles.some(role => route.meta.role.indexOf(role) >= 0)
} else {
return true
}
}
const permission = {
state: {
routers: constantRouterMap,
addRouters: []
},
mutations: {
SET_ROUTERS: (state, routers) => {
state.addRouters = routers;
state.routers = constantRouterMap.concat(routers);
}
},
actions: {
// s
GenerateRoutes({ commit }, data) {
const { roles } = data;
const accessedRouters = asyncRouterMap.filter(v => {
if (roles.indexOf('admin') >= 0) return true;
if (hasPermission(roles, v)) {
if (v.children && v.children.length > 0) {
v.children = v.children.filter(child => {
if (hasPermission(roles, child)) {
return child
}
return false;
});
return v
} else {
return v
}
}
return false;
});
commit('SET_ROUTERS', accessedRouters);
}
}
};
export default permission;

View file

@ -1,38 +0,0 @@
const permission = {
state: {
permissionRoutes: []
},
init(data) {
const { roles, router } = data;
const permissionRoutes = router.filter(v => {
if (roles.indexOf('admin') >= 0) return true;
if (this.hasPermission(roles, v)) {
if (v.children && v.children.length > 0) {
v.children = v.children.filter(child => {
if (this.hasPermission(roles, child)) {
return child
}
return false;
});
return v
} else {
return v
}
}
return false;
});
this.state.permissionRoutes = permissionRoutes;
},
get() {
return this.state.permissionRoutes
},
hasPermission(roles, route) {
if (route.meta && route.meta.role) {
return roles.some(role => route.meta.role.indexOf(role) >= 0)
} else {
return true
}
}
};
export default permission;

View file

@ -28,4 +28,8 @@
} }
}; };
</script> </script>
<style scoped>
.errPage-container{
padding: 30px;
}
</style>

View file

@ -26,7 +26,7 @@
</template> </template>
<script> <script>
import { mapGetters } from 'vuex' import { mapGetters } from 'vuex';
import Levelbar from './Levelbar'; import Levelbar from './Levelbar';
import Hamburger from 'components/Hamburger'; import Hamburger from 'components/Hamburger';
import ErrLog from 'components/ErrLog'; import ErrLog from 'components/ErrLog';
@ -56,7 +56,7 @@
}, },
logout() { logout() {
this.$store.dispatch('LogOut').then(() => { this.$store.dispatch('LogOut').then(() => {
this.$router.push({ path: '/login' }) location.reload();// vue-router bug
}); });
} }
} }

View file

@ -1,6 +1,6 @@
<template> <template>
<el-menu :unique-opened='true' mode="vertical" theme="dark" :default-active="$route.path"> <el-menu :unique-opened='true' mode="vertical" theme="dark" :default-active="$route.path">
<template v-for="item in permissionRoutes" v-if="!item.hidden"> <template v-for="item in permission_routers" v-if="!item.hidden">
<el-submenu :index="item.name" v-if="!item.noDropdown"> <el-submenu :index="item.name" v-if="!item.noDropdown">
<template slot="title"> <template slot="title">
<wscn-icon-svg :icon-class="item.icon||'wenzhang1'" /> {{item.name}} <wscn-icon-svg :icon-class="item.icon||'wenzhang1'" /> {{item.name}}
@ -21,13 +21,14 @@
</template> </template>
<script> <script>
import permissionRoutes from 'store/permission'; import { mapGetters } from 'vuex';
export default { export default {
name: 'Sidebar', name: 'Sidebar',
data() { computed: {
return { ...mapGetters([
permissionRoutes: permissionRoutes.get() 'permission_routers'
} ])
} }
} }
</script> </script>