javascript 之事件處理函數和事件對象

本文主要談及問題:

  • 關于編寫跨瀏覽器的事件處理函數和事件對象

關于編寫跨瀏覽器的事件處理函數和事件對象

  • 為什么要編寫跨瀏覽器的事件處理函數和事件對象

近幾年來,個人對WINDOWS越來越沒愛。至于為什么無愛,我想原因有二:

  • 轉投MAC陣營,覺得MAC比起WINDOWS對用戶太友好了
  • 成為了一個前端工程師

講講第二點。網上一直流傳著一個段子:IE都有勇氣讓你把它設為默認瀏覽器,為什么你還連表白的勇氣都沒有。每每看到這個段子,心都有點痛,我想不僅是我,每個前端工程師都會有這樣的感覺。微軟在IE9之前,完全是think different ,make 更加 different。做出來的東西完全跟標準不一致。FE們為了兼容它沒少花心思。就拿事件模型來說,基本上最常用的模型有兩種:一種是DOM L2 EVENT MODEL.另外一種不用說,相信大家都知道,就是IE EVENT.所以我們就被逼著編寫跨瀏覽器的事件處理函數和事件對象。是不是覺得IE很作死,可是沒辦法啊,人家還占那么多份額,我們只能乖乖地兼容它。

  • 怎么編寫跨瀏覽器的事件處理函數和事件對象

    • 事件處理函數

      • DOM 事件處理函數(這里只談及DOM2)
        DOM2級事件定義了兩種方法:

        • addEventListener(type,handler,useCapture); --添加事件處理函數
        • removeEventListener(type,handler,useCapture); -- 刪除事件處理函數

        這兩個方法都有三個參數:type為事件的類型,handler為事件處理程序的函數,userCapture(可選參數)為事件流類型,冒泡為false,捕獲為true,默認為false。

      • IE 事件處理函數
        IE 事件定義了兩種方法:

        • 添加事件處理函數 :attachEvent("on"+type,handler);
        • 刪除事件處理函數 :detachEvent("on"+type,handler);

        這兩個方法都有兩個參數:第一個參數"on"+type是事件類型(如果在addEventListener的type為click,則此處應該為onclick),第二個參數handler為事件處理程序的函數。注意:IE8之前的瀏覽器只支持事件冒泡,不支持事件捕獲。

      • 區別:

        • 一個元素添加多個事件處理函數的執行順序:如果為一個元素添加多個事件處理函數,addEVentListener會按照添加的順序執行代碼。而attachEvent會按照添加順序的相反順序,如下:

          var btn = document.getElementById('btn');
          btn.addEventListener("click",function(){
          alert("1");
          });
          btn.addEventListener("click",function(){
          alert("2");
          });
          //以上代碼先輸出1,在輸出2   
          
          var btn = document.getElementById('btn');
          btn.attachEvent("onclick"function(){
              alert("1");
          })
          btn.attachEvent("onclick"function(){
           alert("2");
          })
          //以上代碼先輸出2,再輸出1
          
        • 事件處理程序的函數中的this:addEventListener的handler中的this指的是添加該事件處理函數的元素,而attachEvent的handler中的this指的是window對象。如下:

          var bt = document.getElementById("bt");
          bt.addEventListener("click",function(){
            console.log(this === bt)
          })    
          //為true,this指添加事件處理函數的元素
          var bt = document.getElementById("bt");
          bt.attachEvent("onclick",function(){
            console.log(this === window)
          })
          //為true,this指window對象
          
      • 跨瀏覽器事件處理函數 :
        綜合上述的事件處理函數,我們可以寫出跨瀏覽器的事件處理函數,我們可以把它包含在一個對象里。如下:

        var eventForAllBrowsers = {
              addHandler :function(elem,type,handler){
                  if(elem.addEventListener){
                      elem.addEventListener(type,handler,false);
                  } else if(elem.attachEvent){
                      elem.attachEvent("on"+type,handler);
                  } else{
                      elem["on"+type] = handler;
                      //加上對DOM 0 的支持
                  }
              },
              removeHandler:function(elem,type,handler){
                  if(elem.removeEventListener){
                      elem.removeEventListener(type,handler,false)
                  } else if(elem.detachEvent){
                      elem.detachEvent("on"+type,handler);
                  } else {
                      elem["on"+type] = null;
                  }
              }
        }
        
    • 事件對象
    在觸發dom上的某個事件時,會產生一個事件對象event,這個對象中包含著所有與事件有關的信息,該對象在事件觸發完畢后會被自動銷毀。
    • DOM 事件對象
      兼容DOM的瀏覽器會將一個event對象傳入到事件處理程序中。在函數內可以訪問event對象的屬性與方法。如下:
      var bt = document.getElementById('bt');
      bt.addEventListener("click",function(event){
      alert(event.type)
      })
      event對象中包含的常用的屬性和方法:
      • preventDefault() --- 取消事件的默認行為。
      • stopPropagation() --- 取消事件的進一步冒泡。
      • type --- 被觸發的事件的類型。
      • target --- 事件的目標。
    • IE 事件對象
      與兼容DOM的瀏覽器不同,要訪問IE的event對象分兩種情況:
      • 使用DOM 0 方法添加事件處理程序:在程序的函數內,用window.event去訪問event對象。如下:

        var bt = document.getElementById('bt');
        bt.onclick = function(){
          var event = window.event;
          alert(event.type)
        }
        
      • 使用IE 事件處理函數:將event對象傳入事件處理程序的函數內。如下:

        var bt = document.getElementById('bt');
        bt.attachEvent("onclick",function(event){
          alert(event.type)
        })
        

event對象中包含的常用的屬性和方法:
- cacelBubble --- 默認值為false,將其設置為true就可以取消事件冒泡。
- returnValue --- 默認值為true,將其設置為false則可以取消事件的默認行為。
- srcElement --- 事件的目標。
- type --- 被觸發事件的類型。

- 跨瀏覽器的事件對象
  通過檢查兼容DOM的瀏覽器的event對象的方法和屬性是否存在的方法去處理跨瀏覽器兼容性問題。把通用的事件對象的相關函數添加到eventForAllBrowsers上,代碼大致如下:
      var eventForAllBrowsers = {
      //省略上述的代碼
         getEvent :function(event){
                 return event ?event: window.event;
            },
            getTarget:function(event){
                return event.target || event.srcElement;
            },
            preventDefault:function(event){
                if(event.preventDefault){
                    event.preventDefault();
                }else{
                    event.returnValue = false;
                }
            },
            stopPropagation:function(event){
                if(event.stopPropagation){
                    event.stopPropagation();
                }else{
                    event.cancelBubble = true;
                }
            }         
      }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,461評論 6 532
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,538評論 3 417
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,423評論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,991評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,761評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,207評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,268評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,419評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,959評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,782評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,983評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,222評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,653評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,901評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,678評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,978評論 2 374

推薦閱讀更多精彩內容

  • 以下文章為轉載,對理解JavaScript中的事件處理機制很有幫助,淺顯易懂,特分享于此。 什么是事件? 事件(E...
    jxyjxy閱讀 3,057評論 1 10
  • JavaScript 程序采用了異步事件驅動編程模型。在這種程序設計風格下,當文檔、瀏覽器、元素或與之相關的對象發...
    劼哥stone閱讀 1,271評論 3 11
  • 什么是事件: 事件是交互體驗的核心功能 一.事件冒泡: 當一個事件發生時,這個事件會從內向外逐層傳遞。 二.為什么...
    輕描淡寫mua閱讀 522評論 0 0
  • title: 實現通用的事件注冊方法date: 2017-07-15tags: [原生js]categories:...
    始悔不悟閱讀 1,062評論 0 0
  • 事件流: 事件流:頁面接收事件的順序。 IE定義的:事件冒泡流(由最具體的元素依次傳播到DOM樹的最上層的Docu...
    xiaoguo16閱讀 600評論 0 0