jQuery-1.9.1源码分析系列(十)事件系统之事件包装


Posted in Javascript onNovember 20, 2015

在上篇文章给大家介绍了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的事件对象。

事件分两种:鼠标事件和键盘事件(不知道触摸事件何时能加进来)。看一下这两者的详细属性

jQuery-1.9.1源码分析系列(十)事件系统之事件包装jQuery-1.9.1源码分析系列(十)事件系统之事件包装

其中有些是浏览器自己的,非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;
};

第一步构造后的事件对象

jQuery-1.9.1源码分析系列(十)事件系统之事件包装

第二步,分辨出当前事件是那种事件,然后将对应的属性一一从浏览器本地事件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;
}

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

jQuery-1.9.1源码分析系列(十)事件系统之事件包装

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

b. 重载事件方法

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

jQuery-1.9.1源码分析系列(十)事件系统之事件包装

前面分析了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源码分析系列(十)事件系统之事件包装,希望大家喜欢。

Javascript 相关文章推荐
JQuery+DIV自定义滚动条样式的具体实现
Jun 25 Javascript
js导出格式化的excel 实例方法
Jul 17 Javascript
JS实现切换标签页效果实例代码
Nov 01 Javascript
jQuery validate插件功能与用法详解
Dec 15 Javascript
Vue学习笔记进阶篇之函数化组件解析
Jul 21 Javascript
Vue.js组件通信的几种姿势
Oct 23 Javascript
详解vue2.0监听属性的使用心得及搭配计算属性的使用
Jul 18 Javascript
vue 实现左右拖拽元素并且不超过他的父元素的宽度
Nov 30 Javascript
vue+iview动态渲染表格详解
Mar 19 Javascript
详解JavaScript中关于this指向的4种情况
Apr 18 Javascript
Vue项目中使用flow做类型检测的方法
Mar 18 Javascript
JQuery插件tablesorter表格排序实现过程解析
May 28 jQuery
Jquery ajax基础教程
Nov 20 #Javascript
谈谈Jquery ajax中success和complete有哪些不同点
Nov 20 #Javascript
jQuery实现带玻璃流光质感的手风琴特效
Nov 20 #Javascript
JQuery.Ajax()的data参数类型实例详解
Nov 20 #Javascript
Jquery Ajax Error 调试错误的技巧
Nov 20 #Javascript
jQuery实现简单的文件上传进度条效果
Mar 26 #Javascript
jquery实现手风琴效果
Nov 20 #Javascript
You might like
让CodeIgniter的ellipsize()支持中文截断的方法
2014/06/12 PHP
详谈php中 strtr 和 str_replace 的效率问题
2017/05/14 PHP
thinkPHP5框架闭包函数与子查询传参用法示例
2018/08/02 PHP
php转换上传word文件为PDF的方法【基于COM组件】
2019/06/10 PHP
JScript的条件编译
2007/05/29 Javascript
JavaScript获取GridView选择的行内容
2009/04/14 Javascript
javascript 写类方式之四
2009/07/05 Javascript
10个基于jQuery或JavaScript的WYSIWYG 编辑器整理
2010/05/06 Javascript
Javascript获取当前日期的农历日期代码
2014/10/08 Javascript
JavaScript实现向setTimeout执行代码传递参数的方法
2015/04/16 Javascript
javascript实现3D变换的立体圆圈实例
2015/08/06 Javascript
Node.js DES加密的简单实现
2016/07/07 Javascript
angular源码学习第一篇 setupModuleLoader方法
2016/10/20 Javascript
JavaScript仿支付宝6位数字密码输入框
2016/12/29 Javascript
angular+webpack2实战例子
2017/05/23 Javascript
Vue2.0 从零开始_环境搭建操作步骤
2017/06/14 Javascript
利用Vue2.x开发实现JSON树的方法
2018/01/04 Javascript
在Vue中使用highCharts绘制3d饼图的方法
2018/02/08 Javascript
轻松搞定jQuery+JSONP跨域请求的解决方案
2018/03/06 jQuery
JS实现的JSON数组去重算法示例
2018/04/11 Javascript
Angular 2使用路由自定义弹出组件toast操作示例
2019/05/10 Javascript
vue总线机制(bus)知识点详解
2020/05/10 Javascript
Python中用于转换字母为小写的lower()方法使用简介
2015/05/19 Python
tensorflow输出权重值和偏差的方法
2018/02/10 Python
分享Pycharm中一些不为人知的技巧
2018/04/03 Python
python list元素为tuple时的排序方法
2018/04/18 Python
利用Python如何实现数据驱动的接口自动化测试
2018/05/11 Python
犹他州最古老的体育用品公司:Al’s
2020/12/18 全球购物
领班岗位职责范文
2014/02/06 职场文书
21岁生日感言
2014/02/27 职场文书
中学生寄语大全
2014/04/03 职场文书
室内设计专业毕业生求职信
2014/05/02 职场文书
感恩祖国演讲稿
2014/09/09 职场文书
信用卡逾期证明示例
2014/09/13 职场文书
2016年教师学习教师法心得体会
2016/01/20 职场文书
HTML页面中使两个div并排显示的实现
2022/05/15 HTML / CSS