Vue冷門技巧遞歸組件實(shí)踐示例詳解
痛點(diǎn)引出
在平時(shí)的開發(fā)當(dāng)中,渲染側(cè)邊欄導(dǎo)航菜單有時(shí)會(huì)遇到過需要側(cè)邊欄有多層甚至無限層級(jí)的問題。此時(shí)更優(yōu)雅的方式便是使用遞歸組件

<el-menu>
<template slot="title">
<i class="el-icon-location"></i>
<span>菜單</span>
</template>
<el-submenu index="1-1">
<template slot="title">子菜單</template>
<el-menu-item index="1-1-1">子菜單選項(xiàng)1</el-menu-item>
<el-submenu index="1-1-2">
<template slot="title">子菜單的子菜單</template>
<el-menu-item index="1-1-2-1">子菜單的子菜單的選項(xiàng)1</el-menu-item>
</el-submenu>
</el-submenu>
</el-submenu>
</el-menu>
可以看到這部分重復(fù)的代碼可以完全抽離出來做單獨(dú)的組件。
<el-submenu>
<el-menu-item></el-menu-item>
...
</el-submenu>
那么問題來了,在一層當(dāng)中又有重復(fù)的submenu怎么辦?顯然這個(gè)層級(jí)是需要?jiǎng)討B(tài)生成的。比如:
<el-submenu>
<el-menu-item></el-menu-item>
...
<el-submenu>
<el-menu-item></el-menu-item>
...
<el-submenu>
<el-menu-item></el-menu-item>
...
//這里省略很多很多層
</el-submenu>
</el-submenu>
</el-submenu>
很明顯,這里需要可以用遞歸(recursive) 的思想來解決, 那么在template模版當(dāng)中有辦法做這樣的組件嗎?答案當(dāng)然是可以,template模版語法也是支持遞歸。
源碼中的體現(xiàn)
先找找源碼,我們?cè)趕rc/core/global-api.ts當(dāng)中找到initExtend函數(shù),這個(gè)函數(shù)是initGlobalAPI的一個(gè)執(zhí)行步驟,每個(gè)組件創(chuàng)建的時(shí)候都會(huì)去調(diào)用。

可以看到如果命中name,則會(huì)給自己的components的配置項(xiàng)當(dāng)中注冊(cè)自己,使得可以在編譯的時(shí)候可以識(shí)別到自己,從而在template模版語法當(dāng)中去使用。
組件示例封裝
首先定義數(shù)據(jù)結(jié)構(gòu)能描述這樣的菜單
[
{
id: '1',
title: '父菜單',
children:[
{id:'1-1',title:'子選項(xiàng)',children:[]},
{id:'1-1',title:'子菜單',children:[
{id:...,title:...,children:...},
...
]},
...
]
}
]
簡單點(diǎn)描述就是
interface item:{
id: string,
title: string,
children: item[] | []
}
item[]
然后開始封裝組件
// RescursiveMenu.vue
<template>
<el-submenu :index="menuItem.id" v-if="menuItem.children.length">
<template slot="title">{{ menuItem.title }}</template>
<template v-for="item in menuItem.children">
<RecursiveMenu :menuItem="item"/>
</template>
</el-submenu>
<el-menu-item v-else>{{ menuItem.title }}</el-menu-item>
</template>
<script>
export default {
name:"RecursiveMenu",
props: {
menuItem: Object
}
}
</script>
當(dāng)然這只是簡單示例demo,后續(xù)根據(jù)業(yè)務(wù)需求相信難不倒各位看官。
使用:
<el-submenu>
<template v-for="item in menuList.children">
<RecursiveMenu :menuItem="item" :key="item.id"/>
</template>
</el-submenu>
小擴(kuò)展
同樣的,vue也支持jsx/tsx語法 ,使用jsx則需要抽象需要重復(fù)的過程,封裝成渲染函數(shù)來實(shí)現(xiàn)遞歸,這里采用整個(gè)數(shù)組渲染過程抽象重復(fù),來實(shí)現(xiàn)遞歸。
//MyMenu.jsx
export default {
name:"RecursiveMenu",
props: {
menuList: Array,
dafault:()=>([])
},
render(){
const recursiveRender = (menuList)=>{
return menuList.map((menuItem)=>{
return menuItem.children.length > 0 ? (
<elSubmenu index="{menuItem.id}">
<div slot="title">{menuItem.title}</div>
{recursiveRender(menuItem.children)}
</elSubmenu>
):(
<elMenuItem key="{menuItem.id}">{ menuItem.title }</elMenuItem>
)
}
)
}
return (<elMenu>
{recursiveRender(this.menuList)}
</elMenu>
)
}
}
當(dāng)然,如果想用jsx復(fù)刻上訴template當(dāng)中抽象的邏輯,可以寫成這樣:
// RecursiveMenu.jsx
export default {
name:"RecursiveMenu",
props: {
menuItem: Object,
dafault:()=>({})
},
render(){
const recursiveRender = (menuItem)=>{
return menuItem.children.length > 0 ? (
<elSubmenu index="{menuItem.id}">
<div slot="title">{menuItem.title}</div>
{menuItem.children.map(children=>recursiveRender(children))}
</elSubmenu>
):(
<elMenuItem key="{menuItem.id}">{ menuItem.title }</elMenuItem>
)
}
return recursiveRender(this.menuItem)
}
}
總結(jié)
Vue當(dāng)中實(shí)現(xiàn)遞歸渲染,可以使用模版語法和jsx語法。而實(shí)現(xiàn)本質(zhì)上是抽象出重復(fù)的邏輯,以及找到遞歸退出點(diǎn)。
以上就是Vue冷門技巧遞歸組件實(shí)踐示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Vue 遞歸組件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Vue3項(xiàng)目中優(yōu)雅實(shí)現(xiàn)微信授權(quán)登錄的方法
用戶在微信端中訪問第三方網(wǎng)頁,可以通過微信網(wǎng)頁授權(quán)機(jī)制獲取用戶的基本信息,進(jìn)而實(shí)現(xiàn)所需要的業(yè)務(wù)邏輯,這篇文章主要給大家介紹了關(guān)于Vue3項(xiàng)目中優(yōu)雅實(shí)現(xiàn)微信授權(quán)登錄的相關(guān)資料,需要的朋友可以參考下2021-09-09
vue調(diào)試工具vue-devtools安裝及使用方法
本文主要介紹 vue的調(diào)試工具 vue-devtools 的安裝和使用,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2018-11-11
在vs code 中如何創(chuàng)建一個(gè)自己的 Vue 模板代碼
這篇文章主要介紹了在vs code 中如何創(chuàng)建一個(gè)自己的 Vue 模板代碼,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11
vue2.5.2使用http請(qǐng)求獲取靜態(tài)json數(shù)據(jù)的實(shí)例代碼
這篇文章主要介紹了vue2.5.2使用http請(qǐng)求獲取靜態(tài)json數(shù)據(jù)的實(shí)例代碼,需要的朋友可以參考下2018-02-02
Vue Spa切換頁面時(shí)更改標(biāo)題的實(shí)例代碼
本篇文章主要介紹了Vue Spa切換頁面時(shí)更改標(biāo)題的實(shí)例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-07-07

