首先需要明白,浏览器的原生事件是只读的,限制了jQuery对他的操作。举个简单的例子就能明白为什么jQuery非要构造一个新的事件对象。

  在委托处理中,a节点委托b节点在a被click的时候执行fn函数。当事件冒泡到b节点,执行fn的时候上下文环境需要保证正确,是a节点执行了fn而非b节点。如何保证执行fn的上下文环境是a节点的:看源码(红色部分)

  1. //执行
  2. 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之前还做了一步操作

  1. 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)的源码如下

  1. jQuery.Event = function( src, props ) {
  2. // Allow instantiation without the 'new' keyword
  3. if ( !(this instanceof jQuery.Event) ) {
  4. return new jQuery.Event( src, props );
  5. }
  6.  
  7. //src为事件对象
  8. if ( src && src.type ) {
  9. this.originalEvent = src;
  10. this.type = src.type;
  11.  
  12. //事件冒泡的文档可能被标记为阻止默认事件发生;这个函数可以反应是否阻止的标志的正确值
  13. this.isDefaultPrevented = ( src.defaultPrevented || src.returnValue === false ||
  14. src.getPreventDefault && src.getPreventDefault() ) ? returnTrue : returnFalse;
  15.  
  16. //src为事件类型
  17. } else {
  18. this.type = src;
  19. }
  20.  
  21. //将明确提供的特征添加到事件对象上
  22. if ( props ) {
  23. jQuery.extend( this, props );
  24. }
  25.  
  26. //创建一个时间戳如果传入的事件不只一个
  27. this.timeStamp = src && src.timeStamp || jQuery.now();
  28.  
  29. //标记事件已经修正过
  30. this[ jQuery.expando ] = true;
  31. };

  第一步构造后的事件对象

  

  第二步,分辨出当前事件是那种事件,然后将对应的属性一一从浏览器本地事件originalEvent中拷贝过来

  1.   //创建可写的事件对象副本,并格式化一些特征名称
  2. var i, prop, copy,
  3. type = event.type,
  4. originalEvent = event,
  5. fixHook = this.fixHooks[ type ];
  6.  
  7. if ( !fixHook ) {
  8. this.fixHooks[ type ] = fixHook =
  9. //rmouseEvent=/^(?:mouse|contextmenu)|click/
  10. rmouseEvent.test( type ) ? this.mouseHooks :
  11. //rkeyEvent=/^key/
  12. rkeyEvent.test( type ) ? this.keyHooks :
  13. {};
  14. }
      //获得要从原生事件中拷贝过来的属性列表
  15. copy = fixHook.props ? this.props.concat( fixHook.props ) : this.props;
  16.   ...
  17.   //将原生的属性都拷贝到新的事件上
  18. i = copy.length;
  19. while ( i-- ) {
  20. prop = copy[ i ];
  21. event[ prop ] = originalEvent[ prop ];
  22. }

  第三步,相关属性的兼容处理

  1. // IE<9修正target特征值
  2. if ( !event.target ) {
  3. event.target = originalEvent.srcElement || document;
  4. }
  5.  
  6. // Chrome 23+, Safari?,Target特征值不能是文本节点
  7. if ( event.target.nodeType === 3 ) {
  8. event.target = event.target.parentNode;
  9. }
  10.  
  11. // IE<9,对于鼠标/键盘事件, 如果metaKey没有定义则设置metaKey==false
  12. event.metaKey = !!event.metaKey;
  13.  
  14. //调用hooks的filter
  15. return fixHook.filter ? fixHook.filter( event, originalEvent ) : event;

  最后那句代码针对鼠标事件和键盘事件做兼容适配处理。

  fixHook.filter可能是jQuery.event.keyHooks.filter

  1. keyHooks.filter: function( event, original ) {
  2.  
  3. //给键盘事件添加which特征值
  4. if ( event.which == null ) {
  5. event.which = original.charCode != null ? original.charCode : original.keyCode;
  6. }
  7.  
  8. return event;
  9. }

  或这jQuery.event.mouseHooks.filter

  1. mouseHooks.filter: function( event, original ) {
  2. var body, eventDoc, doc,
  3.   button = original.button,
  4.   fromElement = original.fromElement;
  5.  
  6. //如果事件pageX/Y特征不见了,用可用的clientX/Y来计算出来
  7. if ( event.pageX == null && original.clientX != null ) {
  8. eventDoc = event.target.ownerDocument || document;
  9. doc = eventDoc.documentElement;
  10. body = eventDoc.body;
  11.  
  12. event.pageX = original.clientX + ( doc && doc.scrollLeft || body && body.scrollLeft || 0 ) - ( doc && doc.clientLeft || body && body.clientLeft || 0 );
  13. event.pageY = original.clientY + ( doc && doc.scrollTop || body && body.scrollTop || 0 ) - ( doc && doc.clientTop || body && body.clientTop || 0 );
  14. }
  15.  
  16. //如果必要的话添加relatedTarget特征
  17. if ( !event.relatedTarget && fromElement ) {
  18. event.relatedTarget = fromElement === event.target ? original.toElement : fromElement;
  19. }
  20.  
  21. //添加点击事件which特征值: 1 === left; 2 === middle; 3 === right
  22. //备注:button不标准,因此不要是使用
  23. if ( !event.which && button !== undefined ) {
  24. event.which = ( button & 1 ? 1 : ( button & 2 ? 3 : ( button & 4 ? 2 : 0 ) ) );
  25. }
  26.  
  27. return event;
  28. }

  构建完成的最新事件对象如下(以鼠标事件为例)

  

  原生的事件保存在了originalEvent中,target保存了目标节点(委托的节点、事件源),其他信息略过

  

b. 重载事件方法


  构建新的事件对象event = new jQuery.Event( originalEvent )时,事件会继承jQuery.event.prototype中的方法。来看一看有哪些方法

  

  前面分析了jQuery.event.prototype中重载了stopPropagation方法的作用:处了调用事件对象的阻止冒泡方法以外,还有一个作用就是被委托节点有多个被委托事件处理等待处理时,其中一个事件调用了event.stopPropagation()将阻止后续事件处理的执行。点击这里搜索关键字查看

  

  preventDefault函数也是有类似的作用。preventDefault函数中增加了这段代码

  1. this.isDefaultPrevented = returnTrue;

  在触发事件trigger函数和模拟冒泡simulate函数中都会根据isDefaultPrevented()判断是否要执行DOM节点的默认操作。

  

  isImmediatePropagationStopped是stopPropagation特殊用法,isImmediatePropagationStopped会直接阻止掉当前的处理和后面等待执行的事件处理,而stopPropagation会执行完当前的处理,然后阻止后面等待执行的事件处理。

  源码如下

  1. // jQuery.Event基于DOM3事件所指定的ECMAScript语言绑定
  2. // http://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html
  3. jQuery.Event.prototype = {
  4.   isDefaultPrevented: returnFalse,
  5.   isPropagationStopped: returnFalse,
  6.   isImmediatePropagationStopped: returnFalse,
  7.  
  8.   preventDefault: function() {
  9.     var e = this.originalEvent;
  10.     this.isDefaultPrevented = returnTrue;
  11.     if ( !e ) {return; }
  12.  
  13.     if ( e.preventDefault ) {
  14.       e.preventDefault();
  15.  
  16.     //IE支持
  17.     } else {
  18.       e.returnValue = false;
  19.     }
  20.   },
  21.   stopPropagation: function() {
  22.     var e = this.originalEvent;
  23.     this.isPropagationStopped = returnTrue;
  24.     if ( !e ) {return; }
  25.  
  26.     if ( e.stopPropagation ) {
  27.       e.stopPropagation();
  28.     }
  29.  
  30.     // IE支持
  31.     e.cancelBubble = true;
  32.   },
  33.   stopImmediatePropagation: function() {
  34.     this.isImmediatePropagationStopped = returnTrue;
  35.     this.stopPropagation();
  36.   }
  37. }

  如果觉得本文不错,请点击右下方【推荐】!

jQuery-1.9.1源码分析系列(十) 事件系统——事件包装的更多相关文章

  1. jQuery源码分析系列

    声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://git ...

  2. [转]jQuery源码分析系列

    文章转自:jQuery源码分析系列-Aaron 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAaro ...

  3. jQuery源码分析系列(转载来源Aaron.)

    声明:非本文原创文章,转载来源原文链接Aaron. 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAa ...

  4. jQuery源码分析系列——来自Aaron

    jQuery源码分析系列——来自Aaron 转载地址:http://www.cnblogs.com/aaronjs/p/3279314.html 版本截止到2013.8.24 jQuery官方发布最新 ...

  5. jQuery-1.9.1源码分析系列完毕目录整理

    jQuery 1.9.1源码分析已经完毕.目录如下 jQuery-1.9.1源码分析系列(一)整体架构 jQuery-1.9.1源码分析系列(一)整体架构续 jQuery-1.9.1源码分析系列(二) ...

  6. jquery2源码分析系列

    学习jquery的源码对于提高前端的能力很有帮助,下面的系列是我在网上看到的对jquery2的源码的分析.等有时间了好好研究下.我们知道jquery2开始就不支持IE6-8了,从jquery2的源码中 ...

  7. MyCat源码分析系列之——结果合并

    更多MyCat源码分析,请戳MyCat源码分析系列 结果合并 在SQL下发流程和前后端验证流程中介绍过,通过用户验证的后端连接绑定的NIOHandler是MySQLConnectionHandler实 ...

  8. MyCat源码分析系列之——SQL下发

    更多MyCat源码分析,请戳MyCat源码分析系列 SQL下发 SQL下发指的是MyCat将解析并改造完成的SQL语句依次发送至相应的MySQL节点(datanode)的过程,该执行过程由NonBlo ...

  9. MyCat源码分析系列之——BufferPool与缓存机制

    更多MyCat源码分析,请戳MyCat源码分析系列 BufferPool MyCat的缓冲区采用的是java.nio.ByteBuffer,由BufferPool类统一管理,相关的设置在SystemC ...

  10. MyCat源码分析系列之——前后端验证

    更多MyCat源码分析,请戳MyCat源码分析系列 MyCat前端验证 MyCat的前端验证指的是应用连接MyCat时进行的用户验证过程,如使用MySQL客户端时,$ mysql -uroot -pr ...

随机推荐

  1. Palm是一家英國智能手機公司

    據TCL方面介紹,本次收購只涉及品牌,不會涉及員工和其他資產.被收購之後,Palm仍將繼續把總部設於美國加州矽谷,以發揮該區域所獨有的先進技術和人才的優勢. TCL通訊CEO郭愛平表示TCL將把Pal ...

  2. 推荐几个Android自定义的进度条(转载)

    CustomLoading ElasticDownload Circle-Progress-View lzyzsdCircleProgress SquareProgressBar materialis ...

  3. Oracle like查询

    查询方式:LIKE '%xx%' 普通: SELECT * FROM TABLE T WHERE T.COLUNM LIKE '%xx%' 优化:使用 INSTR SELECT * FROM TABL ...

  4. CSS的定位

        定位的基本思想:允许你定义元素框相对于其正常位置应该出现的位置,或者相对于父元素.另一个元素甚至浏览器窗口本身的位置        一切皆为框   div.h1 或 p 元素常常被称为块级元素 ...

  5. CQOI 2016 k远点对

    题目大意:n个点,求第k远的点对的距离 KD树裸题 注意要用堆维护第k远 #include<bits/stdc++.h> #define ll unsigned long long #de ...

  6. 最近碰到的一些 SSL 问题记录

    最近碰到一些 SSL 的小问题,特记录下. 我们有个 Java 实现的 SSL TCP 服务端,为客户端(PC.Android 和 iOS)提供 SSL 接入连接服务.最近有用户反馈其手机上 App ...

  7. Visual Studio 2013 Ultimate的可视化代码功能

    可视化和了解代码综合了如何使用visual studio可视化代码来帮助理解代码: 理解代码和代码之间的关系:(1)Code Map(2)Dependency Graphs 理解代码交互:Sequen ...

  8. Nova PhoneGap框架 第九章 控件

    我们的框架中也提供了一些常用的控件,这些控件大多都依赖于我们的框架,也正是在我们的框架下才使得实现这些控件的变得更简单.但是我们的框架是不依赖与这些控件的,如果你用不上这些控件,你完全可以把相关的代码 ...

  9. 搭建域服务器和DNS

    标签:SQL SERVER/MSSQL SERVER/数据库/DBA/域控制器 概述 因为很多高性能高可用方案都会在域环境中组建,所以了解创建域的一些知识对搭建那些高可用方案很有必要. 环境:wind ...

  10. Angular 2 最终版正式发布

    9月15日,Angular 2 的最终版正式发布了. 作为 Angular 1 的全平台继任者 -- Angular 2 的最终版,意味着什么? 意味着稳定性已经得到了大范围用例的验证: 意味着已经针 ...