mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-06 03:57:54 +08:00
fix: 修改tab逻辑
修改tab逻辑,fixed刷新跳转bug,add静态路由
This commit is contained in:
parent
518689f64d
commit
a0fbe28595
2
.env
2
.env
@ -4,5 +4,7 @@ VITE_BASE_URL=/
|
|||||||
VITE_APP_TITLE = Ench Admin
|
VITE_APP_TITLE = Ench Admin
|
||||||
# 路由模式
|
# 路由模式
|
||||||
VITE_HASH_ROUTE = Y
|
VITE_HASH_ROUTE = Y
|
||||||
|
# 权限路由模式: static | dynamic
|
||||||
|
VITE_AUTH_ROUTE_MODE=dynamic
|
||||||
# 存储前缀
|
# 存储前缀
|
||||||
VITE_STORAGE_PREFIX = ""
|
VITE_STORAGE_PREFIX = ""
|
||||||
|
@ -26,6 +26,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"axios": "^0.27.2",
|
"axios": "^0.27.2",
|
||||||
"pinia": "^2.0.17",
|
"pinia": "^2.0.17",
|
||||||
|
"pinia-plugin-persist": "^1.0.0",
|
||||||
"vue": "^3.2.37",
|
"vue": "^3.2.37",
|
||||||
"vue-router": "^4.1.3"
|
"vue-router": "^4.1.3"
|
||||||
},
|
},
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
:collapsed-width="64"
|
:collapsed-width="64"
|
||||||
:collapsed-icon-size="24"
|
:collapsed-icon-size="24"
|
||||||
:indent="20"
|
:indent="20"
|
||||||
|
accordion
|
||||||
:options="routesStore.menus"
|
:options="routesStore.menus"
|
||||||
:value="routesStore.activeMenu"
|
:value="routesStore.activeMenu"
|
||||||
@update:value="handleClickMenu"
|
@update:value="handleClickMenu"
|
||||||
|
@ -28,8 +28,13 @@ export async function createPermissionGuard(
|
|||||||
// 有登录但是没有路由,初始化路由、侧边菜单等
|
// 有登录但是没有路由,初始化路由、侧边菜单等
|
||||||
await routeStore.initAuthRoute();
|
await routeStore.initAuthRoute();
|
||||||
// 动态路由加载完回到根路由
|
// 动态路由加载完回到根路由
|
||||||
next({ name: 'appRoot' });
|
if (to.name === 'not-found') {
|
||||||
return false;
|
// 动态路由没有加载导致被not-found-page路由捕获,等待权限路由加载好了,回到之前的路由
|
||||||
|
// 若路由是从根路由重定向过来的,重新回到根路由
|
||||||
|
const path = to.redirectedFrom?.fullPath;
|
||||||
|
next({ path, replace: true, query: to.query, hash: to.hash });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// 权限路由已经加载,仍然未找到,重定向到not-found
|
// 权限路由已经加载,仍然未找到,重定向到not-found
|
||||||
// if (to.name === 'not-found-page') {
|
// if (to.name === 'not-found-page') {
|
||||||
|
@ -1,26 +1,7 @@
|
|||||||
import type { App } from 'vue';
|
import type { App } from 'vue';
|
||||||
import { createRouter, createWebHistory, createWebHashHistory, RouteRecordRaw } from 'vue-router';
|
import { createRouter, createWebHistory, createWebHashHistory, RouteRecordRaw } from 'vue-router';
|
||||||
import { setupRouterGuard } from './guard';
|
import { setupRouterGuard } from './guard';
|
||||||
import { BasicLayout } from '@/layouts/index';
|
import { routes } from './routes';
|
||||||
import { constantRoutes } from './routes';
|
|
||||||
|
|
||||||
const routes: RouteRecordRaw[] = [
|
|
||||||
{
|
|
||||||
path: '/',
|
|
||||||
name: 'root',
|
|
||||||
redirect: 'appRoot',
|
|
||||||
component: BasicLayout,
|
|
||||||
children: [...constantRoutes],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: '/login',
|
|
||||||
name: 'login',
|
|
||||||
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
|
||||||
meta: {
|
|
||||||
title: '登录',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const { VITE_HASH_ROUTE = 'N', VITE_BASE_URL } = import.meta.env;
|
const { VITE_HASH_ROUTE = 'N', VITE_BASE_URL } = import.meta.env;
|
||||||
export const router = createRouter({
|
export const router = createRouter({
|
||||||
|
35
src/router/modules/dashboard.ts
Normal file
35
src/router/modules/dashboard.ts
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
import { RouteRecordRaw } from 'vue-router';
|
||||||
|
import { BasicLayout } from '@/layouts/index';
|
||||||
|
export const dashboard: RouteRecordRaw = {
|
||||||
|
path: '/dashboard',
|
||||||
|
name: 'dashboard',
|
||||||
|
redirect: '/dashboard/workbench',
|
||||||
|
component: BasicLayout,
|
||||||
|
meta: {
|
||||||
|
title: '分析页',
|
||||||
|
requiresAuth: true,
|
||||||
|
icon: 'icon-park-outline:analysis',
|
||||||
|
},
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
name: 'dashboard_workbench',
|
||||||
|
path: '/dashboard/workbench',
|
||||||
|
component: () => import('@/views/dashboard/workbench/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '工作台',
|
||||||
|
requiresAuth: true,
|
||||||
|
icon: 'icon-park-outline:alarm',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'dashboard_monitor',
|
||||||
|
path: '/dashboard/monitor',
|
||||||
|
component: () => import('@/views/dashboard/monitor/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '监控页',
|
||||||
|
requiresAuth: true,
|
||||||
|
icon: 'icon-park-outline:anchor',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
3
src/router/modules/index.ts
Normal file
3
src/router/modules/index.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
import { dashboard } from './dashboard';
|
||||||
|
|
||||||
|
export const staticRoutes = [dashboard];
|
@ -1,34 +1,53 @@
|
|||||||
|
import { RouteRecordRaw } from 'vue-router';
|
||||||
|
import { BasicLayout } from '@/layouts/index';
|
||||||
|
|
||||||
/* 页面中的一些固定路由,错误页等 */
|
/* 页面中的一些固定路由,错误页等 */
|
||||||
export const constantRoutes = [
|
export const routes: RouteRecordRaw[] = [
|
||||||
{
|
{
|
||||||
path: '/no-found',
|
path: '/',
|
||||||
name: 'not-found',
|
name: 'root',
|
||||||
component: () => import('@/views/error/not-found/index.vue'),
|
redirect: 'appRoot',
|
||||||
meta: {
|
component: BasicLayout,
|
||||||
title: '找不到页面',
|
children: [
|
||||||
icon: 'icon-park-outline:ghost',
|
{
|
||||||
},
|
path: '/no-found',
|
||||||
|
name: 'not-found',
|
||||||
|
component: () => import('@/views/error/not-found/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '找不到页面',
|
||||||
|
icon: 'icon-park-outline:ghost',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/no-permission',
|
||||||
|
name: 'no-permission',
|
||||||
|
component: () => import('@/views/error/not-permission/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '用户无权限',
|
||||||
|
icon: 'icon-park-outline:error',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/service-error',
|
||||||
|
name: 'service-error',
|
||||||
|
component: () => import('@/views/error/service-error/index.vue'),
|
||||||
|
meta: {
|
||||||
|
title: '服务器错误',
|
||||||
|
icon: 'icon-park-outline:close-wifi',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/:pathMatch(.*)*',
|
||||||
|
redirect: '/no-found',
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: '/no-permission',
|
path: '/login',
|
||||||
name: 'no-permission',
|
name: 'login',
|
||||||
component: () => import('@/views/error/not-permission/index.vue'),
|
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
||||||
meta: {
|
meta: {
|
||||||
title: '用户无权限',
|
title: '登录',
|
||||||
icon: 'icon-park-outline:error',
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
|
||||||
path: '/service-error',
|
|
||||||
name: 'service-error',
|
|
||||||
component: () => import('@/views/error/service-error/index.vue'),
|
|
||||||
meta: {
|
|
||||||
title: '服务器错误',
|
|
||||||
icon: 'icon-park-outline:close-wifi',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: '/:pathMatch(.*)*',
|
|
||||||
redirect: '/no-found',
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
import { createPinia } from 'pinia';
|
import { createPinia } from 'pinia';
|
||||||
|
import piniaPluginPersist from 'pinia-plugin-persist';
|
||||||
import type { App } from 'vue';
|
import type { App } from 'vue';
|
||||||
|
|
||||||
export function setupStore(app: App) {
|
export function setupStore(app: App) {
|
||||||
const store = createPinia();
|
const store = createPinia();
|
||||||
|
store.use(piniaPluginPersist);
|
||||||
app.use(store);
|
app.use(store);
|
||||||
}
|
}
|
||||||
export * from './modules';
|
export * from './modules';
|
||||||
|
@ -41,7 +41,7 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
this.loginLoading = true;
|
this.loginLoading = true;
|
||||||
const { data } = await fetchLogin({ userName, password });
|
const { data } = await fetchLogin({ userName, password });
|
||||||
// 处理登录信息
|
// 处理登录信息
|
||||||
await this.handleAfterLogin(data); // TODO 避免any
|
await this.handleAfterLogin(data);
|
||||||
|
|
||||||
this.loginLoading = false;
|
this.loginLoading = false;
|
||||||
},
|
},
|
||||||
@ -50,9 +50,6 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
async handleAfterLogin(data: Auth.loginToken) {
|
async handleAfterLogin(data: Auth.loginToken) {
|
||||||
// 将token和userInfo保存下来
|
// 将token和userInfo保存下来
|
||||||
const catchSuccess = await this.catchUserInfo(data);
|
const catchSuccess = await this.catchUserInfo(data);
|
||||||
// 初始化侧边菜单
|
|
||||||
// const { initAuthRoute } = useRouteStore();
|
|
||||||
// await initAuthRoute();
|
|
||||||
|
|
||||||
// 登录写入信息成功
|
// 登录写入信息成功
|
||||||
if (catchSuccess) {
|
if (catchSuccess) {
|
||||||
@ -68,9 +65,14 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 如果不成功则重置存储
|
// 如果不成功则重置存储
|
||||||
this.resetAuthStore();
|
this.resetAuthStore();
|
||||||
|
// 登录失败提示
|
||||||
|
window.$notification?.error({
|
||||||
|
title: '登录失败!',
|
||||||
|
content: `验证失败,请检查账号密码`,
|
||||||
|
duration: 3000,
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
/* 缓存用户信息 */
|
/* 缓存用户信息 */
|
||||||
|
@ -4,12 +4,14 @@ import { MenuOption } from 'naive-ui';
|
|||||||
import { createDynamicRoutes } from '@/router/guard/dynamic';
|
import { createDynamicRoutes } from '@/router/guard/dynamic';
|
||||||
import { router } from '@/router';
|
import { router } from '@/router';
|
||||||
import { fetchUserRoutes } from '@/service';
|
import { fetchUserRoutes } from '@/service';
|
||||||
|
import { staticRoutes } from '@/router/modules';
|
||||||
|
|
||||||
interface RoutesStatus {
|
interface RoutesStatus {
|
||||||
isInitAuthRoute: boolean;
|
isInitAuthRoute: boolean;
|
||||||
menus: any;
|
menus: any;
|
||||||
userRoutes: AppRoute.Route[];
|
userRoutes: AppRoute.Route[];
|
||||||
activeMenu: string | null;
|
activeMenu: string | null;
|
||||||
|
authRouteMode: ImportMetaEnv['VITE_AUTH_ROUTE_MODE'];
|
||||||
}
|
}
|
||||||
export const useRouteStore = defineStore('route-store', {
|
export const useRouteStore = defineStore('route-store', {
|
||||||
state: (): RoutesStatus => {
|
state: (): RoutesStatus => {
|
||||||
@ -18,6 +20,7 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
isInitAuthRoute: false,
|
isInitAuthRoute: false,
|
||||||
menus: [],
|
menus: [],
|
||||||
activeMenu: null,
|
activeMenu: null,
|
||||||
|
authRouteMode: import.meta.env.VITE_AUTH_ROUTE_MODE,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
actions: {
|
actions: {
|
||||||
@ -83,6 +86,14 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
// 插入路由表
|
// 插入路由表
|
||||||
router.addRoute(appRoutes);
|
router.addRoute(appRoutes);
|
||||||
},
|
},
|
||||||
|
/* 初始化静态路由 */
|
||||||
|
async initStaticRoute() {
|
||||||
|
/* 将静态路由转换为侧边菜单 */
|
||||||
|
staticRoutes.forEach((route) => {
|
||||||
|
// 插入路由表
|
||||||
|
router.addRoute(route);
|
||||||
|
});
|
||||||
|
},
|
||||||
//* 将返回的路由表渲染成侧边栏 */
|
//* 将返回的路由表渲染成侧边栏 */
|
||||||
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
|
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
|
||||||
return userRoutes
|
return userRoutes
|
||||||
@ -111,7 +122,11 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
},
|
},
|
||||||
async initAuthRoute() {
|
async initAuthRoute() {
|
||||||
this.isInitAuthRoute = false;
|
this.isInitAuthRoute = false;
|
||||||
await this.initDynamicRoute();
|
if (this.authRouteMode === 'dynamic') {
|
||||||
|
await this.initDynamicRoute();
|
||||||
|
} else {
|
||||||
|
await this.initStaticRoute();
|
||||||
|
}
|
||||||
this.isInitAuthRoute = true;
|
this.isInitAuthRoute = true;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -9,6 +9,7 @@ interface TabState {
|
|||||||
path: string;
|
path: string;
|
||||||
}[];
|
}[];
|
||||||
tabs: RouteLocationNormalized[];
|
tabs: RouteLocationNormalized[];
|
||||||
|
tabWhiteList: string[];
|
||||||
currentTab: string;
|
currentTab: string;
|
||||||
}
|
}
|
||||||
export const useTabStore = defineStore('tab-store', {
|
export const useTabStore = defineStore('tab-store', {
|
||||||
@ -22,6 +23,7 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
tabs: [],
|
tabs: [],
|
||||||
|
tabWhiteList: ['not-found', 'no-permission', 'service-error', 'login'],
|
||||||
currentTab: 'dashboard_workbench',
|
currentTab: 'dashboard_workbench',
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@ -42,6 +44,10 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
if (this.hasExistTab(route.name as string)) {
|
if (this.hasExistTab(route.name as string)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
// 如果在白名单内则不添加,错误页等
|
||||||
|
if (this.tabWhiteList.includes(route.name as string)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.tabs.push(route);
|
this.tabs.push(route);
|
||||||
},
|
},
|
||||||
closeTab(name: string) {
|
closeTab(name: string) {
|
||||||
@ -56,7 +62,7 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
if (this.currentTab === name && !isLast) {
|
if (this.currentTab === name && !isLast) {
|
||||||
// 跳转到后一个标签
|
// 跳转到后一个标签
|
||||||
routerPush(this.tabs[index + 1].path);
|
routerPush(this.tabs[index + 1].path);
|
||||||
} else {
|
} else if (this.currentTab === name && isLast) {
|
||||||
// 已经是最后一个了,就跳转前一个
|
// 已经是最后一个了,就跳转前一个
|
||||||
routerPush(this.tabs[index - 1].path);
|
routerPush(this.tabs[index - 1].path);
|
||||||
}
|
}
|
||||||
@ -85,4 +91,7 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
persist: {
|
||||||
|
enabled: true,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
2
src/types/env.d.ts
vendored
2
src/types/env.d.ts
vendored
@ -32,6 +32,8 @@ interface ImportMetaEnv {
|
|||||||
readonly VITE_COMPRESS_TYPE?: 'gzip' | 'brotliCompress' | 'deflate' | 'deflateRaw';
|
readonly VITE_COMPRESS_TYPE?: 'gzip' | 'brotliCompress' | 'deflate' | 'deflateRaw';
|
||||||
/** hash路由模式 */
|
/** hash路由模式 */
|
||||||
readonly VITE_HASH_ROUTE?: 'Y' | 'N';
|
readonly VITE_HASH_ROUTE?: 'Y' | 'N';
|
||||||
|
/** 路由加载模式 */
|
||||||
|
readonly VITE_AUTH_ROUTE_MODE?: 'static' | 'dynamic';
|
||||||
/** 本地存储前缀 */
|
/** 本地存储前缀 */
|
||||||
readonly VITE_STORAGE_PREFIX?: string;
|
readonly VITE_STORAGE_PREFIX?: string;
|
||||||
/** 后端服务的环境类型 */
|
/** 后端服务的环境类型 */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user