2025-07-30 09:13:52 +08:00
|
|
|
|
<template>
|
|
|
|
|
<a-breadcrumb>
|
|
|
|
|
<transition-group name="breadcrumb">
|
|
|
|
|
<div v-for="(item, index) in breadcrumbList" :key="item.meta.title">
|
|
|
|
|
<a-breadcrumb-item v-bind="attrs">
|
|
|
|
|
<span
|
|
|
|
|
v-if="item.redirect === 'noRedirect' || item.redirect === '' || index === breadcrumbList.length - 1"
|
|
|
|
|
class="gi_line_1"
|
|
|
|
|
>{{ item.meta.title }}</span>
|
|
|
|
|
<span v-else class="gi_line_1 breadcrumb-item-title" @click="handleLink(item)">{{ item.meta.title }}</span>
|
|
|
|
|
<icon-right v-if="index !== breadcrumbList.length - 1" />
|
|
|
|
|
</a-breadcrumb-item>
|
|
|
|
|
</div>
|
|
|
|
|
</transition-group>
|
|
|
|
|
</a-breadcrumb>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
import { computed } from 'vue'
|
|
|
|
|
import type { RouteLocationMatched } from 'vue-router'
|
|
|
|
|
import { findTree } from 'xe-utils'
|
|
|
|
|
import { useRouteStore } from '@/stores'
|
|
|
|
|
|
|
|
|
|
const route = useRoute()
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
const { routes } = useRouteStore()
|
|
|
|
|
const attrs = useAttrs()
|
|
|
|
|
|
|
|
|
|
// 缓存home路由
|
|
|
|
|
const home = computed(() => {
|
|
|
|
|
const cloneRoutes = JSON.parse(JSON.stringify(routes)) as RouteLocationMatched[]
|
|
|
|
|
const obj = findTree(cloneRoutes, (i) => i.path === '/dashboard/workplace')
|
|
|
|
|
return obj?.item || null
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// 使用computed计算面包屑列表,避免重复计算
|
|
|
|
|
const breadcrumbList = computed(() => {
|
|
|
|
|
// 跳过重定向路由
|
|
|
|
|
if (route.path.startsWith('/redirect/')) {
|
|
|
|
|
return []
|
|
|
|
|
}
|
2025-08-04 17:10:11 +08:00
|
|
|
|
|
2025-07-30 09:13:52 +08:00
|
|
|
|
const cloneRoutes = JSON.parse(JSON.stringify(routes)) as RouteLocationMatched[]
|
|
|
|
|
const obj = findTree(cloneRoutes, (i) => i.path === route.path)
|
2025-08-04 17:10:11 +08:00
|
|
|
|
|
2025-07-30 09:13:52 +08:00
|
|
|
|
// 获取当前节点的所有上级节点集合,包含当前节点
|
|
|
|
|
const arr = obj ? obj.nodes.filter((item) => item.meta && item.meta.title && item.meta.breadcrumb !== false) : []
|
2025-08-04 17:10:11 +08:00
|
|
|
|
|
2025-07-30 09:13:52 +08:00
|
|
|
|
// 如果有home路由且不是重复的,则添加到开头
|
2025-08-04 17:10:11 +08:00
|
|
|
|
if (home.value && !arr.some(item => item.path === home.value?.path)) {
|
2025-07-30 09:13:52 +08:00
|
|
|
|
return [home.value, ...arr]
|
|
|
|
|
}
|
2025-08-04 17:10:11 +08:00
|
|
|
|
|
2025-07-30 09:13:52 +08:00
|
|
|
|
return arr
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// 路由跳转
|
|
|
|
|
function handleLink(item: RouteLocationMatched) {
|
|
|
|
|
const { redirect, path } = item
|
|
|
|
|
if (redirect) {
|
|
|
|
|
return router.push(redirect as string)
|
|
|
|
|
}
|
|
|
|
|
router.push(path)
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style scoped lang="scss">
|
|
|
|
|
/** breadcrumb-transform 面包屑动画 */
|
|
|
|
|
.breadcrumb-enter-active {
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breadcrumb-enter-from,
|
|
|
|
|
.breadcrumb-leave-active {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
transform: translateX(10px);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
:deep(.arco-breadcrumb-item) {
|
|
|
|
|
padding: 0;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
|
|
|
|
.arco-icon-right {
|
|
|
|
|
margin: 0 4px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breadcrumb-item-title {
|
|
|
|
|
transition: all 0.3s;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
&:hover {
|
|
|
|
|
color: $color-theme;
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
</style>
|