亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

詳解Element 指令clickoutside源碼分析

2019-11-19 12:08:10
字體:
來源:轉載
供稿:網友

clickoutside是Element-ui實現的一個自定義指令,顧名思義,該指令用來處理目標節點之外的點擊事件,常用來處理下拉菜單等展開內容的關閉,在Element-ui的Select選擇器、Dropdown下拉菜單、Popover 彈出框等組件中都用到了該指令,所以這個指令在實現一些自定義組件的時候非常有用。

要分析該源碼,首先要了解一下Vue的自定義指令。自定義指令的定義方式如下:

// 注冊一個全局自定義指令 Vue.directive('directiveName', { bind: function(el, binding, vnode){  // 當指令第一次綁定到元素時調用,常用來進行一些初始化設置 	... }, inserted: function(el, binding, vnode){  // 當被綁定的元素插入到 DOM 中時…… 	... }, update: function(el, binding, vnode, oldVnode){  // 所在組件的 VNode 更新時調用,但是可能發生在其子 VNode 更新之前 	... }, componentUpdated: function(el, binding, vnode, oldVnode){  // 指令所在組件的 VNode 及其子 VNode 全部更新后調用 	... }, unbind: function(el, binding, vnode){  // 只調用一次,指令與元素解綁時調用,類似于beforeDestroy的功能 	... }});

可以看到在配置對象中只有5個可選的鉤子函數,他們的參數有4個,分別是 el、binding、vnode、oldVnode

  • el :指令所綁定的元素,可以用來直接操作DOM
  • binding : 一個包含了自定義詳細信息的對象,內部收集了使用自定義指令時傳入的值、修飾符、參數等數據,詳細信息可以在官方文檔見到,已經說的十分詳細了
  • vnode : Vue編譯生成的虛擬節點
  • oldVnode: 本次Vnode更新之前,上一次產生的虛擬節點,僅在  update  和  componentUpdated  鉤子中可用。

看完了自定義指令的內容,接下來我們就來分析clickoutside的具體實現。

import Vue from 'vue';import { on } from 'element-ui/src/utils/dom';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 = {}) {  ... };}let startClick;let seed = 0;export default { bind(el, binding, vnode) {  ... }, update(el, binding, vnode) {  ... }, unbind(el) {  ... }};

上面是簡化后的源碼,可以看到首先引入Vue和一個用來進行事件綁定的工具函數on,然后定義了兩個全局常量 nodeListctx 。nodeList 是一個 元素搜集器 ,會將頁面中所有綁定了clickoutside指令的dom元素存儲起來,而ctx定義了一個命名空間(必須比較特殊,防止和其它特性重名), 后面會將它添加為元素el的properties ,具體后面會分析到。

接著利用之前引入的Vue進行判斷,非服務端則給文檔對象添加 mousedownmouseup 事件,在 mousedown 事件回調中,將事件對象存儲到 startClick 全局變量中,在 mouseup 事件回調中遍歷 nodeList ,然后 分別執行每一個node( 即之前存儲起來的clickoutside指令綁定的元素el ) ctx 特性中存儲的 documentHandler 函數 。關于ctx property的值會在后面介紹。

最后就是導出了一個 clickoutside 的配置對象,在用到 clickoutside 指令的組件中導入該配置對象,然后在組件中局部注冊后就可以使用了。

該配置對象中使用了 bind、update、unbind 三個鉤子函數來定義clickoutside指令,主要做的事情就是搜集該自定義指令的相關信息,然后存儲到 el 的 ctx 特性上。接下來具體來看一下這個搜集過程。

首先是bind鉤子函數:

bind(el, binding, vnode) { nodeList.push(el); const id = seed++; el[ctx] = {  id,  documentHandler: createDocumentHandler(el, binding, vnode),  methodName: binding.expression,  bindingFn: binding.value };}

這里首先將el直接push到nodeList中,這樣每次有clickoutside指令綁定到頁面上,都會將綁定元素存儲到nodeList當中去,即前面說過的 元素搜集器 。接下來將全局變量seed++,并且賦值給一個臨時變量id,最后就是給el的ctx特性賦值了,它的值是一個對象,內部包括了:

id :前面生成的全局唯一id,用來標識該clickoutside指令

documentHandler :利用 createDocumentHandler 生成的一個回調函數。前面的分析中說到,給頁面綁定的mouseup事件回調中,會遍歷nodeList,分別執行每一個綁定元素el的ctx特性上的documentHandler函數, 這個函數就是在這里生成的 ,至于這個回調函數究竟是做了什么,后面再詳細分析。

methodName :binding.expression,查看自定義指令的文檔可以知道, binding.expression 的值是字符串形式的指令表達式。例如有   <div v-my-directive="1 + 1"></div> ,則 binding.expression 的值為  1 + 1

bindingFn : binding.value,指令的綁定值,還是上面的例子,則 binding.value 的值是 2 (1 + 1等于2),即 指令的值為js表達式的情況下, **binding.expresssion** 為表達式本身,是一個字符串,而 **binding.value** 是該表達式的值。

接著我們看下 update 鉤子:

update(el, binding, vnode) {	el[ctx].documentHandler = createDocumentHandler(el, binding, vnode);	el[ctx].methodName = binding.expression;	el[ctx].bindingFn = binding.value;}

可以看到update鉤子的內容很簡單,就是當組件更新的時候,更新 綁定元素 el 的特性 ctx 中的值。

再接著我們看看最后一個鉤子 unbind :

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];}

這個鉤子也很簡單,就是當 clickoutside 指令與元素el解綁的時候,遍歷 nodeList ,通過ctx特性上的id找到 nodeList 中存儲的當前解綁元素el,將它從nodeList中刪除,并且刪除el上的ctx特性。

以上就是clickoutside指令配置對象中做的所有操作,總結起來就是:

當指令與元素綁定以及組件更新的時候,搜集并設置綁定元素的ctx特性,同時將綁定元素添加到nodeList當中去,當指令與元素解綁的時候,刪除nodeList中存儲的對應的綁定元素,并將之前設置在綁定元素上之前設置的ctx特性刪除掉。

前面說過,給頁面綁定的mouseup事件回調中,會遍歷nodeList,分別執行搜集起來的每一個綁定元素el上的ctx特性中的 documentHandler 函數。而該函數是通過  createDocumentHandler 函數生成的,讓我們看看這個函數都做了什么。

function createDocumentHandler(el, binding, vnode) { return function(mouseup = {}, mousedown = {}) {  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;  if (binding.expression &&   el[ctx].methodName &&   vnode.context[el[ctx].methodName]) {   vnode.context[el[ctx].methodName]();  } else {   el[ctx].bindingFn && el[ctx].bindingFn();  } };}

可以看到,這個函數利用了閉包將傳入的參數緩存起來,然后返回一個函數。在這個返回的函數中,會進行一系列判斷,首先在第一個if里面,判斷了:

  • vnode.context 是否存在,不存在退出
  • mouseup.target 是否存在,不存在退出
  • mousedown.target 是否存在,不存在退出
  • 綁定對象el是否包含 mouseup.target/mousedown.target 子節點,如果包含說明點擊的是綁定元素的內部,則不執行 clickoutside 指令內容
  • 綁定對象el是否等于 mouseup.target ,等于說明點擊的就是綁定元素自身,也不執行 clickoutside 指令內容
  • 最后 vnode.context.popperElm 這部分內容則是 : 判斷是否點擊在下拉菜單的上,如果是,也是沒有點擊在綁定元素外部,不執行clickoutside指令內容

如圖,如果點擊在紅色區域內,則全部不觸發 clickoutside 指令的邏輯。

如果以上條件全部符合,則判斷閉包緩存起來的值,如果 methodName 存在則執行這個方法,如果不存在則執行 bindingFn 。例如:

<template>	<div v-clickoutside="handleClose"></div></template><script> export default {  data(){   return {    visible: false   };  },  methods: {   handleClose(){    this.visible = false;   }  } }</script>

在這個例子中, methodName 或者 bindingFn 就是通過指令傳入的 handleClose 方法。執行該方法,就可以執行 clickoutside 指令的邏輯了

以上就是 documentHandler 方法的生成以及內部邏輯。通過這個方法和之前的分析,我們就可以知道,當頁面綁 mouseup 事件觸發的時候,會遍歷 nodeList ,依次執行每一個綁定元素el的ctx特性上的 documentHandler 方法。而在這個方法內部可以訪問到指令傳入的表達式,在進行一系列判斷之后會執行該表達式,從而達到點擊目標元素外部執行給定邏輯的目的,而這個給定邏輯是通過自定義指令的值,傳到綁定元素el的ctx特性上的。

至此 clickoutside 的源碼就分析完了,可以看到 clickoutside 指令的源碼并不復雜,不過涉及到的內容還是挺多的,有許多東西值得我們學習,比如利用dom元素的特性來存儲額外信息,使用閉包緩存變量,如何判斷點擊在目標元素外部和Vue自定義指令的使用等等。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久久久久欧美| 国产精品久久国产精品99gif| 九九热这里只有精品6| 精品国产一区av| 国产精品7m视频| 91夜夜未满十八勿入爽爽影院| 96sao精品视频在线观看| 久久久久久国产精品三级玉女聊斋| 日韩美女视频中文字幕| 亚洲精品网址在线观看| 精品一区二区三区四区| 国产z一区二区三区| 亚洲www永久成人夜色| 亚洲欧美日韩中文在线制服| 国产热re99久久6国产精品| 亚洲无亚洲人成网站77777| 国产综合在线视频| 亚洲欧美日韩国产成人| 国产精品专区第二| 91高清视频免费观看| 69视频在线播放| 亚洲免费精彩视频| 韩国三级日本三级少妇99| 国产91精品久久久久久久| 日韩免费高清在线观看| 在线视频一区二区| 国产精品网站入口| 日韩毛片中文字幕| 亚洲欧美日韩国产精品| 国产成人黄色av| 91久久久久久| 欧美成人精品影院| 久久精品99久久久香蕉| 精品国内自产拍在线观看| 国产精品视频最多的网站| 久久精品国产91精品亚洲| 亚洲一区二区久久久久久| 成人精品视频在线| 久久天天躁夜夜躁狠狠躁2022| 成人欧美在线观看| 国产欧美亚洲视频| 国产91精品网站| 欧美大片免费观看在线观看网站推荐| 在线看片第一页欧美| 欧美色道久久88综合亚洲精品| 精品成人av一区| 久国内精品在线| 国产91精品黑色丝袜高跟鞋| 性色av一区二区三区红粉影视| 911国产网站尤物在线观看| 91网站免费观看| 久久艹在线视频| 久久久久99精品久久久久| 欧美精品在线免费播放| 欧美成人性色生活仑片| 国产欧美精品一区二区| 国产精品女视频| 久久久久久成人精品| 亚洲福利视频网站| 日韩在线视频网| 欧美极品少妇与黑人| 91精品国产自产在线观看永久| 日韩中文字幕在线免费观看| 亚洲风情亚aⅴ在线发布| 久久久久久久久久久久久久久久久久av| 国产精品福利在线| 美日韩精品免费观看视频| 91精品视频网站| 日韩欧美在线视频日韩欧美在线视频| 蜜月aⅴ免费一区二区三区| 国产成人亚洲综合| 亚洲精品国产美女| 欧美一区二区三区免费视| 国内精品久久久久伊人av| 97在线观看免费高清| 日韩在线免费av| 国产美女久久精品| 久久久久久高潮国产精品视| 在线观看久久av| 美女扒开尿口让男人操亚洲视频网站| 久久久亚洲精选| 久久久亚洲成人| 成人字幕网zmw| 欧美日韩电影在线观看| 国产亚洲精品久久久| 亚洲黄色免费三级| 亚洲欧美国产高清va在线播| 久久69精品久久久久久久电影好| 91在线免费视频| 国产精品普通话| 最新国产成人av网站网址麻豆| 国产精品视频大全| 夜夜嗨av色综合久久久综合网| 色香阁99久久精品久久久| 一区二区三区动漫| 亚洲免费电影在线观看| 日韩精品免费在线播放| 国产精品亚洲自拍| 深夜成人在线观看| 亚洲欧美日韩久久久久久| 色诱女教师一区二区三区| 成人免费视频网| x99av成人免费| 欧美视频在线看| 中文字幕亚洲欧美一区二区三区| 精品久久久久久久中文字幕| 91精品国产综合久久香蕉最新版| 成人激情视频在线播放| 国产精品视频男人的天堂| 欧美夫妻性生活视频| 在线电影欧美日韩一区二区私密| 在线精品高清中文字幕| 国a精品视频大全| 成人h片在线播放免费网站| 亚洲japanese制服美女| 色噜噜狠狠狠综合曰曰曰88av| 欧美做爰性生交视频| 最新国产精品亚洲| 亚洲一区二区三区成人在线视频精品| 992tv成人免费视频| 亚洲精品大尺度| 中文字幕精品网| 久久久国产精品视频| 羞羞色国产精品| 亚洲福利视频久久| 51午夜精品视频| 2019中文字幕在线免费观看| 中文字幕一精品亚洲无线一区| 最近中文字幕2019免费| 久久国产精品首页| 91av视频在线播放| 亚洲第一区在线观看| 日韩成人在线电影网| 国产成人中文字幕| 欧美激情亚洲自拍| 日韩麻豆第一页| 中文字幕欧美精品日韩中文字幕| 精品女厕一区二区三区| 日本a级片电影一区二区| 亚洲激情视频在线播放| 日韩中文字幕在线播放| 日本在线观看天堂男亚洲| 久久久久久高潮国产精品视| 日韩av不卡在线| 亚洲网址你懂得| 国产精品久久久久久搜索| 日韩国产高清视频在线| 97精品伊人久久久大香线蕉| 亚洲人成绝费网站色www| 国产欧美精品va在线观看| 欧美洲成人男女午夜视频| 午夜欧美大片免费观看| 亚洲女人被黑人巨大进入al| 另类天堂视频在线观看| 91国在线精品国内播放| 亚洲人成电影网站色www| 久久久久久国产| 欧美性生交大片免费| 国产午夜精品一区理论片飘花| 欧洲亚洲在线视频| 国精产品一区一区三区有限在线| 欧美极品美女电影一区| 欧美精品18videos性欧美| 国产精品欧美亚洲777777|