feat(menu): 修改菜单高亮自动更新

This commit is contained in:
‘chen.home’ 2022-08-18 00:17:45 +08:00
parent e4dc741844
commit 8a24e76d00
11 changed files with 125 additions and 105 deletions

View File

@ -5,79 +5,6 @@ const Random = Mock.Random;
const token = Random.string('upper', 32, 32); const token = Random.string('upper', 32, 32);
const routeData = {
admin: [
{
name: 'dashboard',
path: '/dashboard',
meta: {
title: '分析页',
requiresAuth: true,
icon: 'icon-park-outline:analysis',
},
children: [
{
name: 'dashboard_workbench',
path: '/dashboard/workbench',
meta: {
title: '工作台',
requiresAuth: true,
icon: 'icon-park-outline:alarm',
},
},
{
name: 'dashboard_monitor',
path: '/dashboard/monitor',
meta: {
title: '监控页',
requiresAuth: true,
icon: 'icon-park-outline:anchor',
},
},
],
},
{
name: 'test',
path: '/test',
meta: {
title: '测试专题',
requiresAuth: true,
icon: 'icon-park-outline:ambulance',
},
children: [
{
name: 'test1',
path: '/test/test1',
meta: {
title: '测试专题1',
requiresAuth: true,
icon: 'icon-park-outline:alarm',
},
},
{
name: 'test2',
path: '/test/test2',
meta: {
title: '测试专题2',
requiresAuth: true,
icon: 'icon-park-outline:pic',
},
},
{
name: 'test3',
path: '/test/test3',
meta: {
title: '测试专题3',
requiresAuth: true,
icon: 'icon-park-outline:tool',
},
},
],
},
],
user: [],
};
const userInfo = { const userInfo = {
userId: '1', userId: '1',
userName: 'admin', userName: 'admin',
@ -144,6 +71,19 @@ const userRoutes = [
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:pic', icon: 'icon-park-outline:pic',
}, },
children: [
{
name: 'test2_detail',
path: '/test/test2/detail',
meta: {
title: '测试专题2的详情页',
requiresAuth: true,
icon: 'icon-park-outline:tool',
hide: true,
activeMenu: '/test/test2',
},
},
],
}, },
{ {
name: 'test3', name: 'test3',
@ -153,6 +93,17 @@ const userRoutes = [
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:tool', icon: 'icon-park-outline:tool',
}, },
children: [
{
name: 'test4',
path: '/test/test3/test4',
meta: {
title: '测试专题4',
requiresAuth: true,
icon: 'icon-park-outline:tool',
},
},
],
}, },
], ],
}, },

View File

@ -10,9 +10,12 @@
<script setup lang="ts"> <script setup lang="ts">
import { computed } from 'vue'; import { computed } from 'vue';
import { useRouter } from 'vue-router'; import { useRouter } from 'vue-router';
import { useRouteStore } from '~/src/store';
const router = useRouter(); const router = useRouter();
const routeStore = useRouteStore();
const routes = computed(() => { const routes = computed(() => {
// return routeStore.createBreadcrumbInRoutes(router.currentRoute.value.name, routeStore.userRoutes);
return router.currentRoute.value.matched.filter((item) => { return router.currentRoute.value.matched.filter((item) => {
return item.meta.title; return item.meta.title;
}); });

View File

@ -1,5 +1,5 @@
<template> <template>
<div class="h-60px text-2xl flex-center overflow-hidden"> <div class="h-60px text-2xl flex-center overflow-hidden cursor-pointer" @click="pushHome">
<SvgIcon name="logo" :size="28" /> <SvgIcon name="logo" :size="28" />
<span v-show="!appStore.collapsed" class="mx-4">{{ appStore.title }}</span> <span v-show="!appStore.collapsed" class="mx-4">{{ appStore.title }}</span>
</div> </div>
@ -7,7 +7,12 @@
<script setup lang="ts"> <script setup lang="ts">
import { useAppStore } from '@/store'; import { useAppStore } from '@/store';
import { useAppRouter } from '@/hook';
const appStore = useAppStore(); const appStore = useAppStore();
const { routerPush } = useAppRouter();
const pushHome = () => {
routerPush('/');
};
</script> </script>
<style scoped></style> <style scoped></style>

View File

@ -5,6 +5,7 @@
:collapsed-icon-size="24" :collapsed-icon-size="24"
:indent="20" :indent="20"
:options="routesStore.menus" :options="routesStore.menus"
:value="routesStore.activeMenu"
@update:value="handleClickMenu" @update:value="handleClickMenu"
/> />
</template> </template>
@ -13,12 +14,13 @@
import { useAppStore } from '@/store'; import { useAppStore } from '@/store';
import { useAppRouter } from '@/hook'; import { useAppRouter } from '@/hook';
import { useRouteStore } from '~/src/store/modules/route'; import { useRouteStore } from '~/src/store/modules/route';
import type { MenuOption } from 'naive-ui';
const { routerPush } = useAppRouter(); const { routerPush } = useAppRouter();
const appStore = useAppStore(); const appStore = useAppStore();
const routesStore = useRouteStore(); const routesStore = useRouteStore();
const handleClickMenu = (key: string) => { const handleClickMenu = (key: string, item: MenuOption) => {
routerPush(key); routerPush(key);
}; };
</script> </script>

View File

@ -37,6 +37,10 @@ export async function createDynamicRoutes(routes: AppRoute.Route[]) {
name: 'appRoot', name: 'appRoot',
redirect: '/dashboard/workbench', redirect: '/dashboard/workbench',
component: BasicLayout, component: BasicLayout,
meta: {
title: '首页',
icon: 'icon-park-outline:home',
},
children: [], children: [],
}; };
// 根据角色过滤后的插入根路由中 // 根据角色过滤后的插入根路由中

View File

@ -13,7 +13,7 @@ export function setupRouterGuard(router: Router) {
}); });
router.afterEach((to) => { router.afterEach((to) => {
// 修改网页标题 // 修改网页标题
document.title = `${to.meta.title}——${appTitle}`; document.title = `${to.meta.title}${appTitle}`;
// 结束 loadingBar // 结束 loadingBar
window.$loadingBar?.finish(); window.$loadingBar?.finish();
}); });

View File

@ -1,7 +1,6 @@
import { RouteLocationNormalized, NavigationGuardNext } from 'vue-router'; import { RouteLocationNormalized, NavigationGuardNext } from 'vue-router';
import { getToken } from '@/utils/auth'; import { getToken } from '@/utils/auth';
import { useRouteStore } from '@/store'; import { useRouteStore } from '@/store';
// import { setDynamicRoutes } from './dynamic';
export async function createPermissionGuard( export async function createPermissionGuard(
to: RouteLocationNormalized, to: RouteLocationNormalized,
@ -26,7 +25,6 @@ export async function createPermissionGuard(
return false; return false;
} }
// 有登录但是没有路由,初始化路由、侧边菜单等 // 有登录但是没有路由,初始化路由、侧边菜单等
// await setDynamicRoutes();
await routeStore.initAuthRoute(); await routeStore.initAuthRoute();
// 动态路由加载完回到根路由 // 动态路由加载完回到根路由
next({ name: 'root' }); next({ name: 'root' });
@ -37,6 +35,12 @@ export async function createPermissionGuard(
// next({ name: 'not-found-page', replace: true }); // next({ name: 'not-found-page', replace: true });
// } // }
// next({ name: 'root' }); // 设置菜单高亮
if (to.meta.activeMenu) {
routeStore.setActiveMenu(to.meta.activeMenu);
} else {
routeStore.setActiveMenu(to.fullPath);
}
next(); next();
} }

View File

@ -8,7 +8,8 @@ import { fetchUserRoutes } from '@/service';
interface RoutesStatus { interface RoutesStatus {
isInitAuthRoute: boolean; isInitAuthRoute: boolean;
menus: any; menus: any;
userRoutes: any; userRoutes: AppRoute.Route[];
activeMenu: string | null;
} }
export const useRouteStore = defineStore('route-store', { export const useRouteStore = defineStore('route-store', {
state: (): RoutesStatus => { state: (): RoutesStatus => {
@ -16,6 +17,7 @@ export const useRouteStore = defineStore('route-store', {
userRoutes: [], userRoutes: [],
isInitAuthRoute: false, isInitAuthRoute: false,
menus: [], menus: [],
activeMenu: null,
}; };
}, },
actions: { actions: {
@ -27,10 +29,38 @@ export const useRouteStore = defineStore('route-store', {
/* 删除后面添加的路由 */ /* 删除后面添加的路由 */
router.removeRoute('appRoot'); router.removeRoute('appRoot');
}, },
/* 根据当前路由的name生成面包屑数据 */
createBreadcrumbInRoutes(name = '/', userRoutes: AppRoute.Route[]) {
return userRoutes.filter((item) => {
if (item.name === name) {
return true;
}
if (item.children) {
return this.hasNameInBreadcrumbsChildren(name, item.children);
}
});
},
/* 判断子路由中是否存在为name的路由 */
hasNameInBreadcrumbsChildren(name: string, userRoutes: AppRoute.Route[]): boolean {
return userRoutes.some((item) => {
if (item.name === name) {
return true;
}
if (item.children) {
return this.hasNameInBreadcrumbsChildren(name, item.children);
}
});
},
/* 设置当前高亮的菜单key */
setActiveMenu(key: string) {
this.activeMenu = key;
},
/* 生成侧边菜单的数据 */
createMenus(userRoutes: AppRoute.Route[]) { createMenus(userRoutes: AppRoute.Route[]) {
this.userRoutes = userRoutes; this.userRoutes = userRoutes;
this.menus = this.transformAuthRoutesToMenus(userRoutes); this.menus = this.transformAuthRoutesToMenus(userRoutes);
}, },
/* 初始化动态路由 */
async initDynamicRoute() { async initDynamicRoute() {
// 根据用户id来获取用户的路由 // 根据用户id来获取用户的路由
const { userId } = getUserInfo(); const { userId } = getUserInfo();
@ -42,26 +72,34 @@ export const useRouteStore = defineStore('route-store', {
// 插入路由表 // 插入路由表
router.addRoute(appRoutes); router.addRoute(appRoutes);
}, },
// 将返回的路由表渲染成侧边栏 //* 将返回的路由表渲染成侧边栏 */
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] { transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
return userRoutes.map((item) => { return userRoutes
const target: MenuOption = { .filter((item) => {
label: item.meta.title, return !item.meta.hide;
key: item.path, })
}; .map((item) => {
// 判断有无图标 const target: MenuOption = {
if (item.meta.icon) { label: item.meta.title,
target.icon = renderIcon(item.meta.icon); key: item.path,
} };
// 判断子元素 // 判断有无图标
if (item.children) { if (item.meta.icon) {
target.children = this.transformAuthRoutesToMenus(item.children); target.icon = renderIcon(item.meta.icon);
} }
return target; // 判断子元素
}); if (item.children) {
const children = this.transformAuthRoutesToMenus(item.children);
// 只有子元素有且不为空时才添加
if (children.length !== 0) {
target.children = children;
}
}
return target;
});
}, },
async initAuthRoute() { async initAuthRoute() {
this.isInitAuthRoute = false;
await this.initDynamicRoute(); await this.initDynamicRoute();
this.isInitAuthRoute = true; this.isInitAuthRoute = true;
}, },

View File

@ -7,7 +7,6 @@ declare namespace Auth {
* - user: 用户 * - user: 用户
* - custom: 自定义角色 * - custom: 自定义角色
*/ */
// type RoleType = keyof typeof import('@/enum').EnumUserRole;
/** 用户信息 */ /** 用户信息 */
interface loginToken { interface loginToken {
@ -28,11 +27,4 @@ declare namespace Auth {
/* 密码 */ /* 密码 */
password: string; password: string;
} }
// interface userRoutes {
// name: string;
// path: string;
// meta: AppRoute.RouteMeta;
// children?: userRoutes[];
// redirect: string;
// }
} }

View File

@ -0,0 +1,7 @@
<template>
<div text-center c-yellow>I prove that you have made the ju mp test2-deail.</div>
</template>
<script setup lang="ts"></script>
<style scoped></style>

View File

@ -0,0 +1,14 @@
<template>
<div text-center c-red>
I prove that you have made the jump test4.
<n-button strong secondary type="success" @click="testMsg">testMsg</n-button>
</div>
</template>
<script setup lang="ts">
const testMsg = () => {
window.$message.error('Once upon a time you dressed so fine');
};
</script>
<style scoped></style>