淺談Vue.js中如何實(shí)現(xiàn)自定義下拉菜單指令
我們利用 Vue.js 的自定義指令能力,來實(shí)現(xiàn)一個(gè)自定義下拉菜單功能。描述如下:
- 點(diǎn)擊按鈕,彈出下拉菜單。
- 點(diǎn)擊下拉菜單之外的區(qū)域,關(guān)閉下拉菜單。
1基礎(chǔ)版
html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<link rel="stylesheet" type="text/css" href="style.css" rel="external nofollow" >
</head>
<body>
<div id="app" v-cloak>
<div class="main" v-outside-click="close">
<button @click="isShow=!isShow">點(diǎn)擊</button>
<div class="dropDown" v-show="isShow">
<p>零售新物種:藥店和便利店合體之后</p>
</div>
</div>
</div>
<script src="https://cdn.bootcss.com/vue/2.2.2/vue.min.js"></script>
<script src="index.js"></script>
</body>
</html>
我們?yōu)榘粹o綁定了 isShow 變量,當(dāng)點(diǎn)擊按鈕時(shí),顯示 class="dropDown" 的 div 元素。
js:
Vue.directive('outside-click', {
bind: function (el, binding, vnode) {
//定義點(diǎn)擊函數(shù)
function clickHandler(e) {
if (el.contains(e.target)) {//如果點(diǎn)擊區(qū)域在所在指令元素內(nèi)部,則直接返回
return false;
}
if (binding.expression) {//如果定義了表達(dá)式,則執(zhí)行表達(dá)式中的函數(shù)
binding.value(e);
}
}
el.vueOutsideClick = clickHandler;
document.addEventListener('click', clickHandler);//綁定到 document 的點(diǎn)擊事件
},
unbind: function (el, binding, vnode) {
document.removeEventListener('click', el.vueOutsideClick);//解綁
delete el.vueOutsideClick;//銷毀
}
});
var app = new Vue({
el: '#app',
data: {
isShow: false
},
methods: {
close: function () {
this.isShow = false;
}
}
});
bind 中:
- 首先在定義了點(diǎn)擊函數(shù),內(nèi)部邏輯為:如果點(diǎn)擊區(qū)域在所在指令元素內(nèi)部,則直接返回;如果定義了表達(dá)式,則執(zhí)行表達(dá)式中的函數(shù)(示例中是 close)。
- 這里用到了 contains 函數(shù), A.contains(B) 是判斷元素 A 是否包含了元素 B。
- 接著在 el 中定義了一個(gè)變量,用于存放剛才定義的點(diǎn)擊函數(shù)。bind() 與 unbind() 通過 el 變量進(jìn)行參數(shù)傳遞。
- 然后綁定到 document 的點(diǎn)擊事件。
unbind 中:
- 解綁在 bind 中綁定的點(diǎn)擊事件。
- 銷毀該變量。
css:
[v-cloak] {
display: none;
}
.main {
width: 125px;
}
button {
display: block;
width: 100%;
color: #ffffff;
background-color: #99CC66;
border: 0;
padding: 6px;
text-align: center;
font-size: 12px;
border-radius: 4px;
cursor: pointer;
position: relative;
outline: none;
}
button:active {
top: 1px;
left: 1px;
}
.dropDown {
width: 100%;
height: 150px;
margin: 5px 0;
font-size: 12px;
background-color: #ffffff;
border-radius: 4px;
box-shadow: 0 1px 6px rgba(0, 0, 0, .2);
}
.dropDown p {
display: inline-block;
padding: 6px;
}
效果:

2 ESC 關(guān)閉
現(xiàn)在讓我們做個(gè)優(yōu)化,即在按下鍵盤的 ESC 鍵時(shí),也能關(guān)閉下拉菜單。
js:
bind: function (el, binding, vnode) {
function clickHandler(e) {
if (el.contains(e.target) && e.keyCode !== 27) {
return false;
}
...
}
...
document.addEventListener('keyup', clickHandler, false);//綁定鍵盤事件
},
unbind: function (el, binding, vnode) {
...
document.removeEventListener('keyup', el.vueOutsideClick);//解綁
...
}
在 bind 函數(shù)中,強(qiáng)化了判斷,如果點(diǎn)擊區(qū)域在所在指令元素內(nèi)部并且沒有按下 ESC 鍵時(shí),才直接返回。即按下 ESC 鍵時(shí),會(huì)執(zhí)行后續(xù)操作(執(zhí)行表達(dá)式中的函數(shù))。
在 unbind 函數(shù)中,也解綁了 keyup 事件。
效果:

3 ESC 為可選項(xiàng)
我們可以把 ESC 作為可選項(xiàng),而這可以通過修飾符來實(shí)現(xiàn)。
js:
bind: function (el, binding, vnode) {
//定義點(diǎn)擊函數(shù)
function clickHandler(e) {
//是否開啟開關(guān)
var escSwitch = (binding.modifiers && binding.modifiers.esc);
if (el.contains(e.target)) {//如果點(diǎn)擊區(qū)域在所在指令元素內(nèi)部時(shí)
if (escSwitch && e.keyCode === 27) {//帶有了 esc 修飾符,則讓程序往下執(zhí)行
} else {//直接返回
return false;
}
}
if (binding.expression) {//如果定義了表達(dá)式,則執(zhí)行表達(dá)式中的函數(shù)
binding.value(e);
}
}
...
}
我們通過 binding.modifiers 來判斷自定義指令是否設(shè)置了 esc 修飾符,然后以此為基礎(chǔ),來編寫后續(xù)邏輯。
html:
<div id="app" v-cloak>
<div class="main" v-outside-click.esc="close">
...
</div>
</div>
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
vue .then和鏈?zhǔn)秸{(diào)用操作方法
這篇文章主要介紹了vue .then和鏈?zhǔn)秸{(diào)用操作方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-07-07
vue指令只能輸入正數(shù)并且只能輸入一個(gè)小數(shù)點(diǎn)的方法
這篇文章主要介紹了vue指令只能輸入正數(shù)并且只能輸入一個(gè)小數(shù)點(diǎn)的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-06-06
Vue+java實(shí)現(xiàn)時(shí)間段的搜索示例
本文主要介紹了Vue+java實(shí)現(xiàn)時(shí)間段的搜索示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
Vue前端判斷數(shù)據(jù)對(duì)象是否為空的實(shí)例
這篇文章主要介紹了Vue前端判斷數(shù)據(jù)對(duì)象是否為空的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-09-09
vue中transition組件在項(xiàng)目中運(yùn)用小結(jié)
這篇文章主要介紹了vue中transition組件在項(xiàng)目中運(yùn)用,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
解決在vue+webpack開發(fā)中出現(xiàn)兩個(gè)或多個(gè)菜單公用一個(gè)組件問題
這篇文章主要介紹了在vue+webpack實(shí)際開發(fā)中出現(xiàn)兩個(gè)或多個(gè)菜單公用一個(gè)組件的解決方案,需要的朋友可以參考下2017-11-11

