https://github.com/ElemeFE/element/blob/d419e260d0fc1463ccbc4f5e45e129ec0e972255/src/utils/clickoutside.jsjavascript
下面的代碼是element ui中的clickoutside.jscss
const on = (function() { if (!Vue.prototype.$isServer && document.addEventListener) { return function(element, event, handler) { if (element && event && handler) { element.addEventListener(event, handler, false); } }; } else { return function(element, event, handler) { if (element && event && handler) { element.attachEvent('on' + event, handler); } }; } })(); import Vue from 'vue'; const nodeList = []; const ctx = '@@clickoutsideContext'; let startClick; let seed = 0; !Vue.prototype.$isServer && on(document, 'mousedown', e => (startClick = e)); !Vue.prototype.$isServer && on(document, 'mouseup', e => { nodeList.forEach(node => node[ctx].documentHandler(e, startClick)); }); function createDocumentHandler(el, binding, vnode) { return function(mouseup = {}, mousedown = {}) { console.log(vnode.context.popperElm) if (!vnode || !vnode.context || !mouseup.target || !mousedown.target || el.contains(mouseup.target) || el.contains(mousedown.target) || el === mouseup.target || (vnode.context.popperElm && (vnode.context.popperElm.contains(mouseup.target) || vnode.context.popperElm.contains(mousedown.target)))){ return; } console.log(binding.expression); if (binding.expression && el[ctx].methodName && vnode.context[el[ctx].methodName]) { vnode.context[el[ctx].methodName](); } else { el[ctx].bindingFn && el[ctx].bindingFn(); } }; } /** * v-clickoutside * @desc 點擊元素外面纔會觸發的事件 * @example * ```vue * <div v-element-clickoutside="handleClose"> * ``` */ export default { bind(el, binding, vnode) { nodeList.push(el); const id = seed++; el[ctx] = { id, documentHandler: createDocumentHandler(el, binding, vnode), methodName: binding.expression, bindingFn: binding.value }; }, update(el, binding, vnode) { el[ctx].documentHandler = createDocumentHandler(el, binding, vnode); el[ctx].methodName = binding.expression; el[ctx].bindingFn = binding.value; }, unbind(el) { let len = nodeList.length; for (let i = 0; i < len; i++) { if (nodeList[i][ctx].id === el[ctx].id) { nodeList.splice(i, 1); break; } } delete el[ctx]; } };
這個是我寫的alert.vuehtml
<template> <div class='dialog-alert' v-show="alert"> <div class='dialog-content'v-clickoutside="hideAlert"> <div class="dialog-body">{{message}}</div> <a href="javascript:;" class="dialog-button" @click="hideAlert">肯定</a> </div> <div class="dialog-mask"></div> </div> </template> <script> import clickoutside from'./../directives/vue-directive-clickout.js' export default{ props:{ alert:{ type:Boolean, required: true }, message:{ type:String, required: true } }, methods:{ hideAlert() { this.$emit('hideAlert'); } }, directives: { clickoutside }, } </script> <style lang="scss" rel="stylesheet/scss" scoped> .dialog-alert { z-index: 21px; .dialog-content { position: absolute; left: 50%; top: 50%; margin-left: -4.6875rem; margin-top: -1.734375rem; z-index: 21; border-radius: .3125rem; background-color: #fff; } .dialog-body, .dialog-button { width: 9.375rem; height: 1.734375rem; margin: 0 auto; line-height: 1.734375rem; } .dialog-body { border-bottom: 1px solid #999; color: #6c6c6c } .dialog-button { color: #ff5000 } } .dialog-mask { position: absolute; width: 100%; height: 100%; top: 0; left: 0; bottom: 0; z-index: 20; background: rgba(0, 0, 0, .6) } </style>
還有我在login.vue中寫的觸發條件;loginIn函數是將alert的值改成true,hideAlert是將alert的值設置爲falsevue
<button class="btn-login" type="button" @click="loginIn">登陸</button> <alert :alert="alert" message="wowoowow" @hideAlert='hideAlert'></alert>
這裏的個人問題就是爲何點擊登陸的時候,彈窗還出的來;點擊的時候hideAlert被執行了 因此應該是false;因此呢我就就好了各類的測試,始終找不到答案。java
最後呢?我在走路回家的路上想通了這個問題,由於mouseup、mousedown先執行的,click時間後執行的。有點小難受。node
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>Document</title> </head> <body> <div id="bbb">點擊</div> <script> document.getElementById('bbb').addEventListener('click', function(){ console.log('click') }, false); document.addEventListener('mouseup', function(){ console.log('mouseup') }, false); document.addEventListener('mousedown', function(){ console.log('mousedown') }, false); </script> </body> </html>
結果:git
測試的結果就是: clickoutside.jsgithub
一、當一個指令在同一頁面上有多個時;綁定的是document事件;document的時間被重複觸發;express