IE和标准DOM事件模型的兼容写法

一、为什么要这么做?

在开发中我们不可避免需要使用到事件模型来进行交互,而我们面对的客户所使用浏览器是各种各样的,所以我们需要来进行兼容,我们主要面对的是两种事件模型:标准DOM2事件模型(现代标准浏览器)和IE事件模型(老版IE浏览器IE6~IE8),他们都给了我们绑定事件处理程序的方法;

  • DOM2事件模型:使用addEvenListener(),它可以接受三个参数1.事件类型2.事件处理方法3.布尔参数,如果是true表示在捕获阶段调用事件处理程序,如果是false,则是在事件冒泡阶段处理;
  • IE事件模型:使用attachEvent(),它可以接受两个参数1.事件处理函数名称,2.事件处理方法,
    另外在触发某个事件的时候都产生一个事件对象event,这个对象会包含着与事件有关的信息,包括产生事件的元素、事件类型等相关信息。但是 DOM2事件模型和IE事件模型的事件对象的属性是不同的,如下:
  • DOM2事件模型事件模型的event对象:
属性/方法 类型 Cool 说明
bubbles Boolean 只读 事件是否冒泡
cancelable Boolean 只读 是否可以取消事件的默认行为
currentTarget Element 只读 事件处理程序当前处理元素
detail Integer 只读 与事件相关细节信息
eventPhase Integer 只读 事件处理程序阶段:1 捕获阶段,2 处于目标阶段,3 冒泡阶段
preventDefault() Function 只读 取消事件默认行为
stopPropagation() Function 只读 取消事件进一步捕获或冒泡
target Element 只读 事件的目标元素
type String 只读 被触发的事件类型
view AbstractView 只读 与事件关联的抽象视图,等同于发生事件的window对象
  • IE事件模型的even对象:
属性/方法 类型 Cool 说明
cancelBubble Boolean 只读 默认为false,设置为true后可以取消事件冒泡
returnValue Boolean 只读 默认为true,设为false可以取消事件默认行为
srcElement Element 只读 事件的目标元素
type String 只读 被触发的事件类型

二、有哪些区别问题需要兼容?

目前来看主要来看需要解决五个问题:

  • attachEvent()和addEvenListener()参数内容和不相同;
  • 事件处理程序的作用域不相同,addEventListener的作用域是元素本身,this是指的触发元素,而attachEvent事件处理程序会在全局变量内运行,this是window。
  • 为一个事件添加多个事件处理程序时,执行顺序不同,addEventListener添加会按照添加顺序执行,而attachEvent添加多个事件处理程序时顺序无规律(添加的方法少的时候大多是按添加顺序的反顺序执行的,但是添加的多了就无规律了),所以添加多个的时候,不依赖执行顺序的还好。

三、如何做?

主要的区别找到了,该如何解决这些问题呢?这里我可以借鉴下jQuery创始人John Resig的做法代码如下:

function addEvent(node, type, handler) {     //node参数为需要被绑定事件的元素;
                                             //type是事件类型,handler是事件处理程序;
    if (!node) return false;          //如果没有这个元素 直接返回false
    if (node.addEventListener) {      //兼容标准DOM事件模型
        node.addEventListener(type, handler, false);
        return true;
    }
    else if (node.attachEvent) {      //兼容IE事件模型
        node['e' + type + handler] = handler;    
        node[type + handler] = function() {
            node['e' + type + handler](window.event); //
        };
        node.attachEvent('on' + type, node[type + handler]);
        return true;
    }
    return false;
}

另外在取消事件处理程序的时候,John Resig也给了我们一个很好的方法:

function removeEvent(node, type, handler) {
    if (!node) return false;
    if (node.removeEventListener) {
        node.removeEventListener(type, handler, false);
        return true;
    }
    else if (node.detachEvent) {
        node.detachEvent('on' + type, node[type + handler]);
        node[type + handler] = null;
    }
    return false;
}

你可能感兴趣的:(IE和标准DOM事件模型的兼容写法)