feat(projects): 添加路由拦截判断权限

This commit is contained in:
‘chen.home’ 2022-08-14 18:05:43 +08:00
parent c646819b23
commit 4acb525777
11 changed files with 90 additions and 39 deletions

View File

@ -13,7 +13,7 @@ const userInfo = {
role: 'admin', role: 'admin',
password: '123456', password: '123456',
token, token,
permissions: [ userRoutes: [
{ {
name: 'dashboard', name: 'dashboard',
path: '/dashboard', path: '/dashboard',

View File

@ -9,10 +9,10 @@ async function setupApp() {
setupAssets(); setupAssets();
// 创建vue实例 // 创建vue实例
const app = createApp(App); const app = createApp(App);
// 安装router
await setupRouter(app);
// 安装pinia全局状态库 // 安装pinia全局状态库
setupStore(app); setupStore(app);
// 安装router
await setupRouter(app);
// 挂载 // 挂载
app.mount('#app'); app.mount('#app');
} }

View File

@ -1,25 +1,20 @@
import type { Router } from 'vue-router'; import type { Router } from 'vue-router';
import { getToken } from '@/utils/auth';
import { createPermissionGuard } from './permission';
const appTitle = import.meta.env.VITE_APP_TITLE; const appTitle = import.meta.env.VITE_APP_TITLE;
export function setupRouterGuard(router: Router) { export function setupRouterGuard(router: Router) {
const isLogin = Boolean(getToken()); router.beforeEach(async (to, from, next) => {
// 开始 loadingBar
router.beforeEach((to, _from, next) => { window.$loadingBar?.start();
// 登录鉴权 // 权限操作
if (!isLogin) { await createPermissionGuard(to, from, next);
if (to.name === 'login') { });
next(); router.afterEach((to) => {
} else {
const redirect = to.fullPath;
next({ path: '/login', query: { redirect } });
}
return false;
}
// 修改网页标题 // 修改网页标题
document.title = `${to.meta.title}——${appTitle}`; document.title = `${to.meta.title}——${appTitle}`;
next(); // 结束 loadingBar
window.$loadingBar?.finish();
}); });
// router.afterEach((_to) => {});
} }

View File

@ -0,0 +1,37 @@
import { RouteLocationNormalized, NavigationGuardNext } from 'vue-router';
import { getToken } from '@/utils/auth';
import { useRouteStore, useAuthStore } from '@/store';
export async function createPermissionGuard(
to: RouteLocationNormalized,
from: RouteLocationNormalized,
next: NavigationGuardNext,
) {
const isLogin = Boolean(getToken());
const routeStore = useRouteStore();
// 判断路由有无进行初始化
if (!routeStore.isInitAuthRoute) {
// 没有初始化路由 => 登录鉴权
// 登录鉴权
if (!isLogin) {
if (to.name === 'login') {
next();
} else {
const redirect = to.fullPath;
next({ path: '/login', query: { redirect } });
}
return false;
}
// 有登录但是没有路由,初始化路由等
await routeStore.initAuthRoute();
}
// 权限路由已经加载仍然未找到重定向到not-found
// if (to.name === 'not-found-page') {
// next({ name: 'not-found-page', replace: true });
// }
// next({ name: 'root' });
next();
}

View File

@ -8,7 +8,7 @@ const routes: RouteRecordRaw[] = [
{ {
path: '/', path: '/',
name: 'root', name: 'root',
redirect: '/test1', redirect: '/test/test1',
component: BasicLayout, component: BasicLayout,
children: [ children: [
{ {

View File

@ -1,6 +1,14 @@
// import { BasicLayout } from '@/layouts/index'; // import { BasicLayout } from '@/layouts/index';
export const constantRoutes = [ export const constantRoutes = [
{
path: '/no-found',
name: 'not-found',
component: () => import('@/views/inherit-page/not-found/index.vue'),
meta: {
title: '找不到页面',
},
},
{ {
path: '/no-permission', path: '/no-permission',
name: 'no-permission', name: 'no-permission',
@ -19,10 +27,10 @@ export const constantRoutes = [
}, },
{ {
path: '/:pathMatch(.*)*', path: '/:pathMatch(.*)*',
name: '404', name: 'not-found-page',
component: () => import('@/views/inherit-page/not-found/index.vue'), component: () => import('@/views/inherit-page/not-found/index.vue'),
meta: { meta: {
title: '错误404', title: '找不到页面',
}, },
}, },
]; ];

View File

@ -50,7 +50,8 @@ export const useAuthStore = defineStore('auth-store', {
// 等待数据写入完成 // 等待数据写入完成
const catchSuccess = await this.catchUserInfo(data); const catchSuccess = await this.catchUserInfo(data);
// 初始化侧边菜单 // 初始化侧边菜单
await this.setRouterStore(data.permissions); const { setMenus } = useRouteStore();
await setMenus();
// 登录写入信息成功 // 登录写入信息成功
if (catchSuccess) { if (catchSuccess) {
// 进行重定向跳转 // 进行重定向跳转
@ -65,7 +66,9 @@ export const useAuthStore = defineStore('auth-store', {
}); });
return; return;
} }
// 如果不成功写到后面
// 如果不成功则重置存储
this.resetAuthStore();
}, },
/* 缓存用户信息 */ /* 缓存用户信息 */
@ -82,11 +85,5 @@ export const useAuthStore = defineStore('auth-store', {
return catchSuccess; return catchSuccess;
}, },
/* 将路由表等信息推送到routeStore */
async setRouterStore(permissions: Auth.UserInfoPermissions[]) {
const { setMenus } = useRouteStore();
setMenus(permissions);
},
}, },
}); });

View File

@ -1,2 +1,3 @@
export * from './app'; export * from './app';
export * from './auth'; export * from './auth';
export * from './route';

View File

@ -1,23 +1,29 @@
import { defineStore } from 'pinia'; import { defineStore } from 'pinia';
import { renderIcon } from '@/utils/icon'; import { renderIcon, getUserInfo } from '@/utils';
import type { MenuOption } from 'naive-ui'; import type { MenuOption } from 'naive-ui';
// import { useAuthStore } from './auth';
interface RuutesStatus { // const authStore = useAuthStore();
interface RoutesStatus {
isInitAuthRoute: boolean;
menus: any; menus: any;
} }
export const useRouteStore = defineStore('route-store', { export const useRouteStore = defineStore('route-store', {
state: (): RuutesStatus => { state: (): RoutesStatus => {
return { return {
isInitAuthRoute: false,
menus: [], menus: [],
}; };
}, },
actions: { actions: {
setMenus(data: Auth.UserInfoPermissions[]) { async setMenus() {
this.menus = this.transformAuthRoutesToMenus(data); const { userRoutes } = getUserInfo();
this.menus = this.transformAuthRoutesToMenus(userRoutes);
}, },
// 将返回的路由表渲染成侧边栏 // 将返回的路由表渲染成侧边栏
transformAuthRoutesToMenus(data: Auth.UserInfoPermissions[]): MenuOption[] { transformAuthRoutesToMenus(userRoutes: Auth.UserInfoPermissions[]): MenuOption[] {
return data.map((item) => { return userRoutes.map((item) => {
const target: MenuOption = { const target: MenuOption = {
label: item.meta.title, label: item.meta.title,
key: item.path, key: item.path,
@ -33,5 +39,10 @@ export const useRouteStore = defineStore('route-store', {
return target; return target;
}); });
}, },
async initAuthRoute() {
await this.setMenus();
this.isInitAuthRoute = true;
},
}, },
}); });

View File

@ -26,7 +26,7 @@ declare namespace Auth {
/* token */ /* token */
token: string; token: string;
/* 权限路由 */ /* 权限路由 */
permissions: UserInfoPermissions[]; userRoutes: UserInfoPermissions[];
} }
interface UserInfoPermissions { interface UserInfoPermissions {
name: string; name: string;

View File

@ -35,10 +35,12 @@ declare namespace AppRoute {
/* 是否开启页面缓存 */ /* 是否开启页面缓存 */
keepAlive?: boolean; keepAlive?: boolean;
/* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */ /* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */
hideMenu?: boolean; hide?: boolean;
/* 菜单排序。 */ /* 菜单排序。 */
order?: number; order?: number;
/* 嵌套外链 */ /* 嵌套外链 */
herf?: string; herf?: string;
/** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */
activeMenu?: RouteKey;
} }
} }