294 lines
8.2 KiB
TypeScript
294 lines
8.2 KiB
TypeScript
import {
|
||
createRouter,
|
||
createWebHistory,
|
||
createWebHashHistory,
|
||
RouteRecordRaw,
|
||
} from 'vue-router';
|
||
import NProgress from 'nprogress';
|
||
import 'nprogress/nprogress.css';
|
||
import BasicLayout from '../layouts/BasicLayout.vue';
|
||
import BlankLayout from '../layouts/BlankLayout.vue';
|
||
import LinkLayout from '../layouts/LinkLayout.vue';
|
||
import { encode } from 'js-base64';
|
||
import { getToken } from '@/plugins/auth-token';
|
||
import { validHttp } from '@/utils/regular-utils';
|
||
import useUserStore from '@/store/modules/user';
|
||
import useAppStore from '@/store/modules/app';
|
||
import useRouterStore from '@/store/modules/router';
|
||
|
||
// NProgress Configuration
|
||
NProgress.configure({ showSpinner: false });
|
||
|
||
// import { MetaRecord, MenuDataItem } from '@ant-design-vue/pro-layout';
|
||
// mate数据类型 MetaRecord
|
||
// 根据/路径构建菜单列表,列表项类型 MenuDataItem
|
||
// https://github.com/vueComponent/pro-components/blob/a19279f3a28190bf11e8c36f316c92dbd3387a6d/packages/pro-layout/src/typings.ts#L16
|
||
// 菜单图标来源 https://ant.design/components/icon 自定义iconfont
|
||
|
||
/**公共路由 */
|
||
const constantRoutes: RouteRecordRaw[] = [
|
||
{
|
||
path: '/',
|
||
name: 'Root',
|
||
meta: { title: '根节点' },
|
||
component: BasicLayout,
|
||
redirect: '/index',
|
||
children: [
|
||
{
|
||
path: '/index',
|
||
name: 'Index',
|
||
meta: { title: '首页', icon: 'icon-pcduan' },
|
||
component: () => import('@/views/index.vue'),
|
||
},
|
||
{
|
||
path: '/dome1',
|
||
name: 'Dome1',
|
||
meta: { title: '示例一', icon: 'icon-ios', hideInMenu: true },
|
||
component: () => import('@/views/dome/dome1.vue'),
|
||
},
|
||
{
|
||
path: '/dome2',
|
||
name: 'Dome2',
|
||
meta: { title: '示例二', icon: 'icon-anzhuo', hideInMenu: true },
|
||
component: () => import('@/views/dome/dome2.vue'),
|
||
},
|
||
{
|
||
path: '/dome3',
|
||
name: 'Dome3',
|
||
meta: { title: '示例三', icon: 'icon-qunzhu', hideInMenu: true },
|
||
component: () => import('@/views/dome/dome3.vue'),
|
||
},
|
||
{
|
||
path: '/domes',
|
||
name: 'Domes',
|
||
meta: {
|
||
title: '示例目录',
|
||
icon: 'icon-zhizuoliucheng',
|
||
hideInMenu: true,
|
||
},
|
||
component: BlankLayout,
|
||
redirect: () => ({ name: 'PageInfo' }),
|
||
children: [
|
||
{
|
||
path: 'page-info',
|
||
name: 'PageInfo',
|
||
meta: { title: '页面信息', icon: 'icon-huifu' },
|
||
component: () => import('../views/domes/page-info.vue'),
|
||
},
|
||
{
|
||
path: 'page-typography',
|
||
name: 'PageTypography',
|
||
meta: { title: '文本信息', icon: 'icon-huizhiguize' },
|
||
component: () => import('../views/domes/page-typography.vue'),
|
||
},
|
||
{
|
||
path: 'dynamic-match/:id(\\d+)',
|
||
name: 'DynamicMatch',
|
||
// 路由 path 默认参数再 meta.params 里
|
||
meta: { title: '动态参数页面', params: { id: 1 }, cache: true },
|
||
component: () => import('../views/domes/dynamic-match.vue'),
|
||
},
|
||
{
|
||
path: 'disabled',
|
||
name: 'Disabled',
|
||
meta: { title: '禁止点击', disabled: true },
|
||
component: () => {},
|
||
},
|
||
{
|
||
path: 'https://github.com/TsMask',
|
||
name: 'BlankGithubTsMask',
|
||
meta: {
|
||
title: 'TsMask-打开新窗',
|
||
icon: 'icon-github',
|
||
target: '_blank',
|
||
},
|
||
component: () => {},
|
||
},
|
||
{
|
||
path: encode('https://www.antdv.com/components/comment-cn'),
|
||
name: 'HttpsAntDesignVue',
|
||
meta: {
|
||
title: 'Antdv-内嵌窗口',
|
||
icon: 'icon-morentouxiang',
|
||
target: null,
|
||
},
|
||
component: LinkLayout,
|
||
},
|
||
],
|
||
},
|
||
{
|
||
path: 'https://github.com/',
|
||
name: 'BlankGithub',
|
||
meta: {
|
||
title: 'Github-打开新窗',
|
||
icon: 'icon-github',
|
||
target: '_blank',
|
||
hideInMenu: true,
|
||
},
|
||
component: () => {},
|
||
},
|
||
{
|
||
path: 'https://3x.antdv.com/components/comment-cn?sdf=12321&id=12&sdnf',
|
||
name: 'SelfAnt Design Vue',
|
||
meta: {
|
||
title: 'Antdv-当前窗口',
|
||
icon: 'icon-morentouxiang',
|
||
target: '_self',
|
||
hideInMenu: true,
|
||
},
|
||
component: LinkLayout,
|
||
},
|
||
{
|
||
path: '/account',
|
||
name: 'Account',
|
||
meta: {
|
||
title: '个人中心',
|
||
hideInMenu: true,
|
||
},
|
||
component: BlankLayout,
|
||
redirect: '/account/settings',
|
||
children: [
|
||
{
|
||
path: 'profile',
|
||
name: 'Profile',
|
||
meta: { title: '个人信息', cache: true },
|
||
component: () => import('@/views/account/profile.vue'),
|
||
},
|
||
{
|
||
path: 'settings',
|
||
name: 'Settings',
|
||
meta: { title: '个人设置', cache: true },
|
||
component: () => import('@/views/account/settings.vue'),
|
||
},
|
||
],
|
||
},
|
||
],
|
||
},
|
||
{
|
||
path: '/login',
|
||
name: 'Login',
|
||
meta: { title: '登录' },
|
||
component: () => import('@/views/login.vue'),
|
||
},
|
||
{
|
||
path: '/register',
|
||
name: 'Register',
|
||
meta: { title: '注册' },
|
||
component: () => import('@/views/register.vue'),
|
||
},
|
||
{
|
||
path: '/403',
|
||
name: 'NotPermission',
|
||
meta: { title: '没有访问权限' },
|
||
component: () => import('@/views/error/403.vue'),
|
||
},
|
||
{
|
||
path: '/redirect',
|
||
name: 'Redirect',
|
||
meta: { title: '重定向' },
|
||
component: BasicLayout,
|
||
children: [
|
||
{
|
||
path: '/redirect/:path(.*)',
|
||
component: () => import('@/views/redirect/index.vue'),
|
||
},
|
||
],
|
||
},
|
||
{
|
||
path: '/:pathMatch(.*)*',
|
||
meta: { title: '找不到匹配页面' },
|
||
component: () => import('@/views/error/404.vue'),
|
||
},
|
||
];
|
||
|
||
// 根据.env配置获取是否带井号和基础路径
|
||
const hasHash = import.meta.env.VITE_HISTORY_HASH;
|
||
const bashUrl = import.meta.env.VITE_HISTORY_BASE_URL;
|
||
|
||
/**全局路由 */
|
||
const router = createRouter({
|
||
history:
|
||
hasHash === 'true'
|
||
? createWebHashHistory(bashUrl)
|
||
: createWebHistory(bashUrl),
|
||
routes: constantRoutes,
|
||
scrollBehavior(to, from, savedPosition) {
|
||
if (savedPosition) {
|
||
return savedPosition;
|
||
} else {
|
||
return { top: 0 };
|
||
}
|
||
},
|
||
});
|
||
|
||
/**全局路由-后置守卫 */
|
||
router.afterEach((to, from, failure) => {
|
||
NProgress.done();
|
||
// 设置标题
|
||
if (to.meta?.title) {
|
||
useAppStore().setTitle(to.meta.title);
|
||
}
|
||
});
|
||
|
||
/**无Token可访问页面地址白名单 */
|
||
const WHITE_LIST: string[] = ['/login', '/auth-redirect', '/bind', '/register'];
|
||
|
||
/**全局路由-前置守卫 */
|
||
router.beforeEach((to, from, next) => {
|
||
NProgress.start();
|
||
const token = getToken();
|
||
|
||
// 没有token
|
||
if (!token) {
|
||
if (WHITE_LIST.includes(to.path)) {
|
||
// 在免登录白名单,直接进入
|
||
next();
|
||
} else {
|
||
// 否则全部重定向到登录页
|
||
next(`/login?redirect=${to.fullPath}`);
|
||
}
|
||
}
|
||
|
||
// 有Token
|
||
if (token) {
|
||
// 防止重复访问登录页面
|
||
if (to.path === '/login') {
|
||
next({ name: 'Index' });
|
||
} else {
|
||
// 判断当前用户是否有角色信息
|
||
const user = useUserStore();
|
||
if (user.roles && user.roles.length === 0) {
|
||
// 获取用户信息
|
||
user
|
||
.fnGetInfo()
|
||
.then(() => {
|
||
return useRouterStore().generateRoutes();
|
||
})
|
||
.then(accessRoutes => {
|
||
// 根据后台配置生成可访问的路由表
|
||
if (accessRoutes && accessRoutes.length !== 0) {
|
||
for (const route of accessRoutes) {
|
||
// 动态添加可访问路由表,http开头会异常
|
||
if (!validHttp(route.path)) {
|
||
router.addRoute(route);
|
||
}
|
||
}
|
||
}
|
||
// 刷新替换原先路由,确保addRoutes已完成
|
||
next({ ...to, replace: true });
|
||
})
|
||
.catch(e => {
|
||
console.error(`[${to.path}]: ${e.message}`);
|
||
user.fnLogOut().finally(() => {
|
||
next({ name: 'Login' });
|
||
});
|
||
});
|
||
} else {
|
||
next();
|
||
}
|
||
}
|
||
}
|
||
});
|
||
|
||
export default router;
|