我们利用 Vue.js 的自定义指令能力,来实现一个自定义下拉菜单功能。描述如下:
- 点击按钮,弹出下拉菜单。
- 点击下拉菜单之外的区域,关闭下拉菜单。
1基础版
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">点击</button> <div class="dropDown" v-show="isShow"> <p>零售新物种:药店和便利店合体之后</p> </div> </div> </div> <script src="/UploadFiles/2021-04-02/vue.min.js">我们为按钮绑定了 isShow 变量,当点击按钮时,显示 class="dropDown" 的 div 元素。
js:
Vue.directive('outside-click', { bind: function (el, binding, vnode) { //定义点击函数 function clickHandler(e) { if (el.contains(e.target)) {//如果点击区域在所在指令元素内部,则直接返回 return false; } if (binding.expression) {//如果定义了表达式,则执行表达式中的函数 binding.value(e); } } el.vueOutsideClick = clickHandler; document.addEventListener('click', clickHandler);//绑定到 document 的点击事件 }, 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 中:
- 首先在定义了点击函数,内部逻辑为:如果点击区域在所在指令元素内部,则直接返回;如果定义了表达式,则执行表达式中的函数(示例中是 close)。
- 这里用到了 contains 函数, A.contains(B) 是判断元素 A 是否包含了元素 B。
- 接着在 el 中定义了一个变量,用于存放刚才定义的点击函数。bind() 与 unbind() 通过 el 变量进行参数传递。
- 然后绑定到 document 的点击事件。
unbind 中:
- 解绑在 bind 中绑定的点击事件。
- 销毁该变量。
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 关闭
现在让我们做个优化,即在按下键盘的 ESC 键时,也能关闭下拉菜单。
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 函数中,强化了判断,如果点击区域在所在指令元素内部并且没有按下 ESC 键时,才直接返回。即按下 ESC 键时,会执行后续操作(执行表达式中的函数)。
在 unbind 函数中,也解绑了 keyup 事件。
效果:
3 ESC 为可选项
我们可以把 ESC 作为可选项,而这可以通过修饰符来实现。
js:
bind: function (el, binding, vnode) { //定义点击函数 function clickHandler(e) { //是否开启开关 var escSwitch = (binding.modifiers && binding.modifiers.esc); if (el.contains(e.target)) {//如果点击区域在所在指令元素内部时 if (escSwitch && e.keyCode === 27) {//带有了 esc 修饰符,则让程序往下执行 } else {//直接返回 return false; } } if (binding.expression) {//如果定义了表达式,则执行表达式中的函数 binding.value(e); } } ... }我们通过 binding.modifiers 来判断自定义指令是否设置了 esc 修饰符,然后以此为基础,来编写后续逻辑。
html:
<div id="app" v-cloak> <div class="main" v-outside-click.esc="close"> ... </div> </div>本文示例代码
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]