mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-06 03:57:54 +08:00
feat: add new route meta
This commit is contained in:
parent
35b4be9cfb
commit
dac65d9182
@ -10,8 +10,8 @@ const router = useRouter()
|
|||||||
function handleTab(route: RouteLocationNormalized) {
|
function handleTab(route: RouteLocationNormalized) {
|
||||||
router.push(route.path)
|
router.push(route.path)
|
||||||
}
|
}
|
||||||
function handleClose(name: string) {
|
function handleClose(path: string) {
|
||||||
tabStore.closeTab(name)
|
tabStore.closeTab(path)
|
||||||
}
|
}
|
||||||
const options = [
|
const options = [
|
||||||
{
|
{
|
||||||
@ -60,16 +60,16 @@ function handleSelect(key: string) {
|
|||||||
appStore.reloadPage()
|
appStore.reloadPage()
|
||||||
},
|
},
|
||||||
closeCurrent() {
|
closeCurrent() {
|
||||||
tabStore.closeTab(currentRoute.value.name)
|
tabStore.closeTab(currentRoute.value.path)
|
||||||
},
|
},
|
||||||
closeOther() {
|
closeOther() {
|
||||||
tabStore.closeOtherTabs(currentRoute.value.name)
|
tabStore.closeOtherTabs(currentRoute.value.path)
|
||||||
},
|
},
|
||||||
closeLeft() {
|
closeLeft() {
|
||||||
tabStore.closeLeftTabs(currentRoute.value.name)
|
tabStore.closeLeftTabs(currentRoute.value.path)
|
||||||
},
|
},
|
||||||
closeRight() {
|
closeRight() {
|
||||||
tabStore.closeRightTabs(currentRoute.value.name)
|
tabStore.closeRightTabs(currentRoute.value.path)
|
||||||
},
|
},
|
||||||
closeAll() {
|
closeAll() {
|
||||||
tabStore.closeAllTabs()
|
tabStore.closeAllTabs()
|
||||||
@ -98,22 +98,22 @@ function onClickoutside() {
|
|||||||
type="card"
|
type="card"
|
||||||
size="small"
|
size="small"
|
||||||
:tabs-padding="15"
|
:tabs-padding="15"
|
||||||
:value="tabStore.currentTab"
|
:value="tabStore.currentTabPath"
|
||||||
@close="handleClose"
|
@close="handleClose"
|
||||||
>
|
>
|
||||||
<n-tab
|
<n-tab
|
||||||
v-for="item in tabStore.inherentTab"
|
v-for="item in tabStore.pinTabs"
|
||||||
:key="item.path"
|
:key="item.path"
|
||||||
:name="item.name"
|
:name="item.path"
|
||||||
@click="router.push('/')"
|
@click="router.push(item.path)"
|
||||||
>
|
>
|
||||||
{{ item.title }}
|
{{ item.meta.title }}
|
||||||
</n-tab>
|
</n-tab>
|
||||||
<n-tab
|
<n-tab
|
||||||
v-for="item in tabStore.tabs"
|
v-for="item in tabStore.tabs"
|
||||||
:key="item.path"
|
:key="item.path"
|
||||||
closable
|
closable
|
||||||
:name="item.name as string"
|
:name="item.path as string"
|
||||||
@click="handleTab(item)"
|
@click="handleTab(item)"
|
||||||
@contextmenu="handleContextMenu($event, item)"
|
@contextmenu="handleContextMenu($event, item)"
|
||||||
>
|
>
|
||||||
|
@ -61,7 +61,7 @@ export function setupRouterGuard(router: Router) {
|
|||||||
// 添加tabs
|
// 添加tabs
|
||||||
tabStore.addTab(to)
|
tabStore.addTab(to)
|
||||||
// 设置高亮标签;
|
// 设置高亮标签;
|
||||||
tabStore.setCurrentTab(to.name as string)
|
tabStore.setCurrentTab(to.path as string)
|
||||||
})
|
})
|
||||||
|
|
||||||
router.afterEach((to) => {
|
router.afterEach((to) => {
|
||||||
|
@ -16,6 +16,7 @@ export const routes: RouteRecordRaw[] = [
|
|||||||
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
component: () => import('@/views/login/index.vue'), // 注意这里要带上 文件后缀.vue
|
||||||
meta: {
|
meta: {
|
||||||
title: '登录',
|
title: '登录',
|
||||||
|
withoutTab: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -24,7 +25,7 @@ export const routes: RouteRecordRaw[] = [
|
|||||||
component: () => import('@/views/error/403/index.vue'),
|
component: () => import('@/views/error/403/index.vue'),
|
||||||
meta: {
|
meta: {
|
||||||
title: '用户无权限',
|
title: '用户无权限',
|
||||||
icon: 'icon-park-outline:error',
|
withoutTab: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -34,6 +35,7 @@ export const routes: RouteRecordRaw[] = [
|
|||||||
meta: {
|
meta: {
|
||||||
title: '找不到页面',
|
title: '找不到页面',
|
||||||
icon: 'icon-park-outline:ghost',
|
icon: 'icon-park-outline:ghost',
|
||||||
|
withoutTab: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -43,6 +45,7 @@ export const routes: RouteRecordRaw[] = [
|
|||||||
meta: {
|
meta: {
|
||||||
title: '服务器错误',
|
title: '服务器错误',
|
||||||
icon: 'icon-park-outline:close-wifi',
|
icon: 'icon-park-outline:close-wifi',
|
||||||
|
withoutTab: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -52,6 +55,7 @@ export const routes: RouteRecordRaw[] = [
|
|||||||
meta: {
|
meta: {
|
||||||
title: '找不到页面',
|
title: '找不到页面',
|
||||||
icon: 'icon-park-outline:ghost',
|
icon: 'icon-park-outline:ghost',
|
||||||
|
withoutTab: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -37,7 +37,7 @@ export const useAuthStore = defineStore('auth-store', {
|
|||||||
routeStore.resetRouteStore()
|
routeStore.resetRouteStore()
|
||||||
// 清空标签栏数据
|
// 清空标签栏数据
|
||||||
const tabStore = useTabStore()
|
const tabStore = useTabStore()
|
||||||
tabStore.tabs.length = 0
|
tabStore.clearAllTabs()
|
||||||
// 重制当前存储库
|
// 重制当前存储库
|
||||||
this.$reset()
|
this.$reset()
|
||||||
// 重定向到登录页
|
// 重定向到登录页
|
||||||
|
@ -2,108 +2,94 @@ import type { RouteLocationNormalized } from 'vue-router'
|
|||||||
import { router } from '@/router'
|
import { router } from '@/router'
|
||||||
|
|
||||||
interface TabState {
|
interface TabState {
|
||||||
inherentTab: {
|
pinTabs: RouteLocationNormalized[]
|
||||||
name: string
|
|
||||||
title: string
|
|
||||||
path: string
|
|
||||||
}[]
|
|
||||||
tabs: RouteLocationNormalized[]
|
tabs: RouteLocationNormalized[]
|
||||||
tabWhiteList: string[]
|
currentTabPath: string
|
||||||
currentTab: string
|
|
||||||
}
|
}
|
||||||
export const useTabStore = defineStore('tab-store', {
|
export const useTabStore = defineStore('tab-store', {
|
||||||
state: (): TabState => {
|
state: (): TabState => {
|
||||||
return {
|
return {
|
||||||
inherentTab: [
|
pinTabs: [],
|
||||||
{
|
|
||||||
name: 'dashboard_workbench',
|
|
||||||
title: '工作台',
|
|
||||||
path: '/',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
tabs: [],
|
tabs: [],
|
||||||
tabWhiteList: ['404', '403', '500', 'login'],
|
currentTabPath: '',
|
||||||
currentTab: 'dashboard_workbench',
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
getters: {
|
|
||||||
inherentTabName(): string[] {
|
|
||||||
return this.inherentTab.map((item) => {
|
|
||||||
return item.name
|
|
||||||
})
|
|
||||||
},
|
|
||||||
},
|
|
||||||
actions: {
|
actions: {
|
||||||
addTab(route: RouteLocationNormalized) {
|
addTab(route: RouteLocationNormalized) {
|
||||||
// 如果已经在固有标签里则不添加
|
// 根据meta确定是否不添加,可用于错误页,登录页等
|
||||||
if (this.inherentTabName.includes(route.name as string))
|
if (route.meta.withoutTab)
|
||||||
return
|
return
|
||||||
|
|
||||||
// 如果标签名称已存在则不添加
|
// 如果标签名称已存在则不添加
|
||||||
if (this.hasExistTab(route.name as string))
|
if (this.hasExistTab(route.path as string))
|
||||||
return
|
|
||||||
|
|
||||||
// 如果在白名单内则不添加,错误页等
|
|
||||||
if (this.tabWhiteList.includes(route.name as string))
|
|
||||||
return
|
return
|
||||||
|
|
||||||
|
// 根据meta.pinTab传递到不同的分组中
|
||||||
|
if (route.meta.pinTab)
|
||||||
|
this.pinTabs.push(route)
|
||||||
|
else
|
||||||
this.tabs.push(route)
|
this.tabs.push(route)
|
||||||
},
|
},
|
||||||
async closeTab(name: string) {
|
async closeTab(path: string) {
|
||||||
const tabsLength = this.tabs.length
|
const tabsLength = this.tabs.length
|
||||||
// 如果动态标签大于一个,才会标签跳转
|
// 如果动态标签大于一个,才会标签跳转
|
||||||
if (this.tabs.length > 1) {
|
if (this.tabs.length > 1) {
|
||||||
// 获取关闭的标签索引
|
// 获取关闭的标签索引
|
||||||
const index = this.getTabIndex(name)
|
const index = this.getTabIndex(path)
|
||||||
const isLast = index + 1 === tabsLength
|
const isLast = index + 1 === tabsLength
|
||||||
// 如果是关闭的当前页面,路由跳转到原先标签的后一个标签
|
// 如果是关闭的当前页面,路由跳转到原先标签的后一个标签
|
||||||
if (this.currentTab === name && !isLast) {
|
if (this.currentTabPath === path && !isLast) {
|
||||||
// 跳转到后一个标签
|
// 跳转到后一个标签
|
||||||
router.push(this.tabs[index + 1].path)
|
router.push(this.tabs[index + 1].path)
|
||||||
}
|
}
|
||||||
else if (this.currentTab === name && isLast) {
|
else if (this.currentTabPath === path && isLast) {
|
||||||
// 已经是最后一个了,就跳转前一个
|
// 已经是最后一个了,就跳转前一个
|
||||||
router.push(this.tabs[index - 1].path)
|
router.push(this.tabs[index - 1].path)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 删除标签
|
// 删除标签
|
||||||
this.tabs = this.tabs.filter((item) => {
|
this.tabs = this.tabs.filter((item) => {
|
||||||
return item.name !== name
|
return item.path !== path
|
||||||
})
|
})
|
||||||
// 删除后如果清空了,就跳转到默认首页
|
// 删除后如果清空了,就跳转到默认首页
|
||||||
if (tabsLength - 1 === 0)
|
if (tabsLength - 1 === 0)
|
||||||
router.push('/')
|
router.push('/')
|
||||||
},
|
},
|
||||||
|
|
||||||
closeOtherTabs(name: string) {
|
closeOtherTabs(path: string) {
|
||||||
const index = this.getTabIndex(name)
|
const index = this.getTabIndex(path)
|
||||||
this.tabs = this.tabs.filter((item, i) => i === index)
|
this.tabs = this.tabs.filter((item, i) => i === index)
|
||||||
},
|
},
|
||||||
closeLeftTabs(name: string) {
|
closeLeftTabs(path: string) {
|
||||||
const index = this.getTabIndex(name)
|
const index = this.getTabIndex(path)
|
||||||
this.tabs = this.tabs.filter((item, i) => i >= index)
|
this.tabs = this.tabs.filter((item, i) => i >= index)
|
||||||
},
|
},
|
||||||
closeRightTabs(name: string) {
|
closeRightTabs(path: string) {
|
||||||
const index = this.getTabIndex(name)
|
const index = this.getTabIndex(path)
|
||||||
this.tabs = this.tabs.filter((item, i) => i <= index)
|
this.tabs = this.tabs.filter((item, i) => i <= index)
|
||||||
},
|
},
|
||||||
async closeAllTabs() {
|
clearAllTabs() {
|
||||||
|
this.tabs.length = 0
|
||||||
|
this.pinTabs.length = 0
|
||||||
|
},
|
||||||
|
closeAllTabs() {
|
||||||
this.tabs.length = 0
|
this.tabs.length = 0
|
||||||
router.push('/')
|
router.push('/')
|
||||||
},
|
},
|
||||||
|
|
||||||
hasExistTab(name: string) {
|
hasExistTab(path: string) {
|
||||||
return this.tabs.some((item) => {
|
const _tabs = [...this.tabs, ...this.pinTabs]
|
||||||
return item.name === name
|
return _tabs.some((item) => {
|
||||||
|
return item.path === path
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
/* 设置当前激活的标签 */
|
/* 设置当前激活的标签 */
|
||||||
setCurrentTab(name: string) {
|
setCurrentTab(path: string) {
|
||||||
this.currentTab = name
|
this.currentTabPath = path
|
||||||
},
|
},
|
||||||
getTabIndex(name: string) {
|
getTabIndex(path: string) {
|
||||||
return this.tabs.findIndex((item) => {
|
return this.tabs.findIndex((item) => {
|
||||||
return item.name === name
|
return item.path === path
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
4
src/typings/route.d.ts
vendored
4
src/typings/route.d.ts
vendored
@ -19,6 +19,10 @@ declare namespace AppRoute {
|
|||||||
herf?: string
|
herf?: string
|
||||||
/** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */
|
/** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */
|
||||||
activeMenu?: string
|
activeMenu?: string
|
||||||
|
/** 当前路由是否会被添加到Tab中 */
|
||||||
|
withoutTab?: boolean
|
||||||
|
/** 当前路由是否会被固定在Tab中,用于一些常驻页面 */
|
||||||
|
pinTab?: boolean
|
||||||
}
|
}
|
||||||
/** 单个路由的类型结构(动态路由模式:后端返回此类型结构的路由) */
|
/** 单个路由的类型结构(动态路由模式:后端返回此类型结构的路由) */
|
||||||
interface baseRoute {
|
interface baseRoute {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user