先定义好路由,包括常页面路由和需要权限的路由数组
import Vue from 'vue'
import Router from 'vue-router'
Vue.use(Router)
import Layout from '@/layout'
export const StaticRouterMap = [
{
path: '/login',
name: 'Login',
component: () => import('@/views/login/index'),
hidden: true
},
{
path: '/404',
component: () => import('@/views/404'),
hidden: true
},
{
path: '/',
component: Layout,
redirect: '/dashboard',
children: [{
path: 'dashboard',
name: 'Home',
component: () => import('@/views/dashboard/index'),
meta: { title: '首页', icon: 'dashboard' }
}]
}
]
export const AsyncRouterMap = []
const createRouter = () =>
new Router({
scrollBehavior: () => ({ y: 0 }),
routes: StaticRouterMap
})
const router = createRouter()
export function resetRouter() {
const newRouter = createRouter()
router.matcher = newRouter.matcher // reset router
}
export default router
第二步和后台沟通返回的数据结构样式
第三步解析后台数据
/**
* 生成路由
* @param {Array} routerlist 格式化路由
* @returns
*/
export function addRouter(routerlist) {
const router = []
try {
routerlist.forEach(e => {
if (e.path !== 'button') {
let e_new = {
path: e.redirect,
name: e.path.split('/')[e.path.split('/').length - 1],
redirect: e.path,
component: () => e.path === 'layout' ? import('@/layout') : import('@/layout')
}
if (e.children) {
const children = addRouterchild(e.children)
// 保存权限
e_new = { ...e_new, children: children }
}
if (e.path) {
e_new = { ...e_new, redirect: e.path }
}
if (e.path) {
e_new = { ...e_new, alwaysShow: true }
}
e_new = { ...e_new, meta: { title: e.title }}
router.push(e_new)
}
})
} catch (error) {
console.error(error)
return []
}
return router
}
export function addRouterchild(routerlist) {
const router = []
try {
routerlist.forEach(e => {
if (e.path !== 'button') {
let e_new = {
path: e.redirect,
name: e.path.split('/')[e.path.split('/').length - 1],
component: (resolve) => require([`@/views${e.path}`], resolve)
}
e_new = { ...e_new, meta: { title: e.title }}
router.push(e_new)
}
})
} catch (error) {
console.error(error)
return []
}
return router
}
第四步在permission.js文件对每次的刷新做判断
/*
* @Author: your name
* @Date: 2020-03-21 08:29:57
* @LastEditTime: 2020-03-21 09:09:43
* @LastEditors: Please set LastEditors
* @Description: In User Settings Edit
* @FilePath: /vue-element-asyncLogin/src/permission.js
*/
import router from './router'
import store from './store'
import user from './store/modules/user'
import { removeToken } from './utils/auth'
import NProgress from 'nprogress' // Progress 进度条
import 'nprogress/nprogress.css' // Progress 进度条样式
import { Message } from 'element-ui'
import Cookies from 'js-cookie'
import { addRouter } from './utils/addRouter'
import request from '@/utils/request'
const whiteList = ['/login']
router.beforeEach((to, from, next) => {
NProgress.start()
if (Cookies.get('Authorization')) {
// 判断cookice是否存在 不存在即为未登录
if (to.path !== '/login') {
if (user.state.init) {
// 获取了动态路由 data一定true,就无需再次请求 直接放行
next()
} else {
// data为false,一定没有获取动态路由,就跳转到获取动态路由的方法
gotoRouter(to, next)
}
} else {
Message({ message: '您已经登录', type: 'info' })
next('/')
}
} else {
if (whiteList.indexOf(to.path) !== -1) {
// 免登陆白名单 直接进入
next()
} else {
if (to.path !== '/login') {
// 重定向到登录页面 不能这么写 因为假如之前的角色是 管理员页面 后又登陆了非管理员 重定向的页面就可能不存在,就会导致404
// next(`/login?redirect=${to.path}`)
next('/login')
} else {
next()
}
}
}
})
router.afterEach((to, from) => {
NProgress.done() // 结束Progress
})
function getRouter() {
var userId = JSON.parse(Cookies.get('userInfo')).userId
return request({
url: `/index/getMenuInfoByUId/${userId}`,
method: 'get'
})
// return p
}
function gotoRouter(to, next) {
getRouter() // 获取动态路由的方法
.then(res => {
const asyncRouter = addRouter(res.data) // 进行递归解析
asyncRouter.push({
path: '*',
redirect: '/404',
hidden: true
})
store.dispatch('user/setRouterList', asyncRouter) // 存储到vuex
router.addRoutes(asyncRouter) // vue-router提供的addRouter方法进行路由
// store.dispatch('user/GetInfo')
store.commit('user/set_init', true)
next({ ...to, replace: true }) // hack方法 确保addRo
})
.catch(e => {
removeToken()
})
}
第五步在登录接口进行路由拼接
// import { login, logout, getInfo } from '@/api/user'
import { login, logout } from '@/api/user'
import Cookies from 'js-cookie'
import { getToken, setToken, removeToken } from '@/utils/auth'
import { resetRouter } from '@/router'
import { StaticRouterMap } from '@/router/index'
import md5 from 'js-md5'
const getDefaultState = () => {
return {
token: getToken(),
name: '',
avatar: '',
roles: [],
init: false, // 是否完成初始化 // 默认未完成 入过坑没加这个
routerList: [] // 动态路由 如果坑没加这个会发生切换账号会保留上一个账号的路由信息
}
}
const state = getDefaultState()
const mutations = {
RESET_STATE: (state) => {
Object.assign(state, getDefaultState())
},
SET_TOKEN: (state, token) => {
state.token = token
},
SET_NAME: (state, name) => {
state.name = name
},
SET_AVATAR: (state, avatar) => {
state.avatar = avatar
},
SET_ROLES: (state, roles) => {
state.roles = roles
},
set_router: (state, RouterList) => {
console.log(RouterList,
state.routerList = RouterList
},
set_init: (state, status) => {
state.init = status //是permission.js中的初始化判断
// 注意data中要引入
}
}
const actions = {
// user login
login({ commit }, userInfo) {
const { username, password } = userInfo
return new Promise((resolve, reject) => {
login({ username: username.trim(), password: md5(password), rememberMe: true }).then(response => {
const data = response
commit('SET_TOKEN', data.token)
setToken(data.token)
commit('SET_NAME', data.userInfo.userName)
Cookies.set('userInfo', data.userInfo)
resolve()
}).catch(error => {
reject(error)
})
})
},
setRouterList({ commit }, routerList) {
commit('set_router', StaticRouterMap.concat(routerList)) // 进行路由拼接并存储
},
// user logout
logout({ commit, state }, userInfo) {
return new Promise((resolve, reject) => {
logout(userInfo).then(() => {
removeToken() // must remove token first
resetRouter()
commit('set_init', false) //重置初始化
commit('RESET_STATE')
resolve()
}).catch(error => {
reject(error)
})
})
},
// remove token
resetToken({ commit }) {
return new Promise(resolve => {
removeToken() // must remove token first
commit('RESET_STATE')
resolve()
})
}
}
export default {
namespaced: true,
state,
mutations,
actions
}
需要把拼接好的路由导出
const getters = {
sidebar: state => state.app.sidebar,
device: state => state.app.device,
token: state => state.user.token,
avatar: state => state.user.avatar,
name: state => state.user.name,
routerList: state => state.user.routerList //划重点语句,少写就不会渲染了
}
export default getters
注意侧边栏渲染的vue页面中,
改成:
完整的vue代码
<template>
<div :class="{'has-logo':showLogo}">
<logo v-if="showLogo" :collapse="isCollapse" />
<el-scrollbar wrap-class="scrollbar-wrapper">
<el-menu
:default-active="activeMenu"
:collapse="isCollapse"
:background-color="variables.menuBg"
:text-color="variables.menuText"
:unique-opened="false"
:active-text-color="variables.menuActiveText"
:collapse-transition="false"
mode="vertical"
>
<sidebar-item v-for="route in routes" :key="route.path" :item="route" :base-path="route.path" />
</el-menu>
</el-scrollbar>
</div>
</template>
<script>
import { mapGetters } from 'vuex'
import Logo from './Logo'
import SidebarItem from './SidebarItem'
import variables from '@/styles/variables.scss'
export default {
components: { SidebarItem, Logo },
computed: {
...mapGetters([
'sidebar'
]),
routes() {
return this.$store.getters.routerList
},
activeMenu() {
const route = this.$route
const { meta, path } = route
// if set path, the sidebar will highlight the path you set
if (meta.activeMenu) {
return meta.activeMenu
}
return path
},
showLogo() {
return this.$store.state.settings.sidebarLogo
},
variables() {
return variables
},
isCollapse() {
return !this.sidebar.opened
}
}
}
</script>