fix: 🐛 修复注销登录location.reload导致接口无响应数据

This commit is contained in:
ray
2024-10-18 22:04:18 +08:00
parent f0e045599b
commit 605d87b3a7
3 changed files with 76 additions and 49 deletions

View File

@@ -3,10 +3,8 @@ import type {
RouteLocationNormalized, RouteLocationNormalized,
RouteRecordRaw, RouteRecordRaw,
} from "vue-router"; } from "vue-router";
import NProgress from "@/utils/nprogress"; import NProgress from "@/utils/nprogress";
import { isLogin } from "@/utils/auth"; import { getToken } from "@/utils/auth";
import router from "@/router"; import router from "@/router";
import { usePermissionStore, useUserStore } from "@/store"; import { usePermissionStore, useUserStore } from "@/store";
@@ -16,22 +14,21 @@ export function setupPermission() {
router.beforeEach(async (to, from, next) => { router.beforeEach(async (to, from, next) => {
NProgress.start(); NProgress.start();
if (isLogin()) {
if (to.path === "/login") {
// 如果已登录,跳转到首页
next({ path: "/" });
NProgress.done();
} else {
const userStore = useUserStore();
const hasRoles =
userStore.user.roles && userStore.user.roles.length > 0;
if (hasRoles) { const isLogin = !!getToken(); // 判断是否登录
// 如果未匹配到任何路由跳转到404页面 if (isLogin) {
if (to.matched.length === 0) { if (to.path === "/login") {
next(from.name ? { name: from.name } : "/404"); // 已登录,访问登录页,跳转到首页
next({ path: "/" });
} else { } else {
// 如果路由参数中有 title覆盖路由元信息中的 title const permissionStore = usePermissionStore();
// 判断路由是否加载过
if (permissionStore.isRoutesLoaded) {
if (to.matched.length === 0) {
// 路由未匹配跳转到404
next("/404");
} else {
// 动态设置页面标题
const title = const title =
(to.params.title as string) || (to.query.title as string); (to.params.title as string) || (to.query.title as string);
if (title) { if (title) {
@@ -40,35 +37,35 @@ export function setupPermission() {
next(); next();
} }
} else { } else {
const permissionStore = usePermissionStore();
try { try {
await userStore.getUserInfo(); // 生成动态路由
const dynamicRoutes = await permissionStore.generateRoutes(); const dynamicRoutes = await permissionStore.generateRoutes();
dynamicRoutes.forEach((route: RouteRecordRaw) => dynamicRoutes.forEach((route: RouteRecordRaw) =>
router.addRoute(route) router.addRoute(route)
); );
next({ ...to, replace: true }); next({ ...to, replace: true }); // 添加动态路由后重新导航
} catch (error) { } catch (error) {
console.error(error); console.error(error);
// 移除 token 并重定向到登录页,携带当前页面路由作为跳转参数 // 路由加载失败,重置 token 并重定向到登录页
await userStore.resetToken(); await useUserStore().clearUserSession();
redirectToLogin(to, next); redirectToLogin(to, next);
NProgress.done(); NProgress.done();
} }
} }
} }
} else { } else {
// 未登录 // 未登录,判断是否在白名单中
if (whiteList.includes(to.path)) { if (whiteList.includes(to.path)) {
next(); // 在白名单,直接进入 next();
} else { } else {
// 不在白名单,重定向到登录页 // 不在白名单,重定向到登录页
redirectToLogin(to, next); redirectToLogin(to, next);
NProgress.done(); NProgress.done(); // 关闭进度条
} }
} }
}); });
// 后置守卫,保证每次路由跳转结束时关闭进度条
router.afterEach(() => { router.afterEach(() => {
NProgress.done(); NProgress.done();
}); });
@@ -90,7 +87,7 @@ export function hasAuth(
value: string | string[], value: string | string[],
type: "button" | "role" = "button" type: "button" | "role" = "button"
) { ) {
const { roles, perms } = useUserStore().user; const { roles, perms } = useUserStore().userInfo;
// 超级管理员 拥有所有权限 // 超级管理员 拥有所有权限
if (type === "button" && roles.includes("ROOT")) { if (type === "button" && roles.includes("ROOT")) {

View File

@@ -1,17 +1,21 @@
import type { RouteRecordRaw } from "vue-router"; import type { RouteRecordRaw } from "vue-router";
import { constantRoutes } from "@/router"; import { constantRoutes } from "@/router";
import { store } from "@/store"; import { store } from "@/store";
import MenuAPI, { type RouteVO } from "@/api/menu"; import MenuAPI, { type RouteVO } from "@/api/system/menu";
const modules = import.meta.glob("../../views/**/**.vue"); const modules = import.meta.glob("../../views/**/**.vue");
const Layout = () => import("@/layout/index.vue"); const Layout = () => import("@/layout/index.vue");
import router from "@/router";
export const usePermissionStore = defineStore("permission", () => { export const usePermissionStore = defineStore("permission", () => {
/** 所有路由,包括静态和动态路由 */ /** 所有路由,包括静态和动态路由 */
const routes = ref<RouteRecordRaw[]>([]); const routes = ref<RouteRecordRaw[]>([]);
/** 混合模式左侧菜单 */ /** 混合模式左侧菜单 */
const mixLeftMenus = ref<RouteRecordRaw[]>([]); const mixLeftMenus = ref<RouteRecordRaw[]>([]);
const isRoutesLoaded = ref(false);
/** /**
* 生成动态路由 * 生成动态路由
*/ */
@@ -21,6 +25,7 @@ export const usePermissionStore = defineStore("permission", () => {
.then((data) => { .then((data) => {
const dynamicRoutes = transformRoutes(data); const dynamicRoutes = transformRoutes(data);
routes.value = constantRoutes.concat(dynamicRoutes); routes.value = constantRoutes.concat(dynamicRoutes);
isRoutesLoaded.value = true;
resolve(dynamicRoutes); resolve(dynamicRoutes);
}) })
.catch((error) => { .catch((error) => {
@@ -41,11 +46,29 @@ export const usePermissionStore = defineStore("permission", () => {
} }
}; };
/**
* 重置路由
*/
const resetRouter = () => {
// 删除动态路由,保留静态路由
routes.value.forEach((route) => {
if (route.name && !constantRoutes.find((r) => r.name === route.name)) {
router.removeRoute(route.name); // 从 router 实例中移除动态路由
}
});
routes.value = [];
mixLeftMenus.value = [];
isRoutesLoaded.value = false;
};
return { return {
routes, routes,
generateRoutes, generateRoutes,
mixLeftMenus, mixLeftMenus,
setMixLeftMenus, setMixLeftMenus,
isRoutesLoaded,
resetRouter,
}; };
}); });

View File

@@ -1,14 +1,14 @@
import AuthAPI, { type LoginData } from "@/api/auth";
import UserAPI, { type UserInfo } from "@/api/user";
import { resetRouter } from "@/router";
import { store } from "@/store"; import { store } from "@/store";
import { setToken, removeToken } from "@/utils/auth"; import { usePermissionStoreHook } from "@/store/modules/permission";
import { useDictStoreHook } from "@/store/modules/dict";
import AuthAPI, { type LoginData } from "@/api/auth";
import UserAPI, { type UserInfo } from "@/api/system/user";
import { setToken, clearToken } from "@/utils/auth";
export const useUserStore = defineStore("user", () => { export const useUserStore = defineStore("user", () => {
const user = ref<UserInfo>({ const userInfo = useStorage<UserInfo>("userInfo", {} as UserInfo);
roles: [],
perms: [],
});
/** /**
* 登录 * 登录
@@ -30,7 +30,11 @@ export const useUserStore = defineStore("user", () => {
}); });
} }
// 获取信息(用户昵称、头像、角色集合、权限集合) /**
* 获取用户信息
*
* @returns {UserInfo} 用户信息
*/
function getUserInfo() { function getUserInfo() {
return new Promise<UserInfo>((resolve, reject) => { return new Promise<UserInfo>((resolve, reject) => {
UserAPI.getInfo() UserAPI.getInfo()
@@ -39,11 +43,7 @@ export const useUserStore = defineStore("user", () => {
reject("Verification failed, please Login again."); reject("Verification failed, please Login again.");
return; return;
} }
if (!data.roles || data.roles.length <= 0) { Object.assign(userInfo.value, { ...data });
reject("getUserInfo: roles must be a non-null array!");
return;
}
Object.assign(user.value, { ...data });
resolve(data); resolve(data);
}) })
.catch((error) => { .catch((error) => {
@@ -52,12 +52,14 @@ export const useUserStore = defineStore("user", () => {
}); });
} }
// user logout /**
* 登出
*/
function logout() { function logout() {
return new Promise<void>((resolve, reject) => { return new Promise<void>((resolve, reject) => {
AuthAPI.logout() AuthAPI.logout()
.then(() => { .then(() => {
location.reload(); // 清空路由 clearUserSession();
resolve(); resolve();
}) })
.catch((error) => { .catch((error) => {
@@ -66,21 +68,26 @@ export const useUserStore = defineStore("user", () => {
}); });
} }
// remove token /**
function resetToken() { * 清理用户会话
*
* @returns
*/
function clearUserSession() {
return new Promise<void>((resolve) => { return new Promise<void>((resolve) => {
removeToken(); clearToken();
resetRouter(); usePermissionStoreHook().resetRouter();
useDictStoreHook().clearDictionaryCache();
resolve(); resolve();
}); });
} }
return { return {
user, userInfo,
login,
getUserInfo, getUserInfo,
login,
logout, logout,
resetToken, clearUserSession,
}; };
}); });