成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

jQuery源碼解析之trigger()

Youngs / 1449人閱讀

摘要:一和的作用和區(qū)別觸發(fā)被選元素上的指定事件以及事件的默認(rèn)行為比如表單提交不會(huì)引起事件比如表單提交的默認(rèn)行為觸發(fā)所有匹配元素的指定事件只觸發(fā)第一個(gè)匹配元素的指定事件會(huì)冒泡不會(huì)冒泡二被點(diǎn)擊了作用看一源碼觸發(fā)事件,是自定義事件的額外參數(shù)源碼行解析本

一、$().trigger()和$().triggerHandler() 的作用和區(qū)別

(1)trigger("focus") 觸發(fā)被選元素上的指定事件(focus)以及事件的默認(rèn)行為(比如表單提交);
triggerHandler(xxx) 不會(huì)引起事件(比如表單提交)的默認(rèn)行為

(2)trigger(xxx) 觸發(fā)所有匹配元素的指定事件;
triggerHandler(xxx) 只觸發(fā)第一個(gè)匹配元素的指定事件

(3)trigger(xxx) 會(huì)冒泡;
triggerHandler(xxx) 不會(huì)冒泡

二、$().trigger()

 $("#one").on("click",function () {
   console.log("one被點(diǎn)擊了")
 })
  
 $("#one").trigger("click")

作用:
看 一、(1)

源碼:

    //觸發(fā)type事件,data是自定義事件的額外參數(shù)
    //源碼9014行
    trigger: function( type, data ) {
      return this.each( function() {
        jQuery.event.trigger( type, data, this );
      } );
    },

解析:
本質(zhì)是調(diào)用的jQuery.event.trigger()方法

三、jQuery.event.trigger()

源碼:

    //源碼8850行
    //type, data, this
    trigger: function( event, data, elem, onlyHandlers ) {
      var i, cur, tmp, bubbleType, ontype, handle, special, lastElement,
        //冒泡路徑數(shù)組
        eventPath = [ elem || document ],
        //判斷event是否有"type"屬性,有則取event.type,沒(méi)有則取event
        type = hasOwn.call( event, "type" ) ? event.type : event,
        //同上
        namespaces = hasOwn.call( event, "namespace" ) ? event.namespace.split( "." ) : [];

      //當(dāng)前元素
      cur = lastElement = tmp = elem = elem || document;
      //文本內(nèi)容或者是注釋則不觸發(fā)事件
      // Don"t do events on text and comment nodes
      if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
        return;
      }
      //由focus/blur轉(zhuǎn)變到focusin/out,現(xiàn)在不觸發(fā)focus/blur事件
      // focus/blur morphs to focusin/out; ensure we"re not firing them right now

      //rfocusMorph:focusin focus|focusout blur
      if ( rfocusMorph.test( type + jQuery.event.triggered ) ) {
        return;
      }
      //可以不看
      if ( type.indexOf( "." ) > -1 ) {

        // Namespaced trigger; create a regexp to match event type in handle()
        namespaces = type.split( "." );
        type = namespaces.shift();
        namespaces.sort();
      }
      //onclick,onfocus等等
      ontype = type.indexOf( ":" ) < 0 && "on" + type;
      //event一般是字符串,所以一般是undefined
      //獲取對(duì)應(yīng)type類(lèi)型的jQuery.event
      // Caller can pass in a jQuery.Event object, Object, or just an event type string
      event = event[ jQuery.expando ] ?
        event :
        //click,false
        new jQuery.Event( type, typeof event === "object" && event );

      // Trigger bitmask: & 1 for native handlers; & 2 for jQuery (always true)
      //onlyHandlers一般為undefined
      //3
      event.isTrigger = onlyHandlers ? 2 : 3;
      //""
      event.namespace = namespaces.join( "." );
      //null
      event.rnamespace = event.namespace ?
        new RegExp( "(^|.)" + namespaces.join( ".(?:.*.|)" ) + "(.|$)" ) :
        null;
      //清空event以防它被復(fù)用
      // Clean up the event in case it is being reused
      event.result = undefined;
      //target屬性為目標(biāo)DOM元素
      //我們一般取的e.target.value,也正是目標(biāo)元素的值
      if ( !event.target ) {
        event.target = elem;
      }
      //復(fù)制data并預(yù)先考慮event,創(chuàng)建handler集合
      // Clone any incoming data and prepend the event, creating the handler arg list

      //簡(jiǎn)單點(diǎn),就是 data=[event]
      data = data == null ?
        [ event ] :
        jQuery.makeArray( data, [ event ] );

      //賦值有需要特殊處理的type
      // Allow special events to draw outside the lines
      special = jQuery.event.special[ type ] || {};

      if ( !onlyHandlers && special.trigger && special.trigger.apply( elem, data ) === false ) {
        return;
      }

      //提前確定事件冒泡的路徑
      // Determine event propagation path in advance, per W3C events spec (#9951)
      //冒泡至document,再到window;關(guān)注全局的ownerDocument
      // Bubble up to document, then to window; watch for a global ownerDocument var (#9724)
      if ( !onlyHandlers && !special.noBubble && !isWindow( elem ) ) {
        //click
        bubbleType = special.delegateType || type;

        //clickclick
        //如果不是focus/blur的話,獲取它的父元素
        if ( !rfocusMorph.test( bubbleType + type ) ) {
          cur = cur.parentNode;
        }
        //for循環(huán)的語(yǔ)法(a; b; c)
        //a在單次循環(huán)開(kāi)始前執(zhí)行
        //b是單次循環(huán)的條件(這里即cur存在)
        //c是單次循環(huán)結(jié)束后執(zhí)行
        for ( ; cur; cur = cur.parentNode ) {
          console.log(cur,"cur8967")
          //將目標(biāo)元素的祖先元素都push進(jìn)數(shù)組
          eventPath.push( cur );
          tmp = cur;
        }
        //只有當(dāng)tmp是document時(shí),將window加上
        // Only add window if we got to document (e.g., not plain obj or detached DOM)
        if ( tmp === ( elem.ownerDocument || document ) ) {
          eventPath.push( tmp.defaultView || tmp.parentWindow || window );
        }
      }
      //觸發(fā)冒泡機(jī)制
      // Fire handlers on the event path
      i = 0;
      //e.stopPropagation()這是阻止冒泡的方法
      //isPropagationStopped() 檢查是否阻止冒泡了,返回boolean
      while ( ( cur = eventPath[ i++ ] ) && !event.isPropagationStopped() ) {
        lastElement = cur;
        event.type = i > 1 ?
          bubbleType :
          special.bindType || type;
        console.log(i,"lastElement8987")
        // jQuery handler
        //( dataPriv.get( cur, "events" ) || {} )[ event.type ]
        // 先判斷cur元素的events是否有綁定click
        //dataPriv.get( cur, "handle" ) 
        //再獲取cur元素的click事件處理程序
        //獲取目標(biāo)元素的觸發(fā)事件的事件處理程序
        handle = ( dataPriv.get( cur, "events" ) || {} )[ event.type ] &&
          //獲取觸發(fā)事件的處理程序
          dataPriv.get( cur, "handle" );
        /*讓冒泡元素執(zhí)行handle,這行代碼是觸發(fā)冒泡機(jī)制的關(guān)鍵*/
        /*在執(zhí)行click事件的處理程序后,自然就會(huì)執(zhí)行e.stopPropagation(),
        * 從而讓event.isPropagationStopped()=true*/
        if ( handle ) {
          handle.apply( cur, data );
        }
        //接下來(lái)處理原生的事件及處理程序
        //click為onclick
        // Native handler
        handle = ontype && cur[ ontype ];
        //如果有綁定原生onclick事件的話
        if ( handle && handle.apply && acceptData( cur ) ) {
          //執(zhí)行onclick事件的處理程序
          event.result = handle.apply( cur, data );
          if ( event.result === false ) {
            //阻止元素的默認(rèn)行為(如提交表單submit)
            event.preventDefault();
          }
        }
      }
      
      event.type = type;
      //如果沒(méi)有人阻止默認(rèn)行為的話,現(xiàn)在就阻止
      /*比如觸發(fā)的click事件,但不會(huì)跳轉(zhuǎn)*/
      // If nobody prevented the default action, do it now
      if ( !onlyHandlers && !event.isDefaultPrevented() ) {
        if ( ( !special._default ||
          special._default.apply( eventPath.pop(), data ) === false ) &&
          acceptData( elem ) ) {
          //在目標(biāo)上,用重復(fù)的命名調(diào)用原生DOM事件,會(huì)在window層面上影響其他元素
          // Call a native DOM method on the target with the same name as the event.
          // Don"t do default actions on window, that"s where global variables be (#6170)
          if ( ontype && isFunction( elem[ type ] ) && !isWindow( elem ) ) {
            //當(dāng)我們觸發(fā)FOO事件(如click)時(shí),不要重復(fù)觸發(fā)它的onFOO(onclick)事件
            // Don"t re-trigger an onFOO event when we call its FOO() method
            tmp = elem[ ontype ];
            //將jQuery對(duì)象的onclick屬性置為null
            //比如就不會(huì)去跳轉(zhuǎn)了
            if ( tmp ) {
              elem[ ontype ] = null;
            }
            //阻止重復(fù)觸發(fā)同樣的事件,因?yàn)槲覀円呀?jīng)把它冒泡了
            // Prevent re-triggering of the same event, since we already bubbled it above
            jQuery.event.triggered = type;
            //如果已經(jīng)執(zhí)行阻止冒泡了,則為window添加阻止冒泡的監(jiān)聽(tīng)
            if ( event.isPropagationStopped() ) {
              lastElement.addEventListener( type, stopPropagationCallback );
            }
            console.log(elem[ type ],"type9053")
            //執(zhí)行type事件
            elem[ type ]();
            if ( event.isPropagationStopped() ) {
              lastElement.removeEventListener( type, stopPropagationCallback );
            }

            jQuery.event.triggered = undefined;

            if ( tmp ) {
              elem[ ontype ] = tmp;
            }

          }
        }
      }
      return event.result;
    },

解析:

(1)trigger()的冒泡機(jī)制的實(shí)現(xiàn)

if ( !onlyHandlers && !special.noBubble && !isWindow( elem ) )中,通過(guò)eventPath存儲(chǔ)目標(biāo)元素的祖先元素:

        //clickclick
        //如果不是focus/blur的話,獲取它的父元素
        if ( !rfocusMorph.test( bubbleType + type ) ) {
          cur = cur.parentNode;
        }
        //for循環(huán)的語(yǔ)法(a; b; c)
        //a在單次循環(huán)開(kāi)始前執(zhí)行
        //b是單次循環(huán)的條件(這里即cur存在)
        //c是單次循環(huán)結(jié)束后執(zhí)行
        for ( ; cur; cur = cur.parentNode ) {
          console.log(cur,"cur8967")
          //將目標(biāo)元素的祖先元素都push進(jìn)數(shù)組
          eventPath.push( cur );
          tmp = cur;
        }
        //只有當(dāng)tmp是document時(shí),將window加上
        // Only add window if we got to document (e.g., not plain obj or detached DOM)
        if ( tmp === ( elem.ownerDocument || document ) ) {
          eventPath.push( tmp.defaultView || tmp.parentWindow || window );
        }

通過(guò)eventPath.push(cur. parentNode)將冒泡元素裝進(jìn)數(shù)組中,并通過(guò)while循環(huán)觸發(fā)冒泡機(jī)制

      //觸發(fā)冒泡機(jī)制
      // Fire handlers on the event path
      i = 0;
      //e.stopPropagation()這是阻止冒泡的方法
      //isPropagationStopped() 檢查是否阻止冒泡了,返回boolean
      while ( ( cur = eventPath[ i++ ] ) && !event.isPropagationStopped() ) {
        lastElement = cur;
        event.type = i > 1 ?
          bubbleType :
          special.bindType || type;
        console.log(i,"lastElement8987")
        // jQuery handler
        //( dataPriv.get( cur, "events" ) || {} )[ event.type ]
        // 先判斷cur元素的events是否有綁定click
        //dataPriv.get( cur, "handle" ) 
        //再獲取cur元素的click事件處理程序
        //獲取目標(biāo)元素的觸發(fā)事件的事件處理程序
        handle = ( dataPriv.get( cur, "events" ) || {} )[ event.type ] &&
          //獲取觸發(fā)事件的處理程序
          dataPriv.get( cur, "handle" );
        /*讓冒泡元素執(zhí)行handle,這行代碼是觸發(fā)冒泡機(jī)制的關(guān)鍵*/
        /*在執(zhí)行click事件的處理程序后,自然就會(huì)執(zhí)行e.stopPropagation(),
        * 從而讓event.isPropagationStopped()=true*/
        if ( handle ) {
          handle.apply( cur, data );
        }
        //接下來(lái)處理原生的事件及處理程序
        //click為onclick
        // Native handler
        handle = ontype && cur[ ontype ];
        //如果有綁定原生onclick事件的話
        if ( handle && handle.apply && acceptData( cur ) ) {
          //執(zhí)行onclick事件的處理程序
          event.result = handle.apply( cur, data );
          if ( event.result === false ) {
            //阻止元素的默認(rèn)行為(如提交表單submit)
            event.preventDefault();
          }
        }
      }

關(guān)鍵代碼是handle.apply( cur, data ),它用來(lái)執(zhí)行cur元素的事件的處理程序。

(2)通過(guò)e.stopPropagation()來(lái)阻止冒泡的原理:



這是one

① 上面這段代碼會(huì)先執(zhí)行$("#one").trigger("click")

② trigger()里會(huì)執(zhí)行到上面(1)的handle.apply( cur, data );,而handle會(huì)執(zhí)行$("#one")click事件的處理程序:

      e.stopPropagation()
      console.log("one被點(diǎn)擊了")

e.stopPropagation()走的是這里:

  //event的屬性賦值
  //源碼5749行
  jQuery.Event.prototype = {
    constructor: jQuery.Event,
    //xxx
    isPropagationStopped: returnFalse, //false
    //xxx
    //xxx
    //當(dāng)執(zhí)行e.stopPropagation()后走這邊
    //源碼5767行
    stopPropagation: function() {
      var e = this.originalEvent;
      //isPropagationStopped方法返回true
      this.isPropagationStopped = returnTrue;

      if ( e && !this.isSimulated ) {
        e.stopPropagation();
      }
    },
}

最后讓isPropagationStopped()方法返回true

④ 注意:$().trigger()里的event也就是click里的event,所以會(huì)影響到while循環(huán)地判斷,從而達(dá)到阻止冒泡循環(huán)的 目的

while ( ( cur = eventPath[ i++ ] ) && !event.isPropagationStopped() ) { }

⑤ 為什么說(shuō)click里的event$().trigger()里的event?

     //event一般是字符串,所以一般是undefined
      //獲取對(duì)應(yīng)type類(lèi)型的jQuery.event
      // Caller can pass in a jQuery.Event object, Object, or just an event type string
      event = event[ jQuery.expando ] ?
        event :
        //click,false
        new jQuery.Event( type, typeof event === "object" && event );

因?yàn)?event 是根據(jù)type(click)類(lèi)型生成的,所以trigger里的event的部分屬性和clickevent屬性相同。

(3)原生js綁定的事件的執(zhí)行,如onclick

    $("#one").click(function(e){
      console.log("one被點(diǎn)擊了")
    })

    document.getElementById("one").onclick=function(){
      console.log("onclick被點(diǎn)擊了")
    }

還是在while循環(huán)中:

        //接下來(lái)處理原生的事件及處理程序
        //click為onclick
        // Native handler
        handle = ontype && cur[ ontype ];
        //如果有綁定原生onclick事件的話
        if ( handle && handle.apply && acceptData( cur ) ) {
          //執(zhí)行onclick事件的處理程序
          event.result = handle.apply( cur, data );
          if ( event.result === false ) {
            //阻止元素的默認(rèn)行為(如提交表單submit)
            event.preventDefault();
          }
        }

也就是說(shuō):
在冒泡循環(huán)機(jī)制中,在執(zhí)行完jQuery綁定的handler后,會(huì)接著執(zhí)行原生JS 綁定的handler!

(4)rfocusMorph

  //匹配focusinfocus或者focusoutblur
  //源碼8872行
  var rfocusMorph = /^(?:focusinfocus|focusoutblur)$/,

(5)jQuery.makeArray()

作用:
用于將一個(gè)類(lèi)似數(shù)組的對(duì)象轉(zhuǎn)換為真正的數(shù)組對(duì)象

注意:
類(lèi)數(shù)組對(duì)象具有許多數(shù)組的屬性(例如length屬性,[]數(shù)組訪問(wèn)運(yùn)算符等),不過(guò)它畢竟不是數(shù)組,缺少?gòu)臄?shù)組的原型對(duì)象上繼承下來(lái)的內(nèi)置方法(例如:pop()、reverse()等)。

源碼:

    //結(jié)果僅供內(nèi)部使用
    // results is for internal usage only
    //源碼442行
    makeArray: function( arr, results ) {
      var ret = results || [];

      if ( arr != null ) {
        //Object()等效于new Object()
        //先將arr轉(zhuǎn)為對(duì)象類(lèi)型,因?yàn)閖s中的array是Object
        if ( isArrayLike( Object( arr ) ) ) {
          //將second合并到first后面
          jQuery.merge( ret,
            typeof arr === "string" ?
              [ arr ] : arr
          );
        } else {
          //ret.push(arr)
          push.call( ret, arr );
        }
      }
      //返回array
      return ret;
    },

$.isArrayLike

作用:
判斷是不是類(lèi)數(shù)組

源碼:

  //判斷是不是類(lèi)數(shù)組
  //源碼561行
  function isArrayLike( obj ) {

    // Support: real iOS 8.2 only (not reproducible in simulator)
    // `in` check used to prevent JIT error (gh-2145)
    // hasOwn isn"t used here due to false negatives
    // regarding Nodelist length in IE
    //后兩個(gè)是兼容性考慮的判斷
    var length = !!obj && "length" in obj && obj.length,
      //obj類(lèi)型
      type = toType( obj );

    if ( isFunction( obj ) || isWindow( obj ) ) {
      return false;
    }

    return type === "array" || length === 0 ||
      typeof length === "number" && length > 0 && ( length - 1 ) in obj;
  }

(6)最后一個(gè)if,觸發(fā)trigger()時(shí),阻止jQuery元素的默認(rèn)行為

if ( !onlyHandlers && !event.isDefaultPrevented() ){
xxx
xxx
}

綜上,trigger一共做了三件事:

(1)觸發(fā)冒泡機(jī)制
(2)觸發(fā)原生綁定事件
(3)阻止元素默認(rèn)行為

最后,附上自己整理的觸發(fā) trigger() 的流程圖:

(完)

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/104382.html

相關(guān)文章

  • jQuery源碼解析click()的事件綁定

    摘要:階段二目標(biāo)瀏覽器找到監(jiān)聽(tīng)器后,就運(yùn)行該監(jiān)聽(tīng)器階段三冒泡目標(biāo)到祖在事件自下而上到達(dá)目標(biāo)節(jié)點(diǎn)的過(guò)程中,瀏覽器會(huì)檢測(cè)不是針對(duì)該事件的監(jiān)聽(tīng)器用來(lái)捕獲事件,并運(yùn)行非捕獲事件的監(jiān)聽(tīng)器。注意下這種情況,是在里的具體實(shí)現(xiàn),即調(diào)用一次后,就執(zhí)行,卸載事件。 showImg(https://segmentfault.com/img/remote/1460000019304809); 前言:這篇依舊長(zhǎng),請(qǐng)耐...

    figofuture 評(píng)論0 收藏0
  • jQuery事件綁定到觸發(fā)全過(guò)程及知識(shí)點(diǎn)補(bǔ)充

    摘要:十的觸發(fā)機(jī)制被點(diǎn)擊了元素本身綁定了一個(gè)事件,但是是原生事件,它是靠綁定來(lái)觸發(fā)事件的。 showImg(https://segmentfault.com/img/remote/1460000019505402); 前言:最重要的還是最后的流程圖,可以試著根據(jù)流程圖手寫(xiě)實(shí)現(xiàn)$().on(),下篇文章會(huì)放出模擬實(shí)現(xiàn)的代碼。 一、舉例 這是A 這是C ...

    Jioby 評(píng)論0 收藏0
  • jQuery源碼解析jQuery.event.dispatch()

    摘要:一起源方法最終是用綁定事件的而方法正是等于二作用觸發(fā)綁定的事件的處理程序源碼源碼行即原生觸發(fā)事件的處理程序修正對(duì)象獲取事件的處理程序集合,結(jié)構(gòu)如下從數(shù)據(jù)緩存中獲取事件處理集合即目標(biāo)元素委托目標(biāo)這段代碼壓根不會(huì)執(zhí)行,因?yàn)槿炙阉鳑](méi)找到結(jié)構(gòu) showImg(https://segmentfault.com/img/remote/1460000019464031); 一、起源jQuery.e...

    GraphQuery 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

Youngs

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<