141 lines
4.6 KiB
JavaScript
141 lines
4.6 KiB
JavaScript
import auth from '@/plugins/auth'
|
||
import router, { constantRoutes, dynamicRoutes } from '@/router'
|
||
import { getRouters } from '@/api/menu'
|
||
import Layout from '@/layout/index'
|
||
import ParentView from '@/components/ParentView'
|
||
import InnerLink from '@/layout/components/InnerLink'
|
||
|
||
const permission = {
|
||
state: {
|
||
routes: [],
|
||
addRoutes: [],
|
||
defaultRoutes: [],
|
||
topbarRouters: [],
|
||
sidebarRouters: []
|
||
},
|
||
mutations: {
|
||
SET_ROUTES: (state, routes) => {
|
||
state.addRoutes = routes
|
||
state.routes = constantRoutes.concat(routes)
|
||
},
|
||
SET_DEFAULT_ROUTES: (state, routes) => {
|
||
state.defaultRoutes = constantRoutes.concat(routes)
|
||
},
|
||
SET_TOPBAR_ROUTES: (state, routes) => {
|
||
state.topbarRouters = routes
|
||
},
|
||
SET_SIDEBAR_ROUTERS: (state, routes) => {
|
||
state.sidebarRouters = routes
|
||
},
|
||
},
|
||
actions: {
|
||
// 生成路由
|
||
GenerateRoutes({ commit }) {
|
||
return new Promise(resolve => {
|
||
// 模拟路由数据(后端服务未运行时使用)
|
||
// 使用dynamicRoutes作为模拟数据,确保侧边栏显示所有动态路由
|
||
const mockRoutes = dynamicRoutes
|
||
const sidebarRoutes = filterAsyncRouter(mockRoutes)
|
||
const rewriteRoutes = filterAsyncRouter(mockRoutes, false, true)
|
||
const asyncRoutes = filterDynamicRoutes(dynamicRoutes)
|
||
rewriteRoutes.push({ path: '*', redirect: '/404', hidden: true })
|
||
router.addRoutes(asyncRoutes)
|
||
commit('SET_ROUTES', rewriteRoutes)
|
||
// 确保首页固定在顶部
|
||
const homeRouteIndex = constantRoutes.findIndex(route => route.children && route.children.some(child => child.name === 'Index'));
|
||
if (homeRouteIndex > 0) {
|
||
// 将首页路由移到最前面
|
||
const routes = [...constantRoutes];
|
||
const homeRoute = routes.splice(homeRouteIndex, 1)[0];
|
||
routes.unshift(homeRoute);
|
||
commit('SET_SIDEBAR_ROUTERS', routes.concat(sidebarRoutes));
|
||
} else {
|
||
commit('SET_SIDEBAR_ROUTERS', constantRoutes.concat(sidebarRoutes));
|
||
}
|
||
commit('SET_DEFAULT_ROUTES', sidebarRoutes)
|
||
commit('SET_TOPBAR_ROUTES', sidebarRoutes)
|
||
resolve(rewriteRoutes)
|
||
})
|
||
}
|
||
}
|
||
}
|
||
|
||
// 遍历后台传来的路由字符串,转换为组件对象
|
||
function filterAsyncRouter(asyncRouterMap, lastRouter = false, type = false) {
|
||
return asyncRouterMap.filter(route => {
|
||
// 只排除特定的学生信息模块路由,保留我们的学生管理模块
|
||
if (route.path === '/student-info') {
|
||
return false;
|
||
}
|
||
|
||
if (type && route.children) {
|
||
route.children = filterChildren(route.children)
|
||
}
|
||
if (route.component) {
|
||
// Layout ParentView 组件特殊处理
|
||
if (route.component === 'Layout') {
|
||
route.component = Layout
|
||
} else if (route.component === 'ParentView') {
|
||
route.component = ParentView
|
||
} else if (route.component === 'InnerLink') {
|
||
route.component = InnerLink
|
||
} else {
|
||
route.component = loadView(route.component)
|
||
}
|
||
}
|
||
if (route.children != null && route.children && route.children.length) {
|
||
// 递归过滤子路由
|
||
route.children = filterAsyncRouter(route.children, route, type)
|
||
// 如果子路由都被过滤掉了,这个父路由也应该被过滤
|
||
if (route.children.length === 0) {
|
||
return false;
|
||
}
|
||
} else {
|
||
delete route['children']
|
||
delete route['redirect']
|
||
}
|
||
return true
|
||
})
|
||
}
|
||
|
||
function filterChildren(childrenMap, lastRouter = false) {
|
||
var children = []
|
||
childrenMap.forEach(el => {
|
||
el.path = lastRouter ? lastRouter.path + '/' + el.path : el.path
|
||
if (el.children && el.children.length && el.component === 'ParentView') {
|
||
children = children.concat(filterChildren(el.children, el))
|
||
} else {
|
||
children.push(el)
|
||
}
|
||
})
|
||
return children
|
||
}
|
||
|
||
// 动态路由遍历,验证是否具备权限
|
||
export function filterDynamicRoutes(routes) {
|
||
const res = []
|
||
routes.forEach(route => {
|
||
if (route.permissions) {
|
||
if (auth.hasPermiOr(route.permissions)) {
|
||
res.push(route)
|
||
}
|
||
} else if (route.roles) {
|
||
if (auth.hasRoleOr(route.roles)) {
|
||
res.push(route)
|
||
}
|
||
}
|
||
})
|
||
return res
|
||
}
|
||
|
||
export const loadView = (view) => {
|
||
if (process.env.NODE_ENV === 'development') {
|
||
return (resolve) => require([`@/views/${view}`], resolve)
|
||
} else {
|
||
// 使用 import 实现生产环境的路由懒加载
|
||
return () => import(`@/views/${view}`)
|
||
}
|
||
}
|
||
|
||
export default permission
|