jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件包裝
在上篇文章給大家介紹了jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件體系結(jié)構(gòu),本篇繼續(xù)給大家介紹jquery1.9.1源碼分析系列相關(guān)知識,具體內(nèi)容請看下文吧。
首先需要明白,瀏覽器的原生事件是只讀的,限制了jQuery對他的操作。舉個簡單的例子就能明白為什么jQuery非要構(gòu)造一個新的事件對象。
在委托處理中,a節(jié)點委托b節(jié)點在a被click的時候執(zhí)行fn函數(shù)。當事件冒泡到b節(jié)點,執(zhí)行fn的時候上下文環(huán)境需要保證正確,是a節(jié)點執(zhí)行了fn而非b節(jié)點。如何保證執(zhí)行fn的上下文環(huán)境是a節(jié)點的:看源碼(紅色部分)
//執(zhí)行 ret = ( (jQuery.event.special[ handleObj.origType ] || {}).handle || handleObj.handler ).apply( matched.elem, args );
使用了apply將執(zhí)行函數(shù)的上下文替換成了a節(jié)點(matched.elem)。還有一點args[0]即是事件對象event。又如何保證event是a節(jié)點的事件的?這就是event.currentTarget這個重要的屬性的功能,所以在執(zhí)行apply之前還做了一步操作
event.currentTarget = matched.elem;
直接更改事件對象的currentTarget屬性,這在瀏覽器本地事件是做不到的。所以才有了基于本地事件構(gòu)造jQuery的事件對象。
事件分兩種:鼠標事件和鍵盤事件(不知道觸摸事件何時能加進來)??匆幌逻@兩者的詳細屬性
其中有些是瀏覽器自己的,非W3C標準的。jQuery將事件屬性分為三塊
鼠標和鍵盤事件共同擁有的屬性jQuery.event.props: "altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" ")
鍵盤事件專有的屬性jQuery.event.keyHooks.props: "char charCode key keyCode".split(" ")
鼠標事件專有的屬性jQuery.event.mouseHooks.props: "button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" ")
a. 構(gòu)造新的事件對象jQuery.event.fix(originalEvent)
構(gòu)造新的事件對象分三步完成
第一步,使用到event = new jQuery.Event( originalEvent ),構(gòu)造新事件對象(不明白new的作用的請點擊這里),并在創(chuàng)建事件的時候加上isDefaultPrevented、originalEvent、type 、timeStamp和事件已經(jīng)被修正過的標記(優(yōu)化使用,避免不必要的處理)。jQuery.Event(src, props)的源碼如下
jQuery.Event = function( src, props ) { // Allow instantiation without the 'new' keyword if ( !(this instanceof jQuery.Event) ) { return new jQuery.Event( src, props ); } //src為事件對象 if ( src && src.type ) { this.originalEvent = src; this.type = src.type; //事件冒泡的文檔可能被標記為阻止默認事件發(fā)生;這個函數(shù)可以反應是否阻止的標志的正確值 this.isDefaultPrevented = ( src.defaultPrevented || src.returnValue === false || src.getPreventDefault && src.getPreventDefault() ) ? returnTrue : returnFalse; //src為事件類型 } else { this.type = src; } //將明確提供的特征添加到事件對象上 if ( props ) { jQuery.extend( this, props ); } //創(chuàng)建一個時間戳如果傳入的事件不只一個 this.timeStamp = src && src.timeStamp || jQuery.now(); //標記事件已經(jīng)修正過 this[ jQuery.expando ] = true; };
第一步構(gòu)造后的事件對象
第二步,分辨出當前事件是那種事件,然后將對應的屬性一一從瀏覽器本地事件originalEvent中拷貝過來
//創(chuàng)建可寫的事件對象副本,并格式化一些特征名稱 var i, prop, copy, type = event.type, originalEvent = event, fixHook = this.fixHooks[ type ]; if ( !fixHook ) { this.fixHooks[ type ] = fixHook = //rmouseEvent=/^(?:mouse|contextmenu)|click/ rmouseEvent.test( type ) ? this.mouseHooks : //rkeyEvent=/^key/ rkeyEvent.test( type ) ? this.keyHooks : {}; } //獲得要從原生事件中拷貝過來的屬性列表 copy = fixHook.props ? this.props.concat( fixHook.props ) : this.props; ... //將原生的屬性都拷貝到新的事件上 i = copy.length; while ( i-- ) { prop = copy[ i ]; event[ prop ] = originalEvent[ prop ]; }
第三步,相關(guān)屬性的兼容處理
// IE<9修正target特征值 if ( !event.target ) { event.target = originalEvent.srcElement || document; } // Chrome 23+, Safari?,Target特征值不能是文本節(jié)點 if ( event.target.nodeType === 3 ) { event.target = event.target.parentNode; } // IE<9,對于鼠標/鍵盤事件, 如果metaKey沒有定義則設(shè)置metaKey==false event.metaKey = !!event.metaKey; //調(diào)用hooks的filter return fixHook.filter ? fixHook.filter( event, originalEvent ) : event;
最后那句代碼針對鼠標事件和鍵盤事件做兼容適配處理。
fixHook.filter可能是jQuery.event.keyHooks.filter
keyHooks.filter: function( event, original ) { //給鍵盤事件添加which特征值 if ( event.which == null ) { event.which = original.charCode != null ? original.charCode : original.keyCode; } return event; }
或這jQuery.event.mouseHooks.filter
mouseHooks.filter: function( event, original ) { var body, eventDoc, doc, button = original.button, fromElement = original.fromElement; //如果事件pageX/Y特征不見了,用可用的clientX/Y來計算出來 if ( event.pageX == null && original.clientX != null ) { eventDoc = event.target.ownerDocument || document; doc = eventDoc.documentElement; body = eventDoc.body; event.pageX = original.clientX + ( doc && doc.scrollLeft || body && body.scrollLeft || 0 ) - ( doc && doc.clientLeft || body && body.clientLeft || 0 ); event.pageY = original.clientY + ( doc && doc.scrollTop || body && body.scrollTop || 0 ) - ( doc && doc.clientTop || body && body.clientTop || 0 ); } //如果必要的話添加relatedTarget特征 if ( !event.relatedTarget && fromElement ) { event.relatedTarget = fromElement === event.target ? original.toElement : fromElement; } //添加點擊事件which特征值: 1 === left; 2 === middle; 3 === right //備注:button不標準,因此不要是使用 if ( !event.which && button !== undefined ) { event.which = ( button & 1 ? 1 : ( button & 2 ? 3 : ( button & 4 ? 2 : 0 ) ) ); } return event; }
構(gòu)建完成的最新事件對象如下(以鼠標事件為例)
原生的事件保存在了originalEvent中,target保存了目標節(jié)點(委托的節(jié)點、事件源),其他信息略過
b. 重載事件方法
構(gòu)建新的事件對象event = new jQuery.Event( originalEvent )時,事件會繼承jQuery.event.prototype中的方法。來看一看有哪些方法
前面分析了jQuery.event.prototype中重載了stopPropagation方法的作用:處了調(diào)用事件對象的阻止冒泡方法以外,還有一個作用就是被委托節(jié)點有多個被委托事件處理等待處理時,其中一個事件調(diào)用了event.stopPropagation()將阻止后續(xù)事件處理的執(zhí)行。點擊這里搜索關(guān)鍵字查看
preventDefault函數(shù)也是有類似的作用。preventDefault函數(shù)中增加了這段代碼
this.isPropagationStopped = returnTrue;
在觸發(fā)事件trigger函數(shù)和模擬冒泡simulate函數(shù)中都會根據(jù)isPropagationStopped()判斷是否要執(zhí)行DOM節(jié)點的默認操作。源碼如下
isImmediatePropagationStopped是stopPropagation特殊用法,isImmediatePropagationStopped會直接阻止掉當前的處理和后面等待執(zhí)行的事件處理,而stopPropagation會執(zhí)行完當前的處理,然后阻止后面等待執(zhí)行的事件處理。
源碼如下
// jQuery.Event基于DOM事件所指定的ECMAScript語言綁定 // http://www.w.org/TR//WD-DOM-Level--Events-/ecma-script-binding.html jQuery.Event.prototype = { isDefaultPrevented: returnFalse, isPropagationStopped: returnFalse, isImmediatePropagationStopped: returnFalse, preventDefault: function() { var e = this.originalEvent; this.isDefaultPrevented = returnTrue; if ( !e ) {return; } if ( e.preventDefault ) { e.preventDefault(); //IE支持 } else { e.returnValue = false; } }, stopPropagation: function() { var e = this.originalEvent; this.isPropagationStopped = returnTrue; if ( !e ) {return; } if ( e.stopPropagation ) { e.stopPropagation(); } // IE支持 e.cancelBubble = true; }, stopImmediatePropagation: function() { this.isImmediatePropagationStopped = returnTrue; this.stopPropagation(); } }
以上就是本文給大家介紹的jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件包裝,希望大家喜歡。
- AJAX 驗證框架13個
- jquery 框架使用教程 AJAX篇
- Jquery AJAX 框架的使用方法
- 基于JQuery框架的AJAX實例代碼
- javascript之AJAX框架使用說明
- asp.net省市三級聯(lián)動的DropDownList+Ajax的三種框架(aspnet/Jquery/ExtJs)示例
- 簡單的前端js+ajax 購物車框架(入門篇)
- jQuery1.9.1針對checkbox的調(diào)整方法(prop)
- 零基礎(chǔ)學習AJAX之AJAX框架
- jQuery 1.9.1源碼分析系列(十)事件系統(tǒng)之綁定事件
- jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件體系結(jié)構(gòu)
- Jquery1.9.1源碼分析系列(六)延時對象應用之jQuery.ready
- Jquery-1.9.1源碼分析系列(十一)之DOM操作
- jQuery 1.9.1源碼分析系列(十三)之位置大小操作
- jQuery 1.9.1源碼分析系列(十四)之常用jQuery工具
- jQuery1.9.1源碼分析系列(十六)ajax之a(chǎn)jax框架
相關(guān)文章
jQuery插件jqGrid動態(tài)獲取列和列字段的方法
這篇文章主要介紹了jQuery插件jqGrid動態(tài)獲取列和列字段的方法,結(jié)合實例形式分析了表格插件jqGrid針對表格字段屬性相關(guān)操作技巧,需要的朋友可以參考下2017-03-03基于CSS3和jQuery實現(xiàn)跟隨鼠標方位的Hover特效
這篇文章主要介紹了基于CSS3和jQuery實現(xiàn)跟隨鼠標方位的Hover特效的相關(guān)資料,需要的朋友可以參考下2016-07-07頁面加載完成后再執(zhí)行JS的jquery寫法以及區(qū)別說明
本篇文章主要是對頁面加載完成后再執(zhí)行JS的jquery寫法以及區(qū)別進行了介紹,需要的朋友可以過來參考下,希望對大家有所幫助2014-02-02