DOM 中的事件处理介绍


Posted in Javascript onJanuary 18, 2012

该接口提供了 'addEventListener' 和 'removeEventListener' 方法,用来绑定或解绑一个 EventListeners 接口到一个 EventTarget。

DOM 2 Events 中定义了 Event 接口,用来提供事件的上下文信息,它提供了若干标准属性和方法。 实现 Event 接口的对象一般作为第一个参数传入事件处理函数,以用来提供当前事件相关的一些信息。

事件注册
根据 DOM 2 Events 中描述,节点使用 'addEventListener' 和 'removeEventListener' 方法绑定和解绑事件监听器,但 IE6 IE7 IE8 不支持这两个方法, 而使用 'attachEvent' 和 'detachEvent' 方法作为替代方案,Opera 两类方法都支持。Chrome Safari Firefox 只支持标准方法。

为了解决浏览器兼容问题,可以自定义函数来解决。例如:

var EventUtil = { 
addHandler: function (element, type, handler) { 
if (element.addEventListener) { 
element.addEventListener(type, handler, false); 
} else if (element.attachEvent) { 
element.attachEvent("on" + type, handler); 
} else { 
element["on" + type] = handler; 
} 
}, 
removeHandler: function (element, type, handler) { 
if (element.removeEventListener) { 
element.removeEventListener(type, handler, false); 
} else if (element.detachEvent) { 
element.detachEvent("on" + type, handler); 
} else { 
element["on" + type] = null; 
} 
} 
};

关于 'addEventListener' 和 'attachEvent' 有几点需要注意:

IE 不支持在捕获阶段触发事件监听器,'attachEvent' 方法没有提供参数说明是否响应在捕获阶段触发的事件;
'addEventListener' 和 'attachEvent' 都可以注册多个事件监听器;
在 Firefox Chrome Safari Opera 中给同一事件注册同一个事件监听器多次,重复注册的会被丢弃;而在 IE 中重复注册的事件监听器会被重复执行多次;
当给同一元素注册了多个事件监听器的时候,IE6 IE7 的事件监听器执行顺序是随机的,IE8 是倒序的,Firefox Chrome Safari Opera 是顺序的;
当元素注册的事件监听器中有非法的事件监听器时(非函数),在 IE Firefox 中会抛出异常,而在 Chrome Safari Opera 中则会忽略非法的事件监听器,继续执行其他的事件监听器。
事件对象
在ie中,事件对象是作为一个全局变量来保存和维护的。 所有的浏览器事件,不管是用户触发的,还是其他事件, 都会更新window.event 对象。 所以在代码中,只要轻松调用 window.event 就可以轻松获取 事件对象, 再 event.srcElement 就可以取得触发事件的元素进行进一步处理。

对于标准的 DOM 处理来说, 事件对象却不是全局对象,一般情况下,是现场发生,现场使用,把事件对象自动传递给对应的事件处理函数。 在代码中,函数的第一个参数就是事件对象了。

为了解决兼容性问题,通常在代码中如下处理:

function handler(e){ 
e = e || window.event; 
}

需要注意的是,使用 <button id="btn" onclick="foo()">按钮1</button> 进行事件注册,标准方式下却不能在事件处理方法中取得事件对象。

原因是 onclick="foo()" 就是直接执行了, foo() 函数,没有任何参数传递给 foo 函数。

有两个办法解决这个问题。

第一,将注册的方法修改为 <button id="btn" onclick="foo(event)">按钮</button>,注意,这里的 event 不是形参,而是实参,必须名为 event。这样 foo 函数就可以得到事件参数了。

第二,不修改注册的代码,在事件处理方法上进行处理。关键在于此时实际上存在事件对象,只不过没有传递给 foo 函数罢了,我们可以找到调用 foo 函数的那个函数,当然这是一个系统函数,没有关系,通过 foo.caller 可以取得当前调用 foo 函数的函数,这个函数的第一个参数就是事件对象,所以,我们可以这样取得这个事件对象了。foo.caller.arguments[0]。

注意:

只有在使用 attachEvent 方法注册事件监听器的时候,IE 才支持使用事件监听器传入的第一个参数作为事件对象的方式;
Chrome Safari Opera 两种获取事件对象的方式都支持;
Firefox 只支持获取事件对象的标准方式。
事件对象的属性
IE 对事件对象的标准属性和方法支持有限,针对大部分属性和方法,IE 都提供了一套替代非标准的替代方案; 而 Firefox Chrome Safari Opera 除了全面支持事件对象的标准属性和方法外,还在不同程度上支持了 IE 提供的非标准替代方案。

使用特性判断使用与标准对应的非标准方法及属性

target

srcElement

preventDefault()

 returnValue

stopPropagation()

cancelBubble

relatedTarget

fromElement toElement

例如:

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; 
} 
}

参考资料:

SD9011: 事件模型在各浏览器中存在差异

Javascript 相关文章推荐
Javascript MD4
Dec 20 Javascript
JavaScript的parseInt 取整使用
May 09 Javascript
javascript学习笔记(十四) window对象使用介绍
Jun 20 Javascript
js操作iframe的一些方法介绍
Jun 25 Javascript
js实现单行文本向上滚动效果实例代码
Nov 28 Javascript
javascript生成随机颜色示例代码
May 05 Javascript
JQuery中DOM事件合成用法实例分析
Jun 13 Javascript
javascript入门教程基础篇
Nov 16 Javascript
JavaScript实现99乘法表及隔行变色实例代码
Feb 24 Javascript
Three.js学习之正交投影照相机
Aug 01 Javascript
详解基于webpack2.x的vue2.x的多页面站点
Aug 21 Javascript
使用layui 渲染table数据表格的实例代码
Aug 19 Javascript
深入理解JavaScript系列(13) This? Yes,this!
Jan 18 #Javascript
JavaScript中常用的运算符小结
Jan 18 #Javascript
jQuery LigerUI 使用教程表格篇(1)
Jan 18 #Javascript
jQuery LigerUI 使用教程入门篇
Jan 18 #Javascript
c#和Javascript操作同一json对象的实现代码
Jan 17 #Javascript
深入理解JavaScript系列(12) 变量对象(Variable Object)
Jan 16 #Javascript
Prototype源码浅析 Enumerable部分之each方法
Jan 16 #Javascript
You might like
PHP iconv 解决utf-8和gb2312编码转换问题
2010/04/12 PHP
关于使用key/value数据库redis和TTSERVER的心得体会
2013/06/28 PHP
php file_get_contents抓取Gzip网页乱码的三种解决方法
2013/11/12 PHP
Laravel4中的Validator验证扩展用法详解
2016/07/26 PHP
phpcms实现验证码替换及phpcms实现全站搜索功能教程详解
2017/12/13 PHP
5个javascript的数字格式化函数分享
2011/12/07 Javascript
Javascript实现网络监测的方法
2015/07/31 Javascript
JavaScript如何禁止Backspace键
2015/12/02 Javascript
js实现新年倒计时效果
2015/12/10 Javascript
基于jQuery实现音乐播放试听列表
2016/04/14 Javascript
Node.js中npm常用命令大全
2016/06/09 Javascript
用jQuery实现可输入多选下拉组合框实例代码
2017/01/18 Javascript
js 递归和定时器的实例解析
2017/02/03 Javascript
angularjs2中父子组件的数据传递的实例代码
2017/07/05 Javascript
laydate 显示结束时间不小于开始时间的实例
2017/08/11 Javascript
详解vue项目接入微信JSSDK的坑
2018/12/14 Javascript
vue 左滑删除功能的示例代码
2019/01/28 Javascript
JavaScript中Dom操作实例详解
2019/07/08 Javascript
vue.js 2.0实现简单分页效果
2019/07/29 Javascript
Vue-cli3项目引入Typescript的实现方法
2019/10/18 Javascript
vue quill editor 使用富文本添加上传音频功能
2020/01/14 Javascript
vue中echarts引入中国地图的案例
2020/07/28 Javascript
Python常用模块用法分析
2014/09/08 Python
Python实现将数据库一键导出为Excel表格的实例
2016/12/30 Python
Python实现PS滤镜功能之波浪特效示例
2018/01/26 Python
用Python去除图像的黑色或白色背景实例
2019/12/12 Python
为什么称python为胶水语言
2020/06/16 Python
html通过canvas转成base64的方法
2019/07/18 HTML / CSS
Clarks英国官方网站:全球领军鞋履品牌
2016/11/26 全球购物
2014年感恩母亲演讲稿
2014/05/27 职场文书
购房协议书范本(无房产证)
2014/10/07 职场文书
乡镇党的群众路线教育实践活动剖析材料
2014/10/09 职场文书
刘公岛导游词
2015/02/05 职场文书
2015年酒店工作总结范文
2015/04/07 职场文书
观看安全警示教育片心得体会
2016/01/15 职场文书
Python下opencv使用hough变换检测直线与圆
2021/06/18 Python