新浦京娱乐场官网-301net-新浦京娱乐www.301net
做最好的网站

如何保证执行fn的上下文环境是a节点的

jQuery-1.9.1源码分析连串(十) 事件系统——事件包装,jquery-1.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的平地风波目的。

 

  事件分三种:鼠标事件和键盘事件(不领悟触摸事件哪一天能加进去)。看一下这三头的详尽属性

  图片 1图片 2

  当中多少是浏览器自身的,非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(original伊夫nt)


  构造新的事件目的分三步成功

  第一步,使用到event = new jQuery.Event( originalEvent ),构造新事件目的(不知道new的功能的请点击这里),并在开创事件的时候拉长isDefaultPrevented、original伊芙nt、type 、timeStamp和事件已经被查对过的标志(优化利用,幸免不必要的拍卖)。jQuery.伊夫nt(src, props)的源码如下

图片 3jQuery.Event = function( src, props ) { // Allow instantiation without the 'new' keyword if ( !(this instanceof jQuery.伊芙nt) ) { 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; }; View Code

  第一步构造后的风浪指标

  图片 4

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

  创设产生的摩登事件指标如下(以鼠标事件为例)

  图片 5

   原生的风浪保存在了originalEvent中,target保存了对象节点(委托的节点、事件源),别的新闻略过

   

b. 重载事件措施


  创设新的风云指标event = new jQuery.Event( originalEvent )时,事件会继续jQuery.event.prototype中的方法。来看一看有怎样方法

  图片 6

  前面深入分析了jQuery.event.prototype中重载了stopPropagation方法的意义:处了调用事件目的的阻止冒泡方法以外,还有贰个成效就是被托付节点有多少个被信托事件管理等待管理时,在那之中一个事变调用了event.stopPropagation()将阻碍后续事件处理的实施。点击这里寻觅关键字查看

  

  preventDefault函数也有左近的功力。preventDefault函数中加进了这段代码

this.isPropagationStopped = returnTrue;

  在触发事件trigger函数和宪章冒泡simulate函数中都会依照isPropagationStopped()判别是还是不是要实行DOM节点的默许操作。源码如下

  

  isImmediatePropagationStopped是stopPropagation特殊用法,isImmediatePropagationStopped会直接堵住掉当前的拍卖和前边伺机实践的事件管理,而stopPropagation会实践完当前的拍卖,然后阻止前面伺机试行的事件管理。

  源码如下

图片 7// jQuery.伊芙nt基于DOM3事变所钦定的ECMAScript语言绑定 // 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();   } } View Code

 

   假诺感觉本文不错,请点击右下方【推荐】!

事件系统——事件包装,jquery-1.9.1源码 首先须求掌握,浏览器的原闯事件是只读的,限制了jQuery对他的操作...

本文由新浦京娱乐场官网-301net-新浦京娱乐www.301net发布于301net网站建设,转载请注明出处:如何保证执行fn的上下文环境是a节点的

您可能还会对下面的文章感兴趣: