mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-06 03:57:54 +08:00
feat(router): 完善路由显示和meta字段功能
This commit is contained in:
parent
dd408611de
commit
d0108abc9e
@ -35,7 +35,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'dashboard',
|
name: 'dashboard',
|
||||||
path: '/dashboard',
|
path: '/dashboard',
|
||||||
redirect: '/dashboard/workbench',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '仪表盘',
|
title: '仪表盘',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -65,7 +64,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'test',
|
name: 'test',
|
||||||
path: '/test',
|
path: '/test',
|
||||||
redirect: '/test/test1',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '多级菜单演示',
|
title: '多级菜单演示',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -76,7 +74,7 @@ const userRoutes = [
|
|||||||
name: 'test1',
|
name: 'test1',
|
||||||
path: '/test/test1',
|
path: '/test/test1',
|
||||||
meta: {
|
meta: {
|
||||||
title: '多级菜单1',
|
title: '接口功能测试',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:list',
|
icon: 'icon-park-outline:list',
|
||||||
},
|
},
|
||||||
@ -85,7 +83,7 @@ const userRoutes = [
|
|||||||
name: 'test2',
|
name: 'test2',
|
||||||
path: '/test/test2',
|
path: '/test/test2',
|
||||||
meta: {
|
meta: {
|
||||||
title: '多级菜单2',
|
title: '多级菜单子页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:list',
|
icon: 'icon-park-outline:list',
|
||||||
},
|
},
|
||||||
@ -94,7 +92,7 @@ const userRoutes = [
|
|||||||
name: 'test2_detail',
|
name: 'test2_detail',
|
||||||
path: '/test/test2/detail',
|
path: '/test/test2/detail',
|
||||||
meta: {
|
meta: {
|
||||||
title: '多级菜单2的详情页',
|
title: '多级菜单的详情页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:list',
|
icon: 'icon-park-outline:list',
|
||||||
hide: true,
|
hide: true,
|
||||||
@ -107,7 +105,7 @@ const userRoutes = [
|
|||||||
name: 'test3',
|
name: 'test3',
|
||||||
path: '/test/test3',
|
path: '/test/test3',
|
||||||
meta: {
|
meta: {
|
||||||
title: '多级菜单3',
|
title: '多级菜单',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:list',
|
icon: 'icon-park-outline:list',
|
||||||
},
|
},
|
||||||
@ -128,7 +126,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'list',
|
name: 'list',
|
||||||
path: '/list',
|
path: '/list',
|
||||||
redirect: '/list/commonList',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '列表页',
|
title: '列表页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -158,7 +155,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'plugin',
|
name: 'plugin',
|
||||||
path: '/plugin',
|
path: '/plugin',
|
||||||
redirect: '/plugin/charts',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '组件示例',
|
title: '组件示例',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -265,7 +261,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'docments',
|
name: 'docments',
|
||||||
path: '/docments',
|
path: '/docments',
|
||||||
redirect: '/docments/vue',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '外链文档',
|
title: '外链文档',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -305,7 +300,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'permission',
|
name: 'permission',
|
||||||
path: '/permission',
|
path: '/permission',
|
||||||
redirect: '/permission/permission',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '权限示例',
|
title: '权限示例',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -336,7 +330,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'error',
|
name: 'error',
|
||||||
path: '/error',
|
path: '/error',
|
||||||
redirect: '/error/403',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '异常页',
|
title: '异常页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
@ -350,6 +343,7 @@ const userRoutes = [
|
|||||||
title: '403页',
|
title: '403页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'carbon:error',
|
icon: 'carbon:error',
|
||||||
|
order: 3,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -359,6 +353,7 @@ const userRoutes = [
|
|||||||
title: '404页',
|
title: '404页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:error',
|
icon: 'icon-park-outline:error',
|
||||||
|
order: 2,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -368,6 +363,7 @@ const userRoutes = [
|
|||||||
title: '500页',
|
title: '500页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'carbon:data-error',
|
icon: 'carbon:data-error',
|
||||||
|
order: 1,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -375,7 +371,6 @@ const userRoutes = [
|
|||||||
{
|
{
|
||||||
name: 'setting',
|
name: 'setting',
|
||||||
path: '/setting',
|
path: '/setting',
|
||||||
redirect: '/setting/account',
|
|
||||||
meta: {
|
meta: {
|
||||||
title: '系统设置',
|
title: '系统设置',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
<template>
|
<template>
|
||||||
<n-breadcrumb class="px-4">
|
<n-breadcrumb class="px-4">
|
||||||
<n-breadcrumb-item v-for="(item, index) in routes" :key="index" @click="routerPush(item.path)">
|
<n-breadcrumb-item
|
||||||
|
v-for="(item, index) in routes"
|
||||||
|
:key="index"
|
||||||
|
@click="routerPush(item.path)"
|
||||||
|
>
|
||||||
<e-icon :icon="item.meta.icon" />
|
<e-icon :icon="item.meta.icon" />
|
||||||
{{ item.meta.title }}
|
{{ item.meta.title }}
|
||||||
</n-breadcrumb-item>
|
</n-breadcrumb-item>
|
||||||
@ -17,7 +21,7 @@ const router = useRouter();
|
|||||||
const routeStore = useRouteStore();
|
const routeStore = useRouteStore();
|
||||||
const { routerPush } = useAppRouter();
|
const { routerPush } = useAppRouter();
|
||||||
const routes = computed(() => {
|
const routes = computed(() => {
|
||||||
return routeStore.createBreadcrumbFromRoutes(router.currentRoute.value.name as string, routeStore.userRoutes);
|
return routeStore.createBreadcrumbFromRoutes(router.currentRoute.value.name as string);
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -1,16 +1,27 @@
|
|||||||
import { RouteRecordRaw } from 'vue-router';
|
import { RouteRecordRaw } from 'vue-router';
|
||||||
import { BasicLayout } from '@/layouts/index';
|
import { BasicLayout } from '@/layouts/index';
|
||||||
import { useRouteStore, useAuthStore } from '@/store';
|
import { useRouteStore } from '@/store';
|
||||||
import { usePermission } from '@/hooks'
|
import { usePermission } from '@/hooks';
|
||||||
|
|
||||||
// 引入所有页面
|
// 引入所有页面
|
||||||
const modules = import.meta.glob('../../views/**/*.vue');
|
const modules = import.meta.glob('../../views/**/*.vue');
|
||||||
|
|
||||||
|
/* 含有子级的路由重定向到第一个子级 */
|
||||||
|
function setRedirect(routes: AppRoute.Route[]) {
|
||||||
|
routes.forEach((route) => {
|
||||||
|
if (route.children) {
|
||||||
|
const nonHiddenChild = route.children.find((child) => !child.meta || !child.meta.hide);
|
||||||
|
if (nonHiddenChild) {
|
||||||
|
route.redirect = nonHiddenChild.path;
|
||||||
|
}
|
||||||
|
setRedirect(route.children);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
/* 路由树转换成一维数组 */
|
/* 路由树转换成一维数组 */
|
||||||
function FlatAuthRoutes(routes: AppRoute.Route[]) {
|
function FlatAuthRoutes(routes: AppRoute.Route[]) {
|
||||||
let result: AppRoute.Route[] = [];
|
let result: AppRoute.Route[] = [];
|
||||||
// 浅拷贝一层去降维,否则影响原数组
|
routes.forEach((item: AppRoute.Route) => {
|
||||||
const _routes = JSON.parse(JSON.stringify(routes));
|
|
||||||
_routes.forEach((item: any) => {
|
|
||||||
if (item.children) {
|
if (item.children) {
|
||||||
const temp = item.children || [];
|
const temp = item.children || [];
|
||||||
delete item.children;
|
delete item.children;
|
||||||
@ -26,8 +37,8 @@ function FlatAuthRoutes(routes: AppRoute.Route[]) {
|
|||||||
function filterPermissionRoutes(routes: AppRoute.Route[]) {
|
function filterPermissionRoutes(routes: AppRoute.Route[]) {
|
||||||
const { hasPermission } = usePermission();
|
const { hasPermission } = usePermission();
|
||||||
return routes.filter((route) => {
|
return routes.filter((route) => {
|
||||||
return hasPermission(route.meta.roles)
|
return hasPermission(route.meta.roles);
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function createCatheRoutes(routes: AppRoute.Route[]) {
|
function createCatheRoutes(routes: AppRoute.Route[]) {
|
||||||
@ -38,15 +49,19 @@ function createCatheRoutes(routes: AppRoute.Route[]) {
|
|||||||
.map((item) => item.name);
|
.map((item) => item.name);
|
||||||
}
|
}
|
||||||
export async function createDynamicRoutes(routes: AppRoute.Route[]) {
|
export async function createDynamicRoutes(routes: AppRoute.Route[]) {
|
||||||
|
/* 复制一层 */
|
||||||
|
let resultRouter = JSON.parse(JSON.stringify(routes));
|
||||||
|
/* 设置路由重定向到子级第一个 */
|
||||||
|
setRedirect(resultRouter);
|
||||||
// 数组降维成一维数组,然后删除所有的childen
|
// 数组降维成一维数组,然后删除所有的childen
|
||||||
const flatRoutes = FlatAuthRoutes(routes);
|
resultRouter = FlatAuthRoutes(resultRouter);
|
||||||
/* 路由权限过滤 */
|
/* 路由权限过滤 */
|
||||||
const permissionRoutes = filterPermissionRoutes(flatRoutes)
|
resultRouter = filterPermissionRoutes(resultRouter);
|
||||||
// 过滤需要缓存的路由name数组
|
// 过滤需要缓存的路由name数组
|
||||||
const routeStore = useRouteStore();
|
const routeStore = useRouteStore();
|
||||||
routeStore.cacheRoutes = createCatheRoutes(permissionRoutes);
|
routeStore.cacheRoutes = createCatheRoutes(resultRouter);
|
||||||
// 生成路由,有redirect的不需要引入文件
|
// 生成路由,有redirect的不需要引入文件
|
||||||
const mapRoutes = permissionRoutes.map((item: any) => {
|
resultRouter = resultRouter.map((item: any) => {
|
||||||
if (!item.redirect) {
|
if (!item.redirect) {
|
||||||
// 动态加载对应页面
|
// 动态加载对应页面
|
||||||
item['component'] = modules[`../../views${item.path}/index.vue`];
|
item['component'] = modules[`../../views${item.path}/index.vue`];
|
||||||
@ -66,6 +81,6 @@ export async function createDynamicRoutes(routes: AppRoute.Route[]) {
|
|||||||
children: [],
|
children: [],
|
||||||
};
|
};
|
||||||
// 根据角色过滤后的插入根路由中
|
// 根据角色过滤后的插入根路由中
|
||||||
appRootRoute.children = mapRoutes as unknown as RouteRecordRaw[];
|
appRootRoute.children = resultRouter as unknown as RouteRecordRaw[];
|
||||||
return appRootRoute;
|
return appRootRoute;
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
import { defineStore } from 'pinia';
|
import { defineStore } from 'pinia';
|
||||||
import { renderIcon,local } from '@/utils';
|
import { renderIcon, local } from '@/utils';
|
||||||
import { MenuOption } from 'naive-ui';
|
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';
|
import { staticRoutes } from '@/router/modules';
|
||||||
import { RouterLink } from 'vue-router'
|
import { RouterLink } from 'vue-router';
|
||||||
import { usePermission } from '@/hooks'
|
import { usePermission } from '@/hooks';
|
||||||
import { h } from 'vue'
|
import { h } from 'vue';
|
||||||
|
|
||||||
interface RoutesStatus {
|
interface RoutesStatus {
|
||||||
isInitAuthRoute: boolean;
|
isInitAuthRoute: boolean;
|
||||||
@ -38,7 +38,7 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
router.removeRoute('appRoot');
|
router.removeRoute('appRoot');
|
||||||
},
|
},
|
||||||
/* 根据当前路由的name生成面包屑数据 */
|
/* 根据当前路由的name生成面包屑数据 */
|
||||||
createBreadcrumbFromRoutes(routeName = '/', userRoutes: AppRoute.Route[]) {
|
createBreadcrumbFromRoutes(routeName = '/') {
|
||||||
const path: AppRoute.Route[] = [];
|
const path: AppRoute.Route[] = [];
|
||||||
// 筛选所有包含目标的各级路由组合成一维数组
|
// 筛选所有包含目标的各级路由组合成一维数组
|
||||||
const getPathfromRoutes = (routeName: string, userRoutes: AppRoute.Route[]) => {
|
const getPathfromRoutes = (routeName: string, userRoutes: AppRoute.Route[]) => {
|
||||||
@ -51,7 +51,7 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
getPathfromRoutes(routeName, userRoutes);
|
getPathfromRoutes(routeName, this.userRoutes);
|
||||||
return path;
|
return path;
|
||||||
},
|
},
|
||||||
/* 判断当前路由和子路由中是否存在为routeName的路由 */
|
/* 判断当前路由和子路由中是否存在为routeName的路由 */
|
||||||
@ -77,57 +77,87 @@ export const useRouteStore = defineStore('route-store', {
|
|||||||
/* 生成侧边菜单的数据 */
|
/* 生成侧边菜单的数据 */
|
||||||
createMenus(userRoutes: AppRoute.Route[]) {
|
createMenus(userRoutes: AppRoute.Route[]) {
|
||||||
this.userRoutes = userRoutes;
|
this.userRoutes = userRoutes;
|
||||||
this.menus = this.transformAuthRoutesToMenus(userRoutes);
|
|
||||||
|
let resultMenus = JSON.parse(JSON.stringify(userRoutes));
|
||||||
|
resultMenus = this.removeHiddenRoutes(resultMenus);
|
||||||
|
this.menus = this.transformAuthRoutesToMenus(resultMenus);
|
||||||
},
|
},
|
||||||
|
/** 过滤不需要显示的菜单 */
|
||||||
|
removeHiddenRoutes(routes: AppRoute.Route[]) {
|
||||||
|
return routes.filter((route) => {
|
||||||
|
if (route.meta && route.meta.hide) {
|
||||||
|
return false;
|
||||||
|
} else if (route.children) {
|
||||||
|
route.children = this.removeHiddenRoutes(route.children);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
//* 将返回的路由表渲染成侧边栏 */
|
//* 将返回的路由表渲染成侧边栏 */
|
||||||
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
|
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
|
||||||
|
return (
|
||||||
return userRoutes
|
userRoutes
|
||||||
/** 隐藏不需要显示的菜单 */
|
/** 过滤没有权限的侧边菜单 */
|
||||||
.filter((item) => {
|
.filter((item: AppRoute.Route) => {
|
||||||
return !item.meta.hide;
|
const { hasPermission } = usePermission();
|
||||||
})
|
return hasPermission(item.meta.roles);
|
||||||
.filter((item: AppRoute.Route) => {
|
})
|
||||||
const { hasPermission } = usePermission();
|
/** 根据order大小菜单排序 */
|
||||||
return hasPermission(item.meta.roles)
|
.sort((a, b) => {
|
||||||
})
|
if (a.meta && a.meta.order && b.meta && b.meta.order) {
|
||||||
/** 转换为侧边菜单数据结构 */
|
return a.meta.order - b.meta.order;
|
||||||
.map((item) => {
|
} else if (a.meta && a.meta.order) {
|
||||||
const target: MenuOption = {
|
return -1;
|
||||||
label: () =>
|
} else if (b.meta && b.meta.order) {
|
||||||
h(
|
return 1;
|
||||||
RouterLink,
|
} else {
|
||||||
{
|
return 0;
|
||||||
to: {
|
|
||||||
path: item.path
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{ default: () => item.meta.title }
|
|
||||||
),
|
|
||||||
key: item.path,
|
|
||||||
icon: renderIcon(item.meta.icon)
|
|
||||||
};
|
|
||||||
/** 判断子元素 */
|
|
||||||
if (item.children) {
|
|
||||||
const children = this.transformAuthRoutesToMenus(item.children);
|
|
||||||
// 只有子元素有且不为空时才添加
|
|
||||||
if (children.length !== 0) {
|
|
||||||
target.children = children;
|
|
||||||
}
|
}
|
||||||
}
|
})
|
||||||
return target;
|
/** 转换为侧边菜单数据结构 */
|
||||||
});
|
.map((item) => {
|
||||||
|
const target: MenuOption = {
|
||||||
|
label:
|
||||||
|
!item.children || item.children.length == 0
|
||||||
|
? () =>
|
||||||
|
h(
|
||||||
|
RouterLink,
|
||||||
|
{
|
||||||
|
to: {
|
||||||
|
path: item.path,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ default: () => item.meta.title }
|
||||||
|
)
|
||||||
|
: item.meta.title,
|
||||||
|
key: item.path,
|
||||||
|
icon: renderIcon(item.meta.icon),
|
||||||
|
};
|
||||||
|
/** 判断子元素 */
|
||||||
|
if (item.children) {
|
||||||
|
const children = this.transformAuthRoutesToMenus(item.children);
|
||||||
|
// 只有子元素有且不为空时才添加
|
||||||
|
if (children.length !== 0) {
|
||||||
|
target.children = children;
|
||||||
|
} else {
|
||||||
|
target.children = undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return target;
|
||||||
|
})
|
||||||
|
);
|
||||||
},
|
},
|
||||||
/* 初始化动态路由 */
|
/* 初始化动态路由 */
|
||||||
async initDynamicRoute() {
|
async initDynamicRoute() {
|
||||||
// 根据用户id来获取用户的路由
|
// 根据用户id来获取用户的路由
|
||||||
const userInfo = local.get('userInfo')
|
const userInfo = local.get('userInfo');
|
||||||
|
|
||||||
if (!userInfo||!userInfo.userId) {
|
if (!userInfo || !userInfo.userId) {
|
||||||
return
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { data: routes } = await fetchUserRoutes({ userId: userInfo.userId});
|
const { data: routes } = await fetchUserRoutes({ userId: userInfo.userId });
|
||||||
// 根据用户返回的路由表来生成真实路由
|
// 根据用户返回的路由表来生成真实路由
|
||||||
const appRoutes = await createDynamicRoutes(routes);
|
const appRoutes = await createDynamicRoutes(routes);
|
||||||
// 生成侧边菜单
|
// 生成侧边菜单
|
||||||
|
@ -1,7 +1,16 @@
|
|||||||
<template>
|
<template>
|
||||||
<n-card title="地图示例(keepalive缓存)">
|
<n-card title="地图示例(keepalive缓存)">
|
||||||
<n-tabs type="line" animated>
|
<n-tabs
|
||||||
<n-tab-pane v-for="item in maps" :key="item.id" :name="item.id" :tab="item.label" class="h-600px">
|
type="line"
|
||||||
|
animated
|
||||||
|
>
|
||||||
|
<n-tab-pane
|
||||||
|
v-for="item in maps"
|
||||||
|
:key="item.id"
|
||||||
|
:name="item.id"
|
||||||
|
:tab="item.label"
|
||||||
|
class="h-600px"
|
||||||
|
>
|
||||||
<component :is="item.component" />
|
<component :is="item.component" />
|
||||||
</n-tab-pane>
|
</n-tab-pane>
|
||||||
</n-tabs>
|
</n-tabs>
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div>
|
<div>
|
||||||
|
<n-h1>接口功能测试</n-h1>
|
||||||
<n-space>
|
<n-space>
|
||||||
<n-button
|
<n-button
|
||||||
strong
|
strong
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
<template>
|
<template>
|
||||||
<div text-center>
|
<div text-center>
|
||||||
I prove that you have made the ju mp test2.
|
这个页面包含了一个不在侧边菜单的详情页面
|
||||||
<n-button @click="routerPush('/test/test2/detail')">跳转详情子页</n-button>
|
<n-button @click="routerPush('/test/test2/detail')">
|
||||||
|
跳转详情子页
|
||||||
|
</n-button>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -1,7 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div></div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts"></script>
|
|
||||||
|
|
||||||
<style scoped></style>
|
|
@ -1,7 +1,17 @@
|
|||||||
<template>
|
<template>
|
||||||
<div text-center c-red>
|
<div
|
||||||
I prove that you have made the jump test4.
|
text-center
|
||||||
<n-button strong secondary type="success" @click="testMsg">testMsg</n-button>
|
c-red
|
||||||
|
>
|
||||||
|
三级菜单页
|
||||||
|
<n-button
|
||||||
|
strong
|
||||||
|
secondary
|
||||||
|
type="success"
|
||||||
|
@click="testMsg"
|
||||||
|
>
|
||||||
|
testMsg
|
||||||
|
</n-button>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user