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

首頁 > 編程 > JavaScript > 正文

jQuery-1.9.1源碼分析系列(十)事件系統之事件包裝

2019-11-20 11:13:05
字體:
來源:轉載
供稿:網友

在上篇文章給大家介紹了jQuery-1.9.1源碼分析系列(十)事件系統之事件體系結構,本篇繼續給大家介紹jquery1.9.1源碼分析系列相關知識,具體內容請看下文吧。

首先需要明白,瀏覽器的原生事件是只讀的,限制了jQuery對他的操作。舉個簡單的例子就能明白為什么jQuery非要構造一個新的事件對象。

  在委托處理中,a節點委托b節點在a被click的時候執行fn函數。當事件冒泡到b節點,執行fn的時候上下文環境需要保證正確,是a節點執行了fn而非b節點。如何保證執行fn的上下文環境是a節點的:看源碼(紅色部分)

//執行ret = ( (jQuery.event.special[ handleObj.origType ] || {}).handle || handleObj.handler ).apply( matched.elem, args );

使用了apply將執行函數的上下文替換成了a節點(matched.elem)。還有一點args[0]即是事件對象event。又如何保證event是a節點的事件的?這就是event.currentTarget這個重要的屬性的功能,所以在執行apply之前還做了一步操作

event.currentTarget = matched.elem;

直接更改事件對象的currentTarget屬性,這在瀏覽器本地事件是做不到的。所以才有了基于本地事件構造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. 構造新的事件對象jQuery.event.fix(originalEvent)

構造新的事件對象分三步完成

第一步,使用到event = new jQuery.Event( originalEvent ),構造新事件對象(不明白new的作用的請點擊這里),并在創建事件的時候加上isDefaultPrevented、originalEvent、type 、timeStamp和事件已經被修正過的標記(優化使用,避免不必要的處理)。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;    //事件冒泡的文檔可能被標記為阻止默認事件發生;這個函數可以反應是否阻止的標志的正確值    this.isDefaultPrevented = ( src.defaultPrevented || src.returnValue === false ||      src.getPreventDefault && src.getPreventDefault() ) ? returnTrue : returnFalse;  //src為事件類型  } else {    this.type = src;  }  //將明確提供的特征添加到事件對象上  if ( props ) {    jQuery.extend( this, props );  }  //創建一個時間戳如果傳入的事件不只一個  this.timeStamp = src && src.timeStamp || jQuery.now();  //標記事件已經修正過  this[ jQuery.expando ] = true;};

第一步構造后的事件對象

  

第二步,分辨出當前事件是那種事件,然后將對應的屬性一一從瀏覽器本地事件originalEvent中拷貝過來

 //創建可寫的事件對象副本,并格式化一些特征名稱  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 ];  }

第三步,相關屬性的兼容處理

 // IE<9修正target特征值  if ( !event.target ) {    event.target = originalEvent.srcElement || document;  }  // Chrome 23+, Safari?,Target特征值不能是文本節點  if ( event.target.nodeType === 3 ) {    event.target = event.target.parentNode;  }  // IE<9,對于鼠標/鍵盤事件, 如果metaKey沒有定義則設置metaKey==false  event.metaKey = !!event.metaKey;  //調用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;}

構建完成的最新事件對象如下(以鼠標事件為例)

  

原生的事件保存在了originalEvent中,target保存了目標節點(委托的節點、事件源),其他信息略過

b. 重載事件方法

  構建新的事件對象event = new jQuery.Event( originalEvent )時,事件會繼承jQuery.event.prototype中的方法。來看一看有哪些方法

  

  前面分析了jQuery.event.prototype中重載了stopPropagation方法的作用:處了調用事件對象的阻止冒泡方法以外,還有一個作用就是被委托節點有多個被委托事件處理等待處理時,其中一個事件調用了event.stopPropagation()將阻止后續事件處理的執行。點擊這里搜索關鍵字查看

  preventDefault函數也是有類似的作用。preventDefault函數中增加了這段代碼

this.isPropagationStopped = returnTrue;

在觸發事件trigger函數和模擬冒泡simulate函數中都會根據isPropagationStopped()判斷是否要執行DOM節點的默認操作。源碼如下

isImmediatePropagationStopped是stopPropagation特殊用法,isImmediatePropagationStopped會直接阻止掉當前的處理和后面等待執行的事件處理,而stopPropagation會執行完當前的處理,然后阻止后面等待執行的事件處理。

源碼如下

// jQuery.Event基于DOM事件所指定的ECMAScript語言綁定// http://www.w.org/TR//WD-DOM-Level--Events-/ecma-script-binding.htmljQuery.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源碼分析系列(十)事件系統之事件包裝,希望大家喜歡。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩精品电影网| 日本欧美在线视频| 久久99视频免费| 亚洲欧美国产va在线影院| 国产免费一区二区三区香蕉精| 精品国产一区久久久| 久久91超碰青草是什么| 国产精品免费观看在线| 97精品国产97久久久久久春色| 日韩在线观看av| 国产精品美女午夜av| 欧美人交a欧美精品| 成人国产在线激情| 日韩视频在线免费| 91精品国产网站| 裸体女人亚洲精品一区| 日韩精品视频免费专区在线播放| www.xxxx精品| 日韩在线免费视频| 国内精品400部情侣激情| 欧美日韩视频免费播放| 91精品在线播放| 亚洲成人动漫在线播放| 97在线精品国自产拍中文| 91天堂在线视频| 国产精品国产三级国产aⅴ浪潮| 成人福利网站在线观看11| 亚洲色图50p| 韩国欧美亚洲国产| 97成人精品区在线播放| 欧美黑人xxxⅹ高潮交| 欧美日韩精品在线观看| 亚洲激情视频网| 久久免费成人精品视频| 久久久精品国产| 国产噜噜噜噜久久久久久久久| 国产成人亚洲综合91| 精品国产欧美一区二区五十路| 久99九色视频在线观看| 91极品女神在线| www日韩中文字幕在线看| 精品性高朝久久久久久久| 97久久精品视频| 欧美猛交ⅹxxx乱大交视频| 日韩中文字幕国产精品| 欧美成人激情视频| 欧美肥老妇视频| 色婷婷综合久久久久中文字幕1| 精品一区二区三区四区| 国产99视频精品免视看7| 日本亚洲欧美成人| 亚洲精品国精品久久99热一| 日韩最新中文字幕电影免费看| 色久欧美在线视频观看| 欧美精品videofree1080p| 欧美放荡办公室videos4k| 成人国产亚洲精品a区天堂华泰| 日韩在线观看精品| 欧美午夜丰满在线18影院| 欧美日韩中文字幕在线视频| 欧亚精品在线观看| 国产在线视频欧美| 欧美疯狂xxxx大交乱88av| 亚洲精品国产综合久久| 国内精久久久久久久久久人| 亚洲男人7777| 亚洲精品美女在线观看| 欧美成人性色生活仑片| 久久久国产视频| 久久亚洲国产精品| 亚洲精品国产精品乱码不99按摩| 欧美日韩中文字幕| 亚洲色图五月天| 国产日韩av在线| 国产婷婷色综合av蜜臀av| 国产精品久久久久久久久久久久久久| 亚洲图片在线综合| 曰本色欧美视频在线| 亚洲男人天堂九九视频| 国产精品精品国产| 91国偷自产一区二区三区的观看方式| 国产精品福利小视频| 精品国产91乱高清在线观看| 国产精品扒开腿做爽爽爽的视频| 亚洲精品日产aⅴ| 色偷偷888欧美精品久久久| 国产欧美精品久久久| 国产精品一区二区久久国产| 国产精品91一区| 国产91|九色| 欧美黄色www| 亚洲美女av网站| 97碰碰碰免费色视频| 国模视频一区二区| 日韩成人xxxx| 亚洲成人黄色网址| 91免费看片在线| 91欧美日韩一区| 国产精品美女www爽爽爽视频| www日韩中文字幕在线看| 国产欧美一区二区三区在线| 国产精品久久久久久影视| 精品久久香蕉国产线看观看亚洲| 国产精品日韩久久久久| 日本精品久久中文字幕佐佐木| 久久亚洲精品网站| 亚洲欧洲在线观看| 国产精品18久久久久久麻辣| 大伊人狠狠躁夜夜躁av一区| 欧美黑人xxxⅹ高潮交| 日韩欧美第一页| 91爱视频在线| 欧美黑人极品猛少妇色xxxxx| 国产精品第2页| 欧美激情欧美激情在线五月| 日韩av电影在线播放| 中文字幕日韩精品在线观看| 日韩亚洲欧美成人| 欧美一区二区三区……| 国产精品免费看久久久香蕉| 久热在线中文字幕色999舞| 日韩国产在线播放| 国产日本欧美视频| 欧美国产精品va在线观看| 亚洲人成啪啪网站| 国产精品 欧美在线| 久久综合色88| 成人高h视频在线| 91精品国产成人www| 欧美肥臀大乳一区二区免费视频| 欧美日韩国产成人在线观看| 日韩av免费在线观看| 久久精品中文字幕| 最近2019中文字幕一页二页| 国产精品久久久久一区二区| 这里只有精品丝袜| 国产精品白嫩初高中害羞小美女| 人人澡人人澡人人看欧美| 在线视频一区二区| 国产精品热视频| 亚洲xxx自由成熟| 最近中文字幕2019免费| 欧美日韩性视频在线| 亚洲激情在线观看视频免费| 日韩欧美国产高清91| 精品久久久香蕉免费精品视频| 国产精品久久久久久影视| 亚洲天堂网站在线观看视频| 亚洲欧美日韩国产成人| 亚洲xxxx做受欧美| 国产亚洲精品美女久久久久| 亚洲大胆人体在线| 亚洲国产成人精品久久久国产成人一区| 在线看国产精品| 欧美另类老肥妇| 久久av.com| 日韩最新中文字幕电影免费看| 国产精品网站视频| 成人午夜高潮视频| 日韩激情视频在线播放| 精品女厕一区二区三区| 欧美整片在线观看| 久久亚洲欧美日韩精品专区| 欧美日韩国产一区二区|