119 lines
3.2 KiB
Vue
Raw Normal View History

<script lang="ts" setup>
import { computed } from "vue"
2022-10-18 15:07:42 +08:00
import { type RouteRecordRaw } from "vue-router"
2022-08-25 16:26:28 +08:00
import SidebarItemLink from "./SidebarItemLink.vue"
import { isExternal } from "@/utils/validate"
import path from "path-browserify"
interface Props {
item: RouteRecordRaw
isCollapse?: boolean
isFirstLevel?: boolean
basePath?: string
}
const props = withDefaults(defineProps<Props>(), {
isCollapse: false,
isFirstLevel: true,
basePath: ""
})
2023-06-15 13:42:58 +08:00
/** 是否始终显示根菜单 */
const alwaysShowRootMenu = computed(() => props.item.meta?.alwaysShow)
/** 显示的子菜单 */
const showingChildren = computed(() => {
return props.item.children?.filter((child) => !child.meta?.hidden) ?? []
})
2022-08-25 16:26:28 +08:00
2023-06-15 13:42:58 +08:00
/** 显示的子菜单数量 */
const showingChildNumber = computed(() => {
2023-06-15 13:42:58 +08:00
return showingChildren.value.length
})
2022-08-25 16:26:28 +08:00
2023-06-15 13:42:58 +08:00
/** 唯一的子菜单项 */
const theOnlyOneChild = computed(() => {
2023-06-15 13:42:58 +08:00
const number = showingChildNumber.value
switch (true) {
case number > 1:
return null
case number === 1:
return showingChildren.value[0]
default:
return { ...props.item, path: "" }
}
})
2023-06-15 13:42:58 +08:00
/** 解析路径 */
const resolvePath = (routePath: string) => {
2023-06-15 13:42:58 +08:00
switch (true) {
case isExternal(routePath):
return routePath
case isExternal(props.basePath):
return props.basePath
default:
return path.resolve(props.basePath, routePath)
}
}
</script>
<template>
2022-08-25 16:26:28 +08:00
<div v-if="!props.item.meta?.hidden" :class="{ 'simple-mode': props.isCollapse, 'first-level': props.isFirstLevel }">
<template v-if="!alwaysShowRootMenu && theOnlyOneChild && !theOnlyOneChild.children">
<SidebarItemLink v-if="theOnlyOneChild.meta" :to="resolvePath(theOnlyOneChild.path)">
<el-menu-item :index="resolvePath(theOnlyOneChild.path)">
<svg-icon v-if="theOnlyOneChild.meta.svgIcon" :name="theOnlyOneChild.meta.svgIcon" />
2022-10-08 15:19:36 +08:00
<component v-else-if="theOnlyOneChild.meta.elIcon" :is="theOnlyOneChild.meta.elIcon" class="el-icon" />
<template v-if="theOnlyOneChild.meta.title" #title>
{{ theOnlyOneChild.meta.title }}
</template>
</el-menu-item>
</SidebarItemLink>
</template>
2023-02-20 14:51:34 +08:00
<el-sub-menu v-else :index="resolvePath(props.item.path)" teleported>
<template #title>
2023-06-15 13:42:58 +08:00
<svg-icon v-if="props.item.meta?.svgIcon" :name="props.item.meta.svgIcon" />
<component v-else-if="props.item.meta?.elIcon" :is="props.item.meta.elIcon" class="el-icon" />
<span v-if="props.item.meta?.title">{{ props.item.meta.title }}</span>
</template>
2022-08-25 16:26:28 +08:00
<template v-if="props.item.children">
<sidebar-item
2022-08-25 16:26:28 +08:00
v-for="child in props.item.children"
:key="child.path"
:item="child"
2022-08-25 16:26:28 +08:00
:is-collapse="props.isCollapse"
:is-first-level="false"
:base-path="resolvePath(child.path)"
/>
</template>
</el-sub-menu>
</div>
</template>
<style lang="scss" scoped>
.svg-icon {
min-width: 1em;
2022-10-08 15:19:36 +08:00
margin-right: 12px;
font-size: 18px;
}
2022-10-08 15:19:36 +08:00
.el-icon {
width: 1em;
2022-10-08 15:19:36 +08:00
margin-right: 12px;
font-size: 18px;
}
2022-10-08 15:19:36 +08:00
.simple-mode {
&.first-level {
:deep(.el-sub-menu) {
.el-sub-menu__icon-arrow {
display: none;
}
span {
visibility: hidden;
}
}
}
}
</style>