手把手教你實(shí)現(xiàn)vue下拉菜單組件

這篇文章我們一起來實(shí)現(xiàn)一個(gè)vue的下拉菜單組件。
像這種基本UI組件,網(wǎng)上已經(jīng)有很多了,為什么要自己實(shí)現(xiàn)呢?其實(shí)并不是有意重復(fù)造輪子,而是想通過這個(gè)過程回顧一下vue組件開發(fā)的一些細(xì)節(jié)和注意事項(xiàng)。

為什么選擇下拉菜單組件?
因?yàn)椋郝槿鸽m小五臟俱全,這個(gè)小小的組件涉及到了不少vue組件開發(fā)的知識(shí)點(diǎn)。
好了,那就開始吧!

首先創(chuàng)建一個(gè)vue-cli的項(xiàng)目,筆者用的是vue-cli3,創(chuàng)建過程略,然后創(chuàng)建一個(gè)vue組件:DropDownList.vue

在編寫模板之前,我們來分析一下這個(gè)組件的視圖結(jié)構(gòu)和功能。
下拉菜單組件應(yīng)該由兩部分組成:

  • 選中項(xiàng)的文本
  • 待選菜單(默認(rèn)隱藏)

它的主要功能包括:

  • 鼠標(biāo)經(jīng)過下拉菜單組件,顯示待選菜單
  • 鼠標(biāo)滑出下拉菜單組件,隱藏待選菜單
  • 鼠標(biāo)點(diǎn)擊待選菜單中的條目,選中項(xiàng)文本更新,組件派發(fā)change事件
    我們編寫如下這樣的模板:
<template>
  <div class="zq-drop-list" @mouseover="onDplOver($event)" @mouseout="onDplOut($event)">
        <span>選中項(xiàng)的文本<i></i></span>
        <ul>
            <li>北京</li>
            <li>上海</li>
            <li>廣州</li>
        </ul>
    </div>
</template>

選中項(xiàng)文本右側(cè)的i標(biāo)簽,用來實(shí)現(xiàn)下拉菜單的三角形圖標(biāo),在下文的css中我們用背景圖來實(shí)現(xiàn)。
我們給根元素div已經(jīng)添加了鼠標(biāo)經(jīng)過和滑出的回調(diào)函數(shù),具體實(shí)現(xiàn)見下文。

接下來我們?yōu)檫@個(gè)下拉菜單編寫樣式,在模板下方添加style標(biāo)簽,為了防止和其他組件的樣式發(fā)生沖突,筆者建議大家在開發(fā)組件時(shí),都給style加上scoped屬性。另外,筆者在這里用到了scss,具體代碼如下:

<style scoped lang="scss">
    .zq-drop-list{
        display: inline-block;
        min-width: 100px;
        position: relative;
        span{
            display: block;
            height: 30px;
            line-height: 30px;
            background: #f1f1f1;
            font-size: 14px;
            text-align: center;
            color: #333333;
            border-radius: 4px;
            i{
                background: url(https://www.easyicon.net/api/resizeApi.php?id=1189852&size=16) no-repeat center center;
                margin-left: 6px;
                display: inline-block;
            }
        }
        ul{
            position: absolute;
            top: 30px;
            left: 0;
            width: 100%;
            margin: 0;
            padding: 0;
            border: solid 1px #f1f1f1;
            border-radius: 4px;
            overflow: hidden;
            li{
                list-style: none;
                height: 30px;
                line-height: 30px;
                font-size: 14px;
                border-bottom: solid 1px #f1f1f1;
                background: #ffffff;
            }
            li:last-child{
                border-bottom: none;
            }
            li:hover{
                background: #f6f6f6;
            }
        }
    }
</style>

關(guān)于樣式,這里就不詳細(xì)展開了,只說其中幾個(gè)需要注意的點(diǎn):

  • 那個(gè)i元素的樣式,我用到了一個(gè)網(wǎng)絡(luò)圖片,大家可以自行更換
  • 待選菜單ul在css里并沒有讓它隱藏,因?yàn)槲覀円ㄟ^js來控制,具體原因見下文
  • 待選菜單ul使用了絕對(duì)定位,因?yàn)楫?dāng)它展開的時(shí)候,不應(yīng)該影響頁面上其他元素的布局

現(xiàn)在這個(gè)組件大概長這個(gè)樣子:


1.png

我們繼續(xù)為這個(gè)組件定義屬性,很顯然,待選菜單應(yīng)該作為屬性傳進(jìn)來,一定不能是內(nèi)部寫死的,屬性定義如下:

<script>
export default {
    name: "DropDownList",
    props:{
        dataList:{
            type:Array,
            default(){
                return [
                    {name: "選項(xiàng)一"},
                    {name: "選項(xiàng)二"}
                ]
            }
        },
        labelProperty:{
            type:String,
            default(){ return "name" }
        }
    },
    data(){
        return {
            activeIndex:0
        }
    },
}

其中dataList就是待選菜單的數(shù)據(jù)源屬性,這里我們給這個(gè)屬性定義了默認(rèn)值,這也是筆者建議大家養(yǎng)成的一個(gè)習(xí)慣,作為一個(gè)組件,最好有默認(rèn)值,因?yàn)楫?dāng)別人使用你的組件時(shí),可以先不設(shè)置相關(guān)屬性,就能看到一個(gè)成品的效果,也能快速查看你這個(gè)組件所需屬性的數(shù)據(jù)細(xì)節(jié)。
另外一個(gè)屬性是labelProperty,這個(gè)屬性的作用是什么?我們實(shí)際項(xiàng)目中的數(shù)據(jù)源,并不一定都含有name這個(gè)字段,因此就可能導(dǎo)致下拉菜單無法渲染數(shù)據(jù)的文本,于是我們定義了這個(gè)屬性用來指定實(shí)際數(shù)據(jù)源渲染文本的字段,這個(gè)字段必須是字符串。這個(gè)屬性的默認(rèn)值是name,因?yàn)樗枰湍J(rèn)數(shù)據(jù)源保持一致。相信你還看到了一個(gè)組件內(nèi)部數(shù)據(jù),activeIndex,這個(gè)是用來表示當(dāng)前選中項(xiàng)的索引的,我們后面會(huì)用到。

現(xiàn)在我們就可以在其他地方引入并使用這個(gè)組件了,雖然它還沒有完成,但我們不妨先讓它顯示在界面上吧:

<template>
  <div class="home">
    <DropList :dataList="dplist" labelProperty="city" @change="onDpChange($event)"></DropList>
    <p>其他文本內(nèi)容</p>
  </div>
</template>
<script>
  import DropList from '@/components/DropDownList.vue'
  //其他代碼略
</script>

這個(gè)頁面引入并使用了我們的DropDownList組件,:dataList="dplist" 綁定了當(dāng)前頁面的dplist數(shù)組到組件的dataList屬性上,這個(gè)數(shù)組中的對(duì)象有一個(gè)city字段,我們希望此字段顯示在下拉菜單上,因此我們?cè)O(shè)置組件的labelProperty為city,我們還給這個(gè)組件注冊(cè)了change事件,這個(gè)組件內(nèi)部需要派發(fā)這個(gè)事件,見下文。

現(xiàn)在我們回到組件的模板部分,發(fā)現(xiàn)它都還是靜態(tài)內(nèi)容,我們把這些靜態(tài)內(nèi)容修改為通過屬性渲染。

<template>
    <div class="zq-drop-list" @mouseover="onDplOver($event)" @mouseout="onDplOut($event)">
        <span>{{dplLable}}<i></i></span>
        <ul>
            <li v-for="(item, index) in dataList" :key="index" @click="onLiClick(index, $event)">{{item[labelProperty]}}</li>
        </ul>
    </div>
</template>

其中待選菜單li的文本是 item[labelProperty] 這樣就能正確的顯示開發(fā)者指定的字段了。
我們看看選中項(xiàng)的文本表達(dá)式:dplLabel,我們并沒有定義這個(gè)屬性,也沒有定義這個(gè)內(nèi)部數(shù)據(jù),它是哪兒來的?選中項(xiàng)的文本應(yīng)該是 dataList[activeIndex][labelProperty] (這個(gè)很好理解吧,有問題請(qǐng)留言),但這個(gè)表達(dá)式太長了,寫在模板里不利于維護(hù),我們就把它寫到計(jì)算屬性里吧。

computed:{
        dplLable(){
            return this.dataList[this.activeIndex][this.labelProperty]
        }
    }

于是才有了上面的dplLabel,計(jì)算屬性真的很好用呢。

現(xiàn)在下拉菜單的視圖和數(shù)據(jù)關(guān)聯(lián)部分我們已經(jīng)寫完了,接下來我們要實(shí)現(xiàn)它的功能。

第一步是先讓待選菜單默認(rèn)隱藏起來,這里我們?yōu)槭裁床恢苯佑胏ss的display:none呢,然后鼠標(biāo)經(jīng)過的時(shí)候display:block不就可以了嗎?因?yàn)檫@樣的話,我們無法實(shí)現(xiàn)點(diǎn)擊待選菜單條目的時(shí)候讓它隱藏,體驗(yàn)不好。我們用js來控制,但vue對(duì)直接訪問dom元素支持的并不好,我們要想在組件初始化的時(shí)候訪問dom元素,有一個(gè)最方便的做法,那就是:自定義指令。

我們?yōu)橄吕藛谓M件添加局部自定義指令,代碼如下:

directives:{
        dpl:{
            bind(el){
                el.style.display = "none";
            }
        }
    },

這個(gè)dpl就是自定義指令啦,請(qǐng)忽略我笨拙的命名哈!然后我們?cè)谧远x指令的鉤子函數(shù)bind方法中,訪問el元素,控制它的style屬性display:none; 最后,把這個(gè)自定義指令加到模板里面的ul標(biāo)簽上。別忘了要加v-,現(xiàn)在看看效果,待選菜單已經(jīng)隱藏了。

<ul v-dpl>

我們利用自定義指令鉤子函數(shù)訪問dom元素,實(shí)現(xiàn)了對(duì)dom的控制,這一點(diǎn)非常實(shí)用!

讓我們繼續(xù)實(shí)現(xiàn)最開始為下拉菜單定義的鼠標(biāo)經(jīng)過和鼠標(biāo)滑出的監(jiān)聽,實(shí)現(xiàn)待選菜單的顯示與隱藏。

onDplOver(event){
    let ul = event.currentTarget.childNodes[1];
    ul.style.display = "block";
},
onDplOut(event){
    let ul = event.currentTarget.childNodes[1];
    ul.style.display = "none";
},

我們?cè)谑髽?biāo)事件中,訪問event的currentTarget對(duì)象,為什么不是target?因?yàn)橄吕藛蔚淖釉匾矔?huì)觸發(fā)這個(gè)事件,如果訪問target,可能不會(huì)是我們預(yù)期的頂層元素。

最后一步,我們實(shí)現(xiàn)待選菜單條目的點(diǎn)擊事件,點(diǎn)擊后,待選菜單隱藏,修改內(nèi)部狀態(tài),派發(fā)change事件。

onLiClick(index){
    let path = event.path || (event.composedPath && event.composedPath()) //兼容火狐和safari
    path[1].style.display = "none";
    this.activeIndex = index;
    this.$emit("change", {
        index:index,
        value:this.dataList[index]
    })
}

這里有一個(gè)細(xì)節(jié)需要注意,我們要通過li元素找到外層ul元素,但path不支持火狐和safari,好在這兩個(gè)瀏覽器支持composedPath,因此才有了第一行代碼的兼容寫法。然后通過修改內(nèi)部數(shù)據(jù)activeIndex實(shí)現(xiàn)選中項(xiàng)文本的更新,最后調(diào)用emit方法向父元素派發(fā)change事件,別忘了把事件對(duì)象封裝好傳出去。

完整的代碼如下:

<template>
    <div class="zq-drop-list" @mouseover="onDplOver($event)" @mouseout="onDplOut($event)">
        <span>{{dplLable}}<i></i></span>
        <ul v-dpl>
            <li v-for="(item, index) in dataList" :key="index" @click="onLiClick(index, $event)">{{item[labelProperty]}}</li>
        </ul>
    </div>
</template>

<script>
export default {
    name: "DropDownList",
    data(){
        return {
            activeIndex:0
        }
    },
    props:{
        dataList:{
            type:Array,
            default(){
                return [
                    {name: "選項(xiàng)一"},
                    {name: "選項(xiàng)二"}
                ]
            }
        },
        labelProperty:{
            type:String,
            default(){ return "name" }
        }
    },
    directives:{
        dpl:{
            bind(el){
                el.style.display = "none";
            }
        }
    },
    methods:{
        onDplOver(event){
            let ul = event.currentTarget.childNodes[1];
            ul.style.display = "block";
        },
        onDplOut(event){
            let ul = event.currentTarget.childNodes[1];
            ul.style.display = "none";
        },
        onLiClick(index){
            let path = event.path || (event.composedPath && event.composedPath()) //兼容火狐和safari
            path[1].style.display = "none";
            this.activeIndex = index;
            this.$emit("change", {
                index:index,
                value:this.dataList[index]
            })
        }
    },
    computed:{
        dplLable(){
            return this.dataList[this.activeIndex][this.labelProperty]
        }
    }
}
</script>


<style scoped lang="scss">
    .zq-drop-list{
        display: inline-block;
        min-width: 100px;
        position: relative;
        span{
            display: block;
            height: 30px;
            line-height: 30px;
            background: #f1f1f1;
            font-size: 14px;
            text-align: center;
            color: #333333;
            border-radius: 4px;
            i{
                background: url(https://www.easyicon.net/api/resizeApi.php?id=1189852&size=16) no-repeat center center;
                margin-left: 6px;
                display: inline-block;
            }
        }
        ul{
            position: absolute;
            top: 30px;
            left: 0;
            width: 100%;
            margin: 0;
            padding: 0;
            border: solid 1px #f1f1f1;
            border-radius: 4px;
            overflow: hidden;
            li{
                list-style: none;
                height: 30px;
                line-height: 30px;
                font-size: 14px;
                border-bottom: solid 1px #f1f1f1;
                background: #ffffff;
            }
            li:last-child{
                border-bottom: none;
            }
            li:hover{
                background: #f6f6f6;
            }
        }
    }
</style>

以上為大家展示了vue如何實(shí)現(xiàn)一個(gè)下拉菜單組件,雖然比較簡單,但也基本涉及到了組件開發(fā)常用的一些特性。

歡迎大家在評(píng)論區(qū)留言自己想了解的前端話題,我會(huì)繼續(xù)推出更多精彩的文章!

原創(chuàng)不易,有錢的捧個(gè)錢場,給個(gè)打賞

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,117評(píng)論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,860評(píng)論 3 423
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 177,128評(píng)論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評(píng)論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,025評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,421評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,642評(píng)論 0 289
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,177評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,970評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,157評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,410評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評(píng)論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,896評(píng)論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,157評(píng)論 2 375

推薦閱讀更多精彩內(nèi)容

  • 前言 您將在本文當(dāng)中了解到,往網(wǎng)頁中添加數(shù)據(jù),從傳統(tǒng)的dom操作過渡到數(shù)據(jù)層操作,實(shí)現(xiàn)同一個(gè)目標(biāo),兩種不同的方式....
    itclanCoder閱讀 25,867評(píng)論 1 12
  • 1.安裝 可以簡單地在頁面引入Vue.js作為獨(dú)立版本,Vue即被注冊(cè)為全局變量,可以在頁面使用了。 如果希望搭建...
    Awey閱讀 11,066評(píng)論 4 129
  • vue概述 在官方文檔中,有一句話對(duì)Vue的定位說的很明確:Vue.js 的核心是一個(gè)允許采用簡潔的模板語法來聲明...
    li4065閱讀 7,262評(píng)論 0 25
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,155評(píng)論 4 61
  • 基于Vue的一些資料 內(nèi)容 UI組件 開發(fā)框架 實(shí)用庫 服務(wù)端 輔助工具 應(yīng)用實(shí)例 Demo示例 element★...
    嘗了又嘗閱讀 1,166評(píng)論 0 1