jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件包裝
在上篇文章給大家介紹了jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件體系結(jié)構(gòu),本篇繼續(xù)給大家介紹jquery1.9.1源碼分析系列相關(guān)知識(shí),具體內(nèi)容請(qǐng)看下文吧。
首先需要明白,瀏覽器的原生事件是只讀的,限制了jQuery對(duì)他的操作。舉個(gè)簡(jiǎn)單的例子就能明白為什么jQuery非要構(gòu)造一個(gè)新的事件對(duì)象。
在委托處理中,a節(jié)點(diǎn)委托b節(jié)點(diǎn)在a被click的時(shí)候執(zhí)行fn函數(shù)。當(dāng)事件冒泡到b節(jié)點(diǎn),執(zhí)行fn的時(shí)候上下文環(huán)境需要保證正確,是a節(jié)點(diǎn)執(zhí)行了fn而非b節(jié)點(diǎn)。如何保證執(zhí)行fn的上下文環(huán)境是a節(jié)點(diǎn)的:看源碼(紅色部分)
//執(zhí)行
ret = ( (jQuery.event.special[ handleObj.origType ] || {}).handle || handleObj.handler ).apply( matched.elem, args );
使用了apply將執(zhí)行函數(shù)的上下文替換成了a節(jié)點(diǎn)(matched.elem)。還有一點(diǎn)args[0]即是事件對(duì)象event。又如何保證event是a節(jié)點(diǎn)的事件的?這就是event.currentTarget這個(gè)重要的屬性的功能,所以在執(zhí)行apply之前還做了一步操作
event.currentTarget = matched.elem;
直接更改事件對(duì)象的currentTarget屬性,這在瀏覽器本地事件是做不到的。所以才有了基于本地事件構(gòu)造jQuery的事件對(duì)象。
事件分兩種:鼠標(biāo)事件和鍵盤(pán)事件(不知道觸摸事件何時(shí)能加進(jìn)來(lái))??匆幌逻@兩者的詳細(xì)屬性


其中有些是瀏覽器自己的,非W3C標(biāo)準(zhǔn)的。jQuery將事件屬性分為三塊
鼠標(biāo)和鍵盤(pán)事件共同擁有的屬性jQuery.event.props: "altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" ")
鍵盤(pán)事件專有的屬性jQuery.event.keyHooks.props: "char charCode key keyCode".split(" ")
鼠標(biāo)事件專有的屬性jQuery.event.mouseHooks.props: "button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" ")
a. 構(gòu)造新的事件對(duì)象jQuery.event.fix(originalEvent)
構(gòu)造新的事件對(duì)象分三步完成
第一步,使用到event = new jQuery.Event( originalEvent ),構(gòu)造新事件對(duì)象(不明白new的作用的請(qǐng)點(diǎn)擊這里),并在創(chuàng)建事件的時(shí)候加上isDefaultPrevented、originalEvent、type 、timeStamp和事件已經(jīng)被修正過(guò)的標(biāo)記(優(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為事件對(duì)象
if ( src && src.type ) {
this.originalEvent = src;
this.type = src.type;
//事件冒泡的文檔可能被標(biāo)記為阻止默認(rèn)事件發(fā)生;這個(gè)函數(shù)可以反應(yīng)是否阻止的標(biāo)志的正確值
this.isDefaultPrevented = ( src.defaultPrevented || src.returnValue === false ||
src.getPreventDefault && src.getPreventDefault() ) ? returnTrue : returnFalse;
//src為事件類型
} else {
this.type = src;
}
//將明確提供的特征添加到事件對(duì)象上
if ( props ) {
jQuery.extend( this, props );
}
//創(chuàng)建一個(gè)時(shí)間戳如果傳入的事件不只一個(gè)
this.timeStamp = src && src.timeStamp || jQuery.now();
//標(biāo)記事件已經(jīng)修正過(guò)
this[ jQuery.expando ] = true;
};
第一步構(gòu)造后的事件對(duì)象

第二步,分辨出當(dāng)前事件是那種事件,然后將對(duì)應(yīng)的屬性一一從瀏覽器本地事件originalEvent中拷貝過(guò)來(lái)
//創(chuàng)建可寫(xiě)的事件對(duì)象副本,并格式化一些特征名稱
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 :
{};
}
//獲得要從原生事件中拷貝過(guò)來(lái)的屬性列表
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é)點(diǎn)
if ( event.target.nodeType === 3 ) {
event.target = event.target.parentNode;
}
// IE<9,對(duì)于鼠標(biāo)/鍵盤(pán)事件, 如果metaKey沒(méi)有定義則設(shè)置metaKey==false
event.metaKey = !!event.metaKey;
//調(diào)用hooks的filter
return fixHook.filter ? fixHook.filter( event, originalEvent ) : event;
最后那句代碼針對(duì)鼠標(biāo)事件和鍵盤(pán)事件做兼容適配處理。
fixHook.filter可能是jQuery.event.keyHooks.filter
keyHooks.filter: function( event, original ) {
//給鍵盤(pán)事件添加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特征不見(jiàn)了,用可用的clientX/Y來(lái)計(jì)算出來(lái)
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;
}
//添加點(diǎn)擊事件which特征值: 1 === left; 2 === middle; 3 === right
//備注:button不標(biāo)準(zhǔn),因此不要是使用
if ( !event.which && button !== undefined ) {
event.which = ( button & 1 ? 1 : ( button & 2 ? 3 : ( button & 4 ? 2 : 0 ) ) );
}
return event;
}
構(gòu)建完成的最新事件對(duì)象如下(以鼠標(biāo)事件為例)

原生的事件保存在了originalEvent中,target保存了目標(biāo)節(jié)點(diǎn)(委托的節(jié)點(diǎn)、事件源),其他信息略過(guò)
b. 重載事件方法
構(gòu)建新的事件對(duì)象event = new jQuery.Event( originalEvent )時(shí),事件會(huì)繼承jQuery.event.prototype中的方法。來(lái)看一看有哪些方法

前面分析了jQuery.event.prototype中重載了stopPropagation方法的作用:處了調(diào)用事件對(duì)象的阻止冒泡方法以外,還有一個(gè)作用就是被委托節(jié)點(diǎn)有多個(gè)被委托事件處理等待處理時(shí),其中一個(gè)事件調(diào)用了event.stopPropagation()將阻止后續(xù)事件處理的執(zhí)行。點(diǎn)擊這里搜索關(guān)鍵字查看
preventDefault函數(shù)也是有類似的作用。preventDefault函數(shù)中增加了這段代碼
this.isPropagationStopped = returnTrue;
在觸發(fā)事件trigger函數(shù)和模擬冒泡simulate函數(shù)中都會(huì)根據(jù)isPropagationStopped()判斷是否要執(zhí)行DOM節(jié)點(diǎn)的默認(rèn)操作。源碼如下
isImmediatePropagationStopped是stopPropagation特殊用法,isImmediatePropagationStopped會(huì)直接阻止掉當(dāng)前的處理和后面等待執(zhí)行的事件處理,而stopPropagation會(huì)執(zhí)行完當(dāng)前的處理,然后阻止后面等待執(zhí)行的事件處理。
源碼如下
// jQuery.Event基于DOM事件所指定的ECMAScript語(yǔ)言綁定
// 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 驗(yàn)證框架13個(gè)
- jquery 框架使用教程 AJAX篇
- Jquery AJAX 框架的使用方法
- 基于JQuery框架的AJAX實(shí)例代碼
- javascript之AJAX框架使用說(shuō)明
- asp.net省市三級(jí)聯(lián)動(dòng)的DropDownList+Ajax的三種框架(aspnet/Jquery/ExtJs)示例
- 簡(jiǎn)單的前端js+ajax 購(gòu)物車框架(入門篇)
- jQuery1.9.1針對(duì)checkbox的調(diào)整方法(prop)
- 零基礎(chǔ)學(xué)習(xí)AJAX之AJAX框架
- jQuery 1.9.1源碼分析系列(十)事件系統(tǒng)之綁定事件
- jQuery-1.9.1源碼分析系列(十)事件系統(tǒng)之事件體系結(jié)構(gòu)
- Jquery1.9.1源碼分析系列(六)延時(shí)對(duì)象應(yīng)用之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 unbind 刪除綁定事件 / 移除標(biāo)簽方法
下面小編就為大家?guī)?lái)一篇詳談jQuery unbind 刪除綁定事件 / 移除標(biāo)簽方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03
jQueryUI如何自定義組件實(shí)現(xiàn)代碼
第一次自定義jQueryUI Widget 又是第一次,現(xiàn)在的感受是jQueryUI Widget能讓你代碼組織得更好,風(fēng)格更一致。2010-11-11
針對(duì)后臺(tái)列表table拖拽比較實(shí)用的jquery拖動(dòng)排序
這篇文章主要為大家詳細(xì)介紹了比較實(shí)用的jquery拖動(dòng)排序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-10-10
使用jquery實(shí)現(xiàn)select添加實(shí)現(xiàn)后臺(tái)權(quán)限添加的效果
使用jquery實(shí)現(xiàn)select添加實(shí)現(xiàn)后臺(tái)權(quán)限添加的效果,需要的朋友可以參考下。2011-05-05
基于jquery實(shí)現(xiàn)自定義的audio
最近接到一個(gè)古早的項(xiàng)目變更,設(shè)計(jì)的音樂(lè)播放的功能是自定義的樣式,對(duì)于jquery的項(xiàng)目第一反應(yīng)是先找插件,然而找了半天沒(méi)找到,最后只能自己實(shí)現(xiàn),文中有詳細(xì)的代碼示例供大家參考,需要的朋友可以參考下2023-12-12
基于jQuery實(shí)現(xiàn)定位導(dǎo)航位置效果
這篇文章主要為大家詳細(xì)介紹了基于jQuery實(shí)現(xiàn)定位導(dǎo)航位置效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
最佳6款用于移動(dòng)網(wǎng)站開(kāi)發(fā)的jQuery 圖片滑塊插件小結(jié)
隨著智能手機(jī)的普及,越來(lái)越多的用戶喜歡通過(guò)手機(jī)中瀏覽網(wǎng)頁(yè)。今天這篇文章為大家推薦最佳6款用于移動(dòng)應(yīng)用的 jQuery 圖片滑塊插件,這些插件很好的處理了觸屏事件,效果平滑,幫助你構(gòu)建用戶體驗(yàn)良好的移動(dòng)網(wǎng)站2012-07-07
jQuery 復(fù)合選擇器應(yīng)用的幾個(gè)例子
這篇文章主要介紹了jQuery 復(fù)合選擇器應(yīng)用的幾個(gè)例子,本文例子所引用的jQuery版本為 jQuery-1.8.3.min.js,喜歡的朋友可以學(xué)習(xí)下2014-09-09
jQuery實(shí)現(xiàn)飛機(jī)大戰(zhàn)游戲
這篇文章主要為大家詳細(xì)介紹了jQuery實(shí)現(xiàn)飛機(jī)大戰(zhàn)游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05

