您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“Vue3 Element-plus和el-menu無限級(jí)菜單組件怎么封裝”的有關(guān)知識(shí),在實(shí)際案例的操作過程中,不少人都會(huì)遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
對(duì)于element中提供給我們的el-menu組件最多可以實(shí)現(xiàn)三層嵌套,如果多一層數(shù)據(jù)只能自己通過變量去加一層,如果加了兩層、三層這種往往是行不通的,所以只能進(jìn)行封裝
MenuData.ts
export default [ { id: "1", name: "第一級(jí)菜單", level: '1', child: [ { id: "11", name: "第二級(jí)菜單", level: '1-1', child: [ { id: "111", name: "第三級(jí)菜單", level: '1-1-1', child: [ { id: "1111", name: "第四級(jí)菜單", level: '1-1-1-1', child: [ { id: "11111", name: "第五級(jí)菜單", level: '1-1-1-1-1', child: [] } ] } ] }] } ] }, { id: "2", name: "第一級(jí)同級(jí)菜單", level: '2', child: [] } ]
封裝思想:
1.對(duì)本身組件進(jìn)行循環(huán)使用,如果有子集使用本身組件 把child數(shù)據(jù)傳給自己
2.如果沒有子集 使用 el-menu-item
以下代碼對(duì)setup( )函數(shù)和setup語法糖分別做了實(shí)現(xiàn)
setup語法糖
<template> <el-menu :default-active="defaultActive" :unique-opened="true" class="el-menu-vertical-demo" > <template v-for="item in menu"> <!-- 如果有子集 --> <template v-if="item.child && item.child.length > 0"> <el-sub-menu :key="item.id" :index="item.level" :disabled="item.meta?.disabled" :popper-append-to-body="false" > <template #title> <i :class="[item.meta?.icon]"></i> <!-- 添加空格 表示下級(jí)--> <span> {{ generateSpaces(item.level) }} </span> <span slot="title"> {{ item.name }}</span> </template> <MenuTree :menu="item.child" :defaultActive="defaultActive" @clickItem="clickItemHandle" /> </el-sub-menu> </template> <!-- 如果沒有子集 --> <template v-else> <el-menu-item :key="item.id" :index="item.level" :disabled="item.meta?.disabled" :popper-append-to-body="false" @click="clickItemHandle(item)" > <i :class="[item.meta?.icon]"></i> <!-- 添加空格 表示下級(jí)--> <span> {{ generateSpaces(item.level) }} </span> <span slot="title">{{ item.name }}</span> </el-menu-item> </template> </template> </el-menu> </template> <script lang="ts" name="MenuTree" setup> // 把下面代碼變成setup語法糖的形式 import type { PropType } from "vue"; import type { MenuItem } from "@/types/lesson"; // type 為了方便寫成這樣 可以根據(jù)自己項(xiàng)目設(shè)定type defineProps({ menu: { type: Array as unknown as PropType<any[]>, required: true, default: () => [], }, defaultActive: { type: String as unknown as PropType<string>, required: true, default: [], }, }); const emit = defineEmits(["update-active-path", "clickItem"]); // 返回的空格字符串 用于顯示菜單層級(jí) const generateSpaces = (level: string) => { let str = ""; level.split("") .filter((it) => it != "-") .forEach(() => { str += " "; }); return str; }; // 點(diǎn)擊當(dāng)前菜單項(xiàng) const clickItemHandle = (item: MenuItem) => { emit("clickItem", item); }; </script> <style scoped lang="less"> .el-menu { width: 288px; } </style>
setup函數(shù)
<template> <el-menu :default-active="defaultActive" :unique-opened="true" class="el-menu-vertical-demo" > <template v-for="item in menu"> <template v-if="item.child && item.child.length > 0"> <el-sub-menu :key="item.id" :index="item.level" :disabled="item.meta?.disabled" :popper-append-to-body="false" > <template #title> <i :class="[item.meta?.icon]"></i> <!-- 添加空格 表示下級(jí)--> <span> {{ generateSpaces(item.level) }} </span> <span slot="title"> {{ item.name }}</span> </template> <MenuTree :menu="item.child" :defaultActive="defaultActive" @clickItem="clickItemHandle" /> </el-sub-menu> </template> <template v-else> <el-menu-item :key="item.id" :index="item.level" :disabled="item.meta?.disabled" :popper-append-to-body="false" @click="clickItemHandle(item)" > <i :class="[item.meta?.icon]"></i> <!-- 添加空格 表示下級(jí)--> <span> {{ generateSpaces(item.level) }} </span> <span slot="title">{{ item.name }}</span> </el-menu-item> </template> </template> </el-menu> </template> <script lang="ts"> import { defineComponent, toRefs } from 'vue'; import type { PropType } from 'vue' import type {MenuItem} from '@/types/lesson' export default defineComponent({ name: 'MenuTree', props: { menu: { type: Array as unknown as PropType<any[]>, required: true, default: () => [], }, defaultActive: { type: String as unknown as PropType<string>, required: true, default: '', }, }, emits: ['update-active-path','clickItem'], setup(props, context) { const { menu, defaultActive } = toRefs(props); const generateSpaces = (level:string) => { let str = '' level.split('').filter(it=>it!='-').forEach(() => { str += ' ' }) return str } const clickItemHandle = (item:MenuItem) => { context.emit('clickItem', item) } return { clickItemHandle, menu, defaultActive, generateSpaces, } }, }); </script> <style scoped lang="less"> .el-menu { width: 288px; } </style>
type就不補(bǔ)充了 可根據(jù)自己項(xiàng)目定義,可臨時(shí)改成any
<template> <MenuTree :menu="menuList" :defaultActive="defaultActive" @clickItem="handleMenuClick" :update-click="handleMenuClick" /> </template> <script setup lang="ts"> import MenuTree from "./components/MenuTree.vue"; import type {MenuItem} from '@/types/lesson' import menuData from './MenuData' const defaultActive = ref<string>(''); // "1-1-1-1" 默認(rèn)選中的數(shù)據(jù) const menuList = ref(menuData) const handleMenuClick = (item:MenuItem) => { console.log('父組件',item); }; </script>
補(bǔ)充default-active變量,如果一開始想默認(rèn)點(diǎn)開第一層的數(shù)據(jù) 就需要找規(guī)律啦
拿到所有的level,通過接口方式返給你 自己平鋪拿到所有的level也好
例如數(shù)據(jù)格式:
let arr = [ "1-1", "1-1-1", "1-1-1-1", "1-1-1-2", "1-1-1-3", "1-1-1-4", "1-1-1-5", "1-1-1-6", "1-1-2", "1-1-2-1" ]
想要的結(jié)果就是 最長(zhǎng)且相同數(shù)字最多的元素 1-1-1-1
arr.sort((a,b)=> b.split('-').length - a.split('-').length)[0]
使用split防止有些字符串是10、11 兩位數(shù)字的
“Vue3 Element-plus和el-menu無限級(jí)菜單組件怎么封裝”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。