feat: add new route meta

This commit is contained in:
chansee97 2024-03-17 11:34:59 +08:00
parent 35b4be9cfb
commit dac65d9182
6 changed files with 59 additions and 65 deletions

View File

@ -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)"
> >

View File

@ -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) => {

View File

@ -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,
}, },
}, },

View File

@ -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()
// 重定向到登录页 // 重定向到登录页

View File

@ -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
}) })
}, },
}, },

View File

@ -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 {