成人精品一区二区三区中文字幕-成人精品一区二区三区-成人精品一级毛片-成人精品亚洲-日本在线视频一区二区-日本在线视频免费

導航首頁 ? 技術教程 ? jQuery-1.9.1源碼分析系列(十)事件系統之事件包裝
全站頭部文字 我要出現在這里
jQuery-1.9.1源碼分析系列(十)事件系統之事件包裝 724 2024-03-20   

在上篇文章給大家介紹了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.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源碼分析系列(十)事件系統之事件包裝,希望大家喜歡。



主站蜘蛛池模板: 奥赛健美| 学校急招水电工一名| 成毅壁纸| 师奶madam 电视剧| 西游记tvb| 盲辉| 05s502图集| 黄视频免费在线播放| 色域在线| 澳亚卫视| 97理伦| 郑艳丽曹查理主演的影片| 报团云南旅游价格| 长谷川未来| 玉匣记白话全书| 邓佳佳| 吉泽明步电影| 座头市 电影| 金三角电影| 电影 英雄| 抖抖2| 以家人之名小说原著| 蓝家宝电影| 女同恋性吃奶舌吻完整版| 约翰尼·西蒙斯| 爱秀直播| 白上之黑电影| 檀健次壁纸| 《愉悦与苦痛》电影| ab变频器中文说明书| 在线免费电影| 海滩乱淫欧美aⅴ大片| 锦绣南歌免费看| 韩宝仪个人简历| 电影喜剧明星演员表| 枕边凶灵完整版免费播放| 日韩在线日韩| 无常电影| 2006年全国一卷语文作文| 我的公公电影| 触底反弹电影|

!!!站長長期在線接!!!

網站、小程序:定制開發/二次開發/仿制開發等

各種疑難雜癥解決/定制接口/定制采集等

站長微信:lxwl520520

站長QQ:1737366103