摘要:一起源方法最終是用綁定事件的而方法正是等于二作用觸發(fā)綁定的事件的處理程序源碼源碼行即原生觸發(fā)事件的處理程序修正對象獲取事件的處理程序集合,結(jié)構(gòu)如下從數(shù)據(jù)緩存中獲取事件處理集合即目標元素委托目標這段代碼壓根不會執(zhí)行,因為全局搜索沒找到結(jié)構(gòu)
一、起源
jQuery.event.add()方法最終是用addEventListener綁定事件的:
elem.addEventListener( type, eventHandle )
而eventHandle方法正是等于jQuery.event.dispatch():
if ( !( eventHandle = elemData.handle ) ) { eventHandle = elemData.handle = function( e ) { return typeof jQuery !== "undefined" && jQuery.event.triggered !== e.type ? jQuery.event.dispatch.apply( elem, arguments ) : undefined; }; }
二、$.event.dispatch()
作用:
觸發(fā)綁定的事件的處理程序
源碼:
//源碼5472行 //nativeEvent即原生MouseEvent //觸發(fā)事件的處理程序 dispatch: function( nativeEvent ) { //修正event對象 // Make a writable jQuery.Event from the native event object var event = jQuery.event.fix( nativeEvent ); console.log(event,"event5479") var i, j, ret, matched, handleObj, handlerQueue, args = new Array( arguments.length ), //獲取click事件的處理程序集合,結(jié)構(gòu)如下: //[ // {type: "click", origType: "click", data: undefined, handler: ?, guid: 1}, // {type: "click", origType: "click", data: undefined, handler: ?, guid: 2}, // delegateCount:0, //] //從數(shù)據(jù)緩存中獲取事件處理集合 handlers = ( dataPriv.get( this, "events" ) || {} )[ event.type ] || [], //click:{ // trigger:{}, // _default:{} //} special = jQuery.event.special[ event.type ] || {}; // Use the fix-ed jQuery.Event rather than the (read-only) native event args[ 0 ] = event; for ( i = 1; i < arguments.length; i++ ) { args[ i ] = arguments[ i ]; } //this即目標元素 //delegateTarget:委托目標 event.delegateTarget = this; //這段代碼壓根不會執(zhí)行,因為全局搜索沒找到preDispatch // Call the preDispatch hook for the mapped type, and let it bail if desired if ( special.preDispatch && special.preDispatch.call( this, event ) === false ) { return; } // Determine handlers //結(jié)構(gòu)如下 //[{ // elem:xx, // handlers:[ // {type: "click", origType: "click", data: undefined, handler: ?, guid: 1}, // {type: "click", origType: "click", data: undefined, handler: ?, guid: 2}, // ] //}] //獲取handler隊列 handlerQueue = jQuery.event.handlers.call( this, event, handlers ); // Run delegates first; they may want to stop propagation beneath us i = 0; //沒有執(zhí)行stopPropagation()的話 console.log(handlerQueue,"handlerQueue5525") //先判斷有沒有冒泡 //再判斷有沒有阻止剩下的handler執(zhí)行 while ( ( matched = handlerQueue[ i++ ] ) && !event.isPropagationStopped() ) { console.log(matched,"matched5542") event.currentTarget = matched.elem; j = 0; //handleObj即單個事件處理程序 //沒有執(zhí)行stopImmediatePropagation()的話 //依次執(zhí)行每一個handler while ( ( handleObj = matched.handlers[ j++ ] ) && !event.isImmediatePropagationStopped() ) { // Triggered event must either 1) have no namespace, or 2) have namespace(s) // a subset or equal to those in the bound event (both can have no namespace). if ( !event.rnamespace || event.rnamespace.test( handleObj.namespace ) ) { //通過循環(huán)將為event添加handleObj和handleObj.data event.handleObj = handleObj; event.data = handleObj.data; //關(guān)鍵代碼,執(zhí)行事件處理程序handler ret = ( ( jQuery.event.special[ handleObj.origType ] || {} ).handle || handleObj.handler ).apply( matched.elem, args ); if ( ret !== undefined ) { //event.result賦值ret if ( ( event.result = ret ) === false ) { //阻止默認行為 event.preventDefault(); //阻止冒泡 event.stopPropagation(); } } } } } // Call the postDispatch hook for the mapped type if ( special.postDispatch ) { special.postDispatch.call( this, event ); } console.log(handlers,"event5587") //undefined return event.result; },
解析:
(1)jQuery.event.fix()
作用:
將原生事件對象MouseEvent修正(fix)成jQuery的event對象
源碼:
//源碼5700行 fix: function( originalEvent ) { //如果存在屬性id則原樣返回(因為已處理成jQueryEvent) return originalEvent[ jQuery.expando ] ? originalEvent : new jQuery.Event( originalEvent ); },
解析:
可以看到fix的本質(zhì)是新建一個event對象,再看jQuery.Event()方法
(2)jQuery.Event()
源碼:
//click,false //修正event對象 //源碼5777行 //src即MouseEvent jQuery.Event = function( src, props ) { // Allow instantiation without the "new" keyword if ( !( this instanceof jQuery.Event ) ) { return new jQuery.Event( src, props ); } // Event object //src.type=click if ( src && src.type ) { //MouseEvent this.originalEvent = src; //click this.type = src.type; // Events bubbling up the document may have been marked as prevented // by a handler lower down the tree; reflect the correct value. this.isDefaultPrevented = src.defaultPrevented || src.defaultPrevented === undefined && // Support: Android <=2.3 only src.returnValue === false ? returnTrue : returnFalse; // Create target properties // Support: Safari <=6 - 7 only // Target should not be a text node (#504, #13143) this.target = ( src.target && src.target.nodeType === 3 ) ? src.target.parentNode : src.target; this.currentTarget = src.currentTarget; this.relatedTarget = src.relatedTarget; // Event type } else { //click this.type = src; } // Put explicitly provided properties onto the event object //false if ( props ) { jQuery.extend( this, props ); } // Create a timestamp if incoming event doesn"t have one this.timeStamp = src && src.timeStamp || Date.now(); // Mark it as fixed //修正的標志 this[ jQuery.expando ] = true; };
解析:
簡單來說,就是把原生event事件上的常用屬性賦值到了jQuery的event上
$("#A").on("click" ,function (event) { //這個就是jQuery.Event()構(gòu)建出的event console.log(event,"A被點擊了") })
jQuery的event結(jié)構(gòu)如下:
//click的event就是jQuery.Event jQuery.Event{ handleObj{ data:undefined, guid: 2, handler:function(){console.log("A被點擊了")}, namespace: "clickA", origType: "click", selector: "#B", type: "click.clickA", }, originalEvent:{ //就是MouseEvent }, target:div#B, type: "click", delegateTarget: div#A, //fix 的標志 jQuery331087940272164138: true, currentTarget: div#A, isDefaultPrevented:xxx, timeStamp:Date.now(), isDefaultPrevented:function(){return false} }
注意下originalEvent 和jQuery.extend( this, props )
前者就是原生MouseEvent,只是將原生event作為jQuery.event的originalEvent屬性了;
后者是擴展屬性,如果開發(fā)者想額外加入自定義屬性的話。
(3)dataPriv.get( this, "events" )
注意:
jQuery的數(shù)據(jù)緩存里的events和上面說的event是不同的
數(shù)據(jù)緩存的events是用來結(jié)構(gòu)如下:
{ click:[ { type: "click", origType: "click", data: undefined, handler: function(){console.log("B委托A綁定click事件")}, guid: 1, namespace: "", needsContext: undefined, selector: #B, }, { type: "click", origType: "click", data: undefined, handler: function(){console.log("A綁定click事件")}, guid: 2, namespace: "", needsContext: undefined, selector: undefined, }, //事件委托的數(shù)量 delegateCount:1, ], focus:[ { type: "focus", origType: "focus", data: undefined, handler: function(){console.log("A綁定focus事件")}, guid: 3, namespace: "", needsContext: undefined, selector: undefined, }, delegateCount:0, ], }
(4) jQuery.event.handlers
作用:
獲取handler隊列
源碼:
jQuery.event = { //源碼5547行 //組裝事件處理隊列 //event是fix過的MouseEvent, handlers handlers: function( event, handlers ) { var i, handleObj, sel, matchedHandlers, matchedSelectors, handlerQueue = [], //0 delegateCount = handlers.delegateCount, //目標元素 cur = event.target; //handlers,第一個handler是委托事件,第二個handler是自身事件 // Find delegate handlers if ( delegateCount && // Support: IE <=9 // Black-hole SVG
解析:
注意下這個雙層循環(huán),目的是把每一層的委托事件的集合push進matchedHandlers,然后再將matchedHandlers放進handlerQueue隊列
在處理完每層的委托事件后,將剩下的自身綁定事件再push進handlerQueue隊列中
也就是說,handlerQueue的結(jié)構(gòu)如下:
[ //委托事件 { elem:xx, handlers:[ {type: "click", origType: "click", data: undefined, handler: ?, guid: 1}, {type: "click", origType: "click", data: undefined, handler: ?, guid: 2}, ] }, //自身綁定事件 { elem:xxx, handlers:[ {type: "click", origType: "click", data: undefined, handler: ?, guid: 3}, {type: "click", origType: "click", data: undefined, handler: ?, guid: 4}, ] }, ]
(5)回過頭再往下看dispatch 源碼,是兩個while循環(huán),舉個例子來說明下:
這是A$("#A").on("click" ,function (event) { console.log(event,"A被點擊了") }) $("#A").on("click" ,"#B",function (event) { console.log(event,"點擊了B,即B委托A的click事件被點擊了") })這是B
那么會
先循環(huán)并執(zhí)行委托事件,
即handler=function (event) {console.log(event,"點擊了B,即B委托A的click事件被點擊了")},
再循環(huán)并執(zhí)行目標元素自身綁定事件,
即handler=function (event) {console.log(event,"A被點擊了")}
前提是冒泡不被阻止
最后,執(zhí)行click事件的事件處理程序的關(guān)鍵代碼如下:
handleObj.handler.apply( matched.elem, args )
(完)
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/104663.html
摘要:源碼源碼行被點擊了點擊了,即委托的事件被點擊了優(yōu)先添加委托,再添加其他即委托在上的事件數(shù)量在下標為的位置插入委托事件解析可以看到,是優(yōu)先添加委托事件,再添加自身事件,觸發(fā)事件的時候也是按這個順序。 showImg(https://segmentfault.com/img/remote/1460000019419722); 前言:請先回顧下我之前寫的一篇文章:JavaScript之事件委...
摘要:十的觸發(fā)機制被點擊了元素本身綁定了一個事件,但是是原生事件,它是靠綁定來觸發(fā)事件的。 showImg(https://segmentfault.com/img/remote/1460000019505402); 前言:最重要的還是最后的流程圖,可以試著根據(jù)流程圖手寫實現(xiàn)$().on(),下篇文章會放出模擬實現(xiàn)的代碼。 一、舉例 這是A 這是C ...
摘要:專門為事件建立一個有問題,直接退出如果是一個事件處理對象,且有屬性。參考源碼分析事件體系結(jié)構(gòu)解密事件核心綁定設(shè)計一解密事件核心委托設(shè)計二本文在上的源碼地址,歡迎來。 歡迎來我的專欄查看系列文章。 通過前面一章對于 addEvent 庫的介紹,它的兼容性超級棒,據(jù)說對于 IE4、5 都有很好的兼容性,這和 jQuery 的原理是一致的,而在 jQuery 中,有一個對象與其相對應(yīng),那就是...
閱讀 2857·2023-04-25 18:58
閱讀 984·2021-11-25 09:43
閱讀 1220·2021-10-25 09:46
閱讀 3506·2021-09-09 11:40
閱讀 1707·2021-08-05 09:59
閱讀 878·2019-08-29 15:07
閱讀 966·2019-08-29 12:48
閱讀 707·2019-08-29 11:19