vue3结合element-plus封装ems菜单栏组件
效果如下:ok,开始代码nav-menu.vue<template><div class="nav-menu"><div class="logo"><img class="img" src="~@/assets/img/logo.svg" alt="logo" /><span v-if="!collapse" class="title">
·
效果如下:

ok,开始代码
nav-menu.vue
<template>
<div class="nav-menu">
<div class="logo">
<img class="img" src="~@/assets/img/logo.svg" alt="logo" />
<span v-if="!collapse" class="title">Vue3+TS</span>
</div>
<el-menu
:default-active="defaultValue"
class="el-menu-vertical"
:collapse="collapse"
background-color="#0c2135"
text-color="#b7bdc3"
active-text-color="#0a60bd"
>
<template v-for="item in userMenus" :key="item.id">
<!-- 二级菜单 -->
<template v-if="item.type === 1">
<!-- 二级菜单的可以展开的标题 -->
<el-submenu :index="item.id + ''">
<template #title>
<i v-if="item.icon" :class="item.icon"></i>
<span>{{ item.name }}</span>
</template>
<!-- 遍历里面的item -->
<template v-for="subitem in item.children" :key="subitem.id">
<el-menu-item
:index="subitem.id + ''"
@click="handleMenuItemClick(subitem)"
>
<i v-if="subitem.icon" :class="subitem.icon"></i>
<span>{{ subitem.name }}</span>
</el-menu-item>
</template>
</el-submenu>
</template>
<!-- 一级菜单 -->
<template v-else-if="item.type === 2">
<el-menu-item :index="item.id + ''">
<i v-if="item.icon" :class="item.icon"></i>
<span>{{ item.name }}</span>
</el-menu-item>
</template>
</template>
</el-menu>
</div>
</template>
<script lang="ts">
import { defineComponent, computed, ref } from 'vue'
import { useStore } from '@/store'
import { useRouter, useRoute } from 'vue-router'
import { pathMapToMenu } from '@/utils/map-menus'
// vuex - typescript => pinia
export default defineComponent({
props: {
collapse: {
type: Boolean,
default: false
}
},
setup() {
// store
const store = useStore()
const userMenus = computed(() => store.state.login.userMenus)
// router
const router = useRouter()
const route = useRoute()
const currentPath = route.path
// data
const menu = pathMapToMenu(userMenus.value, currentPath)
const defaultValue = ref(menu.id + '')
// event handle
const handleMenuItemClick = (item: any) => {
router.push({
path: item.url ?? '/not-found'
})
}
return {
userMenus,
defaultValue,
handleMenuItemClick
}
}
})
</script>
<style scoped lang="less">
.nav-menu {
height: 100%;
background-color: #001529;
.logo {
display: flex;
height: 28px;
padding: 12px 10px 8px 10px;
flex-direction: row;
justify-content: flex-start;
align-items: center;
.img {
height: 100%;
margin: 0 10px;
}
.title {
font-size: 16px;
font-weight: 700;
color: white;
}
}
.el-menu {
border-right: none;
}
// 目录
.el-submenu {
background-color: #001529 !important;
// 二级菜单 ( 默认背景 )
.el-menu-item {
padding-left: 50px !important;
background-color: #0c2135 !important;
}
}
::v-deep .el-submenu__title {
background-color: #001529 !important;
}
// hover 高亮
.el-menu-item:hover {
color: #fff !important; // 菜单
}
.el-menu-item.is-active {
color: #fff !important;
background-color: #0a60bd !important;
}
}
.el-menu-vertical:not(.el-menu--collapse) {
width: 100%;
height: calc(100% - 48px);
}
</style>
userMenus是后台请求回来的菜单栏接口数据

pathMapToMenu方法
export function pathMapToMenu(
userMenus: any[],
currentPath: string,
breadcrumbs?: IBreadcrumb[]
): any {
for (const menu of userMenus) {
if (menu.type === 1) {
const findMenu = pathMapToMenu(menu.children ?? [], currentPath)
if (findMenu) {
breadcrumbs?.push({ name: menu.name })
breadcrumbs?.push({ name: findMenu.name })
return findMenu
}
} else if (menu.type === 2 && menu.url === currentPath) {
return menu
}
}
}
使用呢??
<el-aside :width="isCollapse ? '60px' : '210px'">
<nav-menu :collapse="isCollapse" />
</el-aside>
如果不明白的可以去github里面有所有代码,github项目地址https://github.com/lsh555/vue3-ems
更多推荐


所有评论(0)