fix: 🐛 修复注销登录location.reload导致接口无响应数据
This commit is contained in:
@@ -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")) {
|
||||||
|
|||||||
@@ -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,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -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,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user