成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

向zepto.js學(xué)習(xí)如何手動(dòng)(trigger)觸發(fā)DOM事件

fuyi501 / 2141人閱讀

摘要:好啦我們已經(jīng)解決了是啥的疑問了,現(xiàn)在回去開始一步步解讀如何實(shí)現(xiàn)手動(dòng)觸發(fā)事件。我們主要看看這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)事件的大部分步驟和內(nèi)容。

前言

前端在最近幾年實(shí)在火爆異常,vue、react、angular各路框架層出不窮,咱們要是不知道個(gè)雙向數(shù)據(jù)綁定,不曉得啥是虛擬DOM,也許就被鄙視了?;馃岬谋澈笸彩菬o盡的浮躁,學(xué)習(xí)這些先進(jìn)流行的類庫或者框架可以讓我們走的更快,但是靜下心來回歸基礎(chǔ),把基石打牢固,卻可以讓我們走的更穩(wěn),更遠(yuǎn)。

最近一直在看zepto的源碼,希望通過學(xué)習(xí)它掌握一些框架設(shè)計(jì)的技巧,也將很久不再拾起的js基礎(chǔ)重新溫習(xí)鞏固一遍。如果你對(duì)這個(gè)系列感興趣,歡迎點(diǎn)擊下方地址watch,隨時(shí)關(guān)注動(dòng)態(tài)。這篇文章主要想說一下zepto中事件模塊(event.js)的trigger實(shí)現(xiàn)原理。

原文地址

倉(cāng)庫地址

event.js模塊

zepto中由許多小的模塊組合合成,基礎(chǔ)的zepto.js模塊,event.js事件處理模塊,ajax.js請(qǐng)求處理模塊等等。其中event.js事件處理模塊的核心完成了zepto中事件綁定on,移除off還有手動(dòng)觸發(fā)trigger等功能。我們簡(jiǎn)單回顧下如何使用zepto的這三大功能。

  • 1
  • 2
let $list = $(".list")

let cb1 = function (e, name) {
  console.log(1, name)
}

let cb2 = function (e, name) {
  console.log(2, name)
}

$list.on("click", cb1)
$list.on("click", cb2)

// 移除事件

// 我們可以指定移除click事件的某個(gè)事件處理程序
$list.off("click", cb1)
// 也可以直接移除click事件
$list.off("click")

// 手動(dòng)觸發(fā)事件
$list.trigger("click", "qianlongo")

哥們你逗我呢,jQuery,zepto多熟了,誰不會(huì)用這個(gè)啊!客觀別急,我們今天主要是慢慢來看看它源碼怎么實(shí)現(xiàn)的。

一步步看trigger怎么實(shí)現(xiàn)

直接上代碼

$.fn.trigger = function (event, args) {
  // 對(duì)傳入的event進(jìn)行處理,如果是字符串或者純對(duì)象,得到一個(gè)自己創(chuàng)建的事件對(duì)象
  // 如果傳入的已經(jīng)是個(gè)經(jīng)過$.Event處理的對(duì)象,則放入compatible再次改造(其實(shí)就是添加了幾個(gè)方法,和重寫了幾個(gè)方法)
  event = (isString(event) || $.isPlainObject(event)) ? $.Event(event) : compatible(event)
  // args傳遞給事件處理程序的參數(shù)
  event._args = args
  return this.each(function () {
    // handle focus(), blur() by calling them directly
    if (event.type in focus && typeof this[event.type] == "function") this[event.type]()
    // items in the collection might not be DOM elements
    // 觸發(fā)dom事件
    else if ("dispatchEvent" in this) this.dispatchEvent(event)
    // 因?yàn)閦epto對(duì)象內(nèi)部的元素不一定是dom元素,此時(shí)直接觸發(fā)回調(diào)函數(shù)
    else $(this).triggerHandler(event, args)
  })
}

直接貼出trigger函數(shù)的代碼可能我們是懵逼的

$是啥?。。。?/p>

$.fn是啥?。。?!

$.isPlainObject又是啥啊?。。?/p>

$.Event又是什么鬼?

仿佛有一連串的問題等待著我們解決。

為了直接切入不易理解,我們先來看看zepto中是如何給基礎(chǔ)的zepto.js模塊添加功能的

首先看看zepto.js模塊

var Zepto = (function () {
  // xxxx
  var $ = function (selector, context) {
    return zepto.init(selector, context)
  }
  return $

  zepto.Z.prototype = Z.prototype = $.fn
  // xxxx
})()

window.Zepto = Zepto
window.$ === undefined && (window.$ = Zepto)

盡量刪除了一些不必要的代碼,可以看到我們平時(shí)使用的Zepto其實(shí)就是其匿名函數(shù)自執(zhí)行內(nèi)部導(dǎo)出的一個(gè)函數(shù)。而$.fn就是其原型

如何給zepto.js模塊增添功能

zepto.js模塊只有一些基礎(chǔ)的功能,比如給dom添加事件的功能就沒有,怎么添加呢?

(function ($) {
  // xxx
  $.fn.on = function () {//xxxx}
  $.fn.off = function () {//xxxx}
  $.fn.trigger = function () {//xxxx}
  $.Event = function () {//xxx}
  // xxx
})(Zepto)

最后縮減掉其他的干擾代碼,可以看到所謂的給zepto.js模塊增添功能,基本上就是在其原型上添加新的方法或者直接在$函數(shù)上定一些靜態(tài)方法。

好啦我們已經(jīng)解決了$,$.fn是啥的疑問了,現(xiàn)在回去開始一步步解讀如何實(shí)現(xiàn)手動(dòng)觸發(fā)事件。

重新看trigger函數(shù)源碼
$.fn.trigger = function (event, args) {
  // 對(duì)傳入的event進(jìn)行處理,如果是字符串或者純對(duì)象,得到一個(gè)自己創(chuàng)建的事件對(duì)象
  // 如果傳入的已經(jīng)是個(gè)經(jīng)過$.Event處理的對(duì)象,則放入compatible再次改造(其實(shí)就是添加了幾個(gè)方法,和重寫了幾個(gè)方法)
  event = (isString(event) || $.isPlainObject(event)) ? $.Event(event) : compatible(event)
  // args傳遞給事件處理程序的參數(shù)
  event._args = args

  // xxx
}

先把后面的一些代碼給刪除了,我們先理解這幾句代碼。其中非常重要的一個(gè)函數(shù)就是$.Event,至于

isString => 判斷是不是字符串

isPlainObject => 判斷是不是存粹的對(duì)象(必須是對(duì)象,window對(duì)象除外,該對(duì)象的原型必須和Object的原型一致)

compatible => 其實(shí)就是對(duì)事件對(duì)象event做一些擴(kuò)展,比如添加一些方法,重寫一些方法之類的。

這幾個(gè)方法暫時(shí)可以不需要太多關(guān)心.

我們主要看看$.Event,這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)dom事件的大部分步驟和內(nèi)容。

我們主要看看$.Event,這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)dom事件的大部分步驟和內(nèi)容。

我們主要看看$.Event,這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)dom事件的大部分步驟和內(nèi)容。

$.Event是個(gè)啥

創(chuàng)建并初始化一個(gè)指定的dom事件對(duì)象, 如果給定了props,則將其擴(kuò)展到事件對(duì)象上

 $.Event = function (type, props) {
  // 當(dāng)type是個(gè)對(duì)象時(shí),比如{type: "click", data: "qianlongo"}
  if (!isString(type)) props = type, type = props.type
  // click,mousedown,mouseup mousemove對(duì)應(yīng)MouseEvent
  // 其他事件對(duì)應(yīng)為Events
  // 并把bubbles設(shè)置為true,表示事件冒泡,為false則不冒泡
  var event = document.createEvent(specialEvents[type] || "Events"), bubbles = true
  // 當(dāng)props存在的時(shí)候,對(duì)props進(jìn)行循環(huán)處理,將其屬性擴(kuò)展到event對(duì)象上
  if (props) for (var name in props) (name == "bubbles") ? (bubbles = !!props[name]) : (event[name] = props[name])
  // 初始化事件對(duì)象,第一個(gè)為事件類型,第二個(gè)為冒泡與否,第三個(gè)為是否可以通過preventDefault來阻止瀏覽器默認(rèn)行為
  event.initEvent(type, bubbles, true)
  // 再對(duì)創(chuàng)造出來的時(shí)間對(duì)象處理一番并返回
  return compatible(event)
}

注釋已經(jīng)寫的很清楚了,這個(gè)函數(shù)就是返回一個(gè)經(jīng)過初始化了的事件對(duì)象

到這里我們直接歸納一下要手動(dòng)觸發(fā)一個(gè)dom事件的基本步驟

手動(dòng)觸發(fā)一個(gè)dom事件,需要3步,如果你對(duì)document.createEvent,不是很熟悉,可以點(diǎn)擊查看。

創(chuàng)建一個(gè)事件對(duì)象 document.createEvent(event)

初始化事件對(duì)象 event.initEvent(type, bubbles, true)

分發(fā)事件 dom.dispatchEvent(event)

到這里已經(jīng)完成了前面兩步,還剩最后一步了,在來看trigger剩下的代碼

手動(dòng)觸發(fā)dom事件最后一步
 $.fn.trigger = function (event, args) {
  // 對(duì)傳入的event進(jìn)行處理,如果是字符串或者純對(duì)象,得到一個(gè)自己創(chuàng)建的事件對(duì)象
  // 如果傳入的已經(jīng)是個(gè)經(jīng)過$.Event處理的對(duì)象,則放入compatible再次改造(其實(shí)就是添加了幾個(gè)方法,和重寫了幾個(gè)方法)
  event = (isString(event) || $.isPlainObject(event)) ? $.Event(event) : compatible(event)
  // args傳遞給事件處理程序的參數(shù)
  event._args = args
  return this.each(function () {
    // handle focus(), blur() by calling them directly
    if (event.type in focus && typeof this[event.type] == "function") this[event.type]()
    // items in the collection might not be DOM elements
    // 觸發(fā)dom事件
    else if ("dispatchEvent" in this) this.dispatchEvent(event)
    // 因?yàn)閦epto對(duì)象內(nèi)部的元素不一定是dom元素,此時(shí)直接觸發(fā)回調(diào)函數(shù)
    else $(this).triggerHandler(event, args)
  })
}

最后一步其實(shí)就是將當(dāng)前選中的元素進(jìn)行一次each遍歷,然后判斷要觸發(fā)的事件是不是focus或者blur,如果是就直接執(zhí)行。

再進(jìn)一步,如果dispatchEvent方法在當(dāng)前的dom元素屬性中存在,那么便將該事件觸發(fā)。(為什么要這一步呢?因?yàn)槲覀冎?()函數(shù)的使用方式有很多,有些方式得到的zepto對(duì)象是沒有選中dom節(jié)點(diǎn)的)

最后還有一個(gè)else分支,這個(gè)分支處理走的不是手動(dòng)觸發(fā)事件,而是直接觸發(fā)注冊(cè)事件時(shí)添加的事件處理程序(因?yàn)檫@里涉及到zepto事件模塊中如何管理元素與事件隊(duì)列的映射關(guān)系,篇幅會(huì)比較長(zhǎng),會(huì)在接下來的文章中說,這里不展開說明)

手動(dòng)diy一個(gè)

根據(jù)上面的描述,手動(dòng)觸發(fā)DOM事件,原來并沒有那么神奇,完成三步,即可達(dá)到目標(biāo)。我們自己來手動(dòng)寫一個(gè)示例

  • 1
  • 2
  • 3
let $list = document.querySelector(".list")
let $item1 = document.querySelector(".item1")

$list.addEventListener("click", function () {
  console.log(this.innerHTML)
}, false)

$item1.addEventListener("click", function () {
  console.log(this.innerHTML)
}, false)

// 1. 創(chuàng)建一個(gè)事件對(duì)象 document.createEvent(event)
let event = document.createEvent("Event")
// 2. 初始化事件對(duì)象 event.initEvent(type, bubbles, true)
event.initEvent("click", true, true)
// 3. 分發(fā)事件  dom.dispatchEvent(event)
$item1.dispatchEvent(event)

這個(gè)時(shí)候控制臺(tái)打印出了

1

  • 1
  • 2
  • 3
  • 1是item1的事件處理函數(shù)打印出來的。

    后面的li那部分則是list打印出來的。

    如果將initEvent的第二個(gè)參數(shù)設(shè)置為false,將不允許冒泡,則只會(huì)打印出1

    結(jié)尾

    如果這部分對(duì)你有點(diǎn)點(diǎn)幫助,點(diǎn)個(gè)star好不好呀! ???

    倉(cāng)庫地址

    文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

    轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/112103.html

    相關(guān)文章

    • zepto.js學(xué)習(xí)如何手動(dòng)(trigger)觸發(fā)DOM事件

      摘要:好啦我們已經(jīng)解決了是啥的疑問了,現(xiàn)在回去開始一步步解讀如何實(shí)現(xiàn)手動(dòng)觸發(fā)事件。我們主要看看這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)事件的大部分步驟和內(nèi)容。 前言 前端在最近幾年實(shí)在火爆異常,vue、react、angular各路框架層出不窮,咱們要是不知道個(gè)雙向數(shù)據(jù)綁定,不曉得啥是虛擬DOM,也許就被鄙視了。火熱的背后往往也是無盡的浮躁,學(xué)習(xí)這些先進(jìn)流行的類庫或者框架可以讓我們走的更快,但是靜下心...

      spacewander 評(píng)論0 收藏0
    • zepto.js學(xué)習(xí)如何手動(dòng)(trigger)觸發(fā)DOM事件

      摘要:好啦我們已經(jīng)解決了是啥的疑問了,現(xiàn)在回去開始一步步解讀如何實(shí)現(xiàn)手動(dòng)觸發(fā)事件。我們主要看看這里面幾乎含有如何手動(dòng)觸發(fā)一個(gè)事件的大部分步驟和內(nèi)容。 前言 前端在最近幾年實(shí)在火爆異常,vue、react、angular各路框架層出不窮,咱們要是不知道個(gè)雙向數(shù)據(jù)綁定,不曉得啥是虛擬DOM,也許就被鄙視了?;馃岬谋澈笸彩菬o盡的浮躁,學(xué)習(xí)這些先進(jìn)流行的類庫或者框架可以讓我們走的更快,但是靜下心...

      Julylovin 評(píng)論0 收藏0
    • 如何實(shí)現(xiàn)swipe、tap、longTap等自定義事件

      摘要:分別存儲(chǔ)事件的定時(shí)器。事件定時(shí)器延時(shí)時(shí)間存儲(chǔ)事件對(duì)象滑動(dòng)方向判斷我們根據(jù)下圖以及對(duì)應(yīng)的代碼來理解滑動(dòng)的時(shí)候方向是如何判定的。取消長(zhǎng)按,以及取消所有事件取消長(zhǎng)按取消所有事件方式都是類似,先調(diào)用取消定時(shí)器,然后釋放對(duì)應(yīng)的變量,等候垃圾回收。 前言 移動(dòng)端原生支持touchstart、touchmove、touchend等事件,但是在平常業(yè)務(wù)中我們經(jīng)常需要使用swipe、tap、double...

      羅志環(huán) 評(píng)論0 收藏0
    • Zepto源碼之Event模塊

      摘要:不支持事件冒泡帶來的直接后果是不能進(jìn)行事件委托,所以需要對(duì)和事件進(jìn)行模擬。調(diào)用函數(shù),分隔出參數(shù)的事件名和命名空間。這里判斷是否為函數(shù),即第一種傳參方式,調(diào)用函數(shù)的方法,將上下文對(duì)象作為的第一個(gè)參數(shù),如果存在,則與的參數(shù)合并。 Event 模塊是 Zepto 必備的模塊之一,由于對(duì) Event Api 不太熟,Event 對(duì)象也比較復(fù)雜,所以乍一看 Event 模塊的源碼,有點(diǎn)懵,細(xì)看下...

      vpants 評(píng)論0 收藏0

    發(fā)表評(píng)論

    0條評(píng)論

    最新活動(dòng)
    閱讀需要支付1元查看
    <