mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-05 12:44:27 +08:00
modify: support multi tabs
This commit is contained in:
parent
308e0b4bf9
commit
e7148081a9
@ -16,10 +16,7 @@ const appStore = useAppStore()
|
|||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
function handleTab(route: RouteLocationNormalized) {
|
function handleTab(route: RouteLocationNormalized) {
|
||||||
router.push(route.path)
|
router.push(route.fullPath)
|
||||||
}
|
|
||||||
function handleClose(path: string) {
|
|
||||||
tabStore.closeTab(path)
|
|
||||||
}
|
}
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
const options = computed(() => {
|
const options = computed(() => {
|
||||||
@ -71,16 +68,16 @@ function handleSelect(key: string) {
|
|||||||
appStore.reloadPage()
|
appStore.reloadPage()
|
||||||
},
|
},
|
||||||
closeCurrent() {
|
closeCurrent() {
|
||||||
tabStore.closeTab(currentRoute.value.path)
|
tabStore.closeTab(currentRoute.value.fullPath)
|
||||||
},
|
},
|
||||||
closeOther() {
|
closeOther() {
|
||||||
tabStore.closeOtherTabs(currentRoute.value.path)
|
tabStore.closeOtherTabs(currentRoute.value.fullPath)
|
||||||
},
|
},
|
||||||
closeLeft() {
|
closeLeft() {
|
||||||
tabStore.closeLeftTabs(currentRoute.value.path)
|
tabStore.closeLeftTabs(currentRoute.value.fullPath)
|
||||||
},
|
},
|
||||||
closeRight() {
|
closeRight() {
|
||||||
tabStore.closeRightTabs(currentRoute.value.path)
|
tabStore.closeRightTabs(currentRoute.value.fullPath)
|
||||||
},
|
},
|
||||||
closeAll() {
|
closeAll() {
|
||||||
tabStore.closeAllTabs()
|
tabStore.closeAllTabs()
|
||||||
@ -110,13 +107,13 @@ function onClickoutside() {
|
|||||||
size="small"
|
size="small"
|
||||||
:tabs-padding="10"
|
:tabs-padding="10"
|
||||||
:value="tabStore.currentTabPath"
|
:value="tabStore.currentTabPath"
|
||||||
@close="handleClose"
|
@close="tabStore.closeTab"
|
||||||
>
|
>
|
||||||
<n-tab
|
<n-tab
|
||||||
v-for="item in tabStore.pinTabs"
|
v-for="item in tabStore.pinTabs"
|
||||||
:key="item.path"
|
:key="item.fullPath"
|
||||||
:name="item.path"
|
:name="item.fullPath"
|
||||||
@click="router.push(item.path)"
|
@click="router.push(item.fullPath)"
|
||||||
>
|
>
|
||||||
<div class="flex-x-center gap-2">
|
<div class="flex-x-center gap-2">
|
||||||
<nova-icon :icon="item.meta.icon" /> {{ $t(`route.${String(item.name)}`, item.meta.title) }}
|
<nova-icon :icon="item.meta.icon" /> {{ $t(`route.${String(item.name)}`, item.meta.title) }}
|
||||||
@ -124,9 +121,9 @@ function onClickoutside() {
|
|||||||
</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.fullPath"
|
||||||
closable
|
closable
|
||||||
:name="item.path"
|
:name="item.fullPath"
|
||||||
@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.path as string)
|
tabStore.setCurrentTab(to.fullPath as string)
|
||||||
})
|
})
|
||||||
|
|
||||||
router.afterEach((to) => {
|
router.afterEach((to) => {
|
||||||
|
@ -58,7 +58,7 @@ export const staticRoutes: AppRoute.RowRoute[] = [
|
|||||||
{
|
{
|
||||||
name: 'multi2Detail',
|
name: 'multi2Detail',
|
||||||
path: '/multi/multi2/detail',
|
path: '/multi/multi2/detail',
|
||||||
title: '多级菜单的详情页',
|
title: '菜单详情页',
|
||||||
requiresAuth: true,
|
requiresAuth: true,
|
||||||
icon: 'icon-park-outline:list',
|
icon: 'icon-park-outline:list',
|
||||||
hide: true,
|
hide: true,
|
||||||
|
@ -24,7 +24,7 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
return
|
return
|
||||||
|
|
||||||
// 如果标签名称已存在则不添加
|
// 如果标签名称已存在则不添加
|
||||||
if (this.hasExistTab(route.path as string))
|
if (this.hasExistTab(route.fullPath as string))
|
||||||
return
|
return
|
||||||
|
|
||||||
// 根据meta.pinTab传递到不同的分组中
|
// 根据meta.pinTab传递到不同的分组中
|
||||||
@ -33,42 +33,42 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
else
|
else
|
||||||
this.tabs.push(route)
|
this.tabs.push(route)
|
||||||
},
|
},
|
||||||
async closeTab(path: string) {
|
async closeTab(fullPath: 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(path)
|
const index = this.getTabIndex(fullPath)
|
||||||
const isLast = index + 1 === tabsLength
|
const isLast = index + 1 === tabsLength
|
||||||
// 如果是关闭的当前页面,路由跳转到原先标签的后一个标签
|
// 如果是关闭的当前页面,路由跳转到原先标签的后一个标签
|
||||||
if (this.currentTabPath === path && !isLast) {
|
if (this.currentTabPath === fullPath && !isLast) {
|
||||||
// 跳转到后一个标签
|
// 跳转到后一个标签
|
||||||
router.push(this.tabs[index + 1].path)
|
router.push(this.tabs[index + 1].fullPath)
|
||||||
}
|
}
|
||||||
else if (this.currentTabPath === path && isLast) {
|
else if (this.currentTabPath === fullPath && isLast) {
|
||||||
// 已经是最后一个了,就跳转前一个
|
// 已经是最后一个了,就跳转前一个
|
||||||
router.push(this.tabs[index - 1].path)
|
router.push(this.tabs[index - 1].fullPath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 删除标签
|
// 删除标签
|
||||||
this.tabs = this.tabs.filter((item) => {
|
this.tabs = this.tabs.filter((item) => {
|
||||||
return item.path !== path
|
return item.fullPath !== fullPath
|
||||||
})
|
})
|
||||||
// 删除后如果清空了,就跳转到默认首页
|
// 删除后如果清空了,就跳转到默认首页
|
||||||
if (tabsLength - 1 === 0)
|
if (tabsLength - 1 === 0)
|
||||||
router.push('/')
|
router.push('/')
|
||||||
},
|
},
|
||||||
|
|
||||||
closeOtherTabs(path: string) {
|
closeOtherTabs(fullPath: string) {
|
||||||
const index = this.getTabIndex(path)
|
const index = this.getTabIndex(fullPath)
|
||||||
this.tabs = this.tabs.filter((item, i) => i === index)
|
this.tabs = this.tabs.filter((item, i) => i === index)
|
||||||
},
|
},
|
||||||
closeLeftTabs(path: string) {
|
closeLeftTabs(fullPath: string) {
|
||||||
const index = this.getTabIndex(path)
|
const index = this.getTabIndex(fullPath)
|
||||||
this.tabs = this.tabs.filter((item, i) => i >= index)
|
this.tabs = this.tabs.filter((item, i) => i >= index)
|
||||||
},
|
},
|
||||||
closeRightTabs(path: string) {
|
closeRightTabs(fullPath: string) {
|
||||||
const index = this.getTabIndex(path)
|
const index = this.getTabIndex(fullPath)
|
||||||
this.tabs = this.tabs.filter((item, i) => i <= index)
|
this.tabs = this.tabs.filter((item, i) => i <= index)
|
||||||
},
|
},
|
||||||
clearAllTabs() {
|
clearAllTabs() {
|
||||||
@ -80,21 +80,25 @@ export const useTabStore = defineStore('tab-store', {
|
|||||||
router.push('/')
|
router.push('/')
|
||||||
},
|
},
|
||||||
|
|
||||||
hasExistTab(path: string) {
|
hasExistTab(fullPath: string) {
|
||||||
const _tabs = [...this.tabs, ...this.pinTabs]
|
const _tabs = [...this.tabs, ...this.pinTabs]
|
||||||
return _tabs.some((item) => {
|
return _tabs.some((item) => {
|
||||||
return item.path === path
|
return item.fullPath === fullPath
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
/* 设置当前激活的标签 */
|
/* 设置当前激活的标签 */
|
||||||
setCurrentTab(path: string) {
|
setCurrentTab(fullPath: string) {
|
||||||
this.currentTabPath = path
|
this.currentTabPath = fullPath
|
||||||
},
|
},
|
||||||
getTabIndex(path: string) {
|
getTabIndex(fullPath: string) {
|
||||||
return this.tabs.findIndex((item) => {
|
return this.tabs.findIndex((item) => {
|
||||||
return item.path === path
|
return item.fullPath === fullPath
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
modifyTab(fullPath: string, modifyFn: (route: RouteLocationNormalized) => void) {
|
||||||
|
const index = this.getTabIndex(fullPath)
|
||||||
|
modifyFn(this.tabs[index])
|
||||||
|
},
|
||||||
},
|
},
|
||||||
persist: {
|
persist: {
|
||||||
storage: sessionStorage,
|
storage: sessionStorage,
|
||||||
|
@ -1,9 +1,25 @@
|
|||||||
<script setup lang="ts"></script>
|
<script setup lang="ts">
|
||||||
|
import { useTabStore } from '@/store'
|
||||||
|
|
||||||
|
const { modifyTab } = useTabStore()
|
||||||
|
|
||||||
|
const { fullPath, query } = useRoute()
|
||||||
|
|
||||||
|
modifyTab(fullPath, (target) => {
|
||||||
|
target.meta.title = `详情页${query.id}`
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
<n-space vertical>
|
||||||
<n-alert title="目前可公开的情报" type="warning">
|
<n-alert title="目前可公开的情报" type="warning">
|
||||||
这是详情子页,他不会出现在侧边栏,他其实是上个页面的同级,并不是下级,这个要注意
|
这是详情子页,他不会出现在侧边栏,他其实是上个页面的同级,并不是下级,这个要注意
|
||||||
</n-alert>
|
</n-alert>
|
||||||
|
|
||||||
|
<n-h2>
|
||||||
|
详情页id:{{ query.id }}
|
||||||
|
</n-h2>
|
||||||
|
</n-space>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
@ -5,8 +5,14 @@ const router = useRouter()
|
|||||||
<template>
|
<template>
|
||||||
<n-card class="h-130vh">
|
<n-card class="h-130vh">
|
||||||
这个页面包含了一个不在侧边菜单的详情页面
|
这个页面包含了一个不在侧边菜单的详情页面
|
||||||
<n-button @click="router.push('/multi/multi2/detail')">
|
<n-button @click="router.push({ path: '/multi/multi2/detail', query: { id: 1 } })">
|
||||||
跳转详情子页
|
跳转详情子页1
|
||||||
|
</n-button>
|
||||||
|
<n-button @click="router.push({ path: '/multi/multi2/detail', query: { id: 2 } })">
|
||||||
|
跳转详情子页2
|
||||||
|
</n-button>
|
||||||
|
<n-button @click="router.push({ path: '/multi/multi2/detail', query: { id: 3 } })">
|
||||||
|
跳转详情子页3
|
||||||
</n-button>
|
</n-button>
|
||||||
</n-card>
|
</n-card>
|
||||||
</template>
|
</template>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user