init: 初始系统模板

This commit is contained in:
TsMask
2023-09-05 14:38:23 +08:00
parent a5bc16ae4f
commit 1075c8ae4f
130 changed files with 22531 additions and 1 deletions

289
src/router/index.ts Normal file
View File

@@ -0,0 +1,289 @@
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', cache: true },
component: () => import('@/views/index.vue'),
},
{
path: '/dome1',
name: 'Dome1',
meta: { title: '示例一', icon: 'icon-ios' },
component: () => import('@/views/dome/dome1.vue'),
},
{
path: '/dome2',
name: 'Dome2',
meta: { title: '示例二', icon: 'icon-anzhuo' },
component: () => import('@/views/dome/dome2.vue'),
},
{
path: '/dome3',
name: 'Dome3',
meta: { title: '示例三', icon: 'icon-qunzhu' },
component: () => import('@/views/dome/dome3.vue'),
},
{
path: '/domes',
name: 'Domes',
meta: {
title: '示例目录',
icon: 'icon-zhizuoliucheng',
},
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',
},
component: () => {},
},
{
path: 'https://www.antdv.com/components/comment-cn?sdf=12321&id=12&sdnf',
name: 'SelfAnt Design Vue',
meta: {
title: 'Antdv-当前窗口',
icon: 'icon-morentouxiang',
target: '_self',
},
component: LinkLayout,
},
{
path: '/account',
name: 'Account',
meta: {
title: '个人中心',
},
component: BlankLayout,
redirect: '/account/profile',
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;