您好,欢迎来到三六零分类信息网!老站,搜索引擎当天收录,欢迎发信息

事件模型在各浏览器中存在差异_javascript技巧

2024/2/20 17:55:25发布31次查看
标准参考根据 w3c dom 2 events 描述,eventtarget 接口被所有支持 dom 事件模型的节点(node)实现。 该接口提供了 'addeventlistener' 和 'removeeventlistener' 方法,用来绑定或解绑一个 eventlisteners 接口到一个 eventtarget。
dom 2 events 中定义了 event 接口,用来提供事件的上下文信息,它提供了若干标准属性和方法。 实现 event 接口的对象一般作为第一个参数传入事件处理函数,以用来提供当前事件相关的一些信息。
dom 2 事件模型允许 dom 实现支持事件的多模块,如果有必要,事件模型允许附加新的事件模块。 为了共同使用性的目的,dom 会定义一个包含低级别的,依赖设备的事件模块、一个 ui 逻辑事件模块和一个文档变化事件模块的 ui 事件模块。 第三方实现(比如浏览器厂商)在定义一个新类型事件的时候,事件名称一定不能使用大小写无关的 'dom' 字符串作开头,该前缀是为未来的 dom 事件模块保留的。
dom 2 已经定义了一个 ui 事件类型模块和一个鼠标事件类型的模块,它们分别对应 uievent 和 mouseevent 接口。 这两个接口提供了若干标准属性和方法,以获知事件发生时的一些信息。
关于 eventtarget 接口的详细信息,请参考 dom 2 events 1.3. event listener registration。
关于 event 接口的详细信息,请参考 dom 2 events 1.4. event interface。
关于事件模块的详细信息,请参考 dom 2 events 1.6. event module definitions。
问题描述各浏览器对元素绑定、解绑事件监听器的方法,事件对象的获取,以及 event 对象的实现上存在差异。
造成的影响如果使用某浏览器特有的事件相关的属性及方法编写代码,则可能造成兼容性问题,导致代码报错,功能失效。
受影响的浏览器所有浏览器
问题分析1. 只有 ie opera 支持使用 'attachevent' 和 'detachevent' 方法绑定和解绑事件监听器根据 dom 2 events 中描述,节点使用 'addeventlistener' 和 'removeeventlistener' 方法绑定和解绑事件监听器,但 ie6 ie7 ie8 不支持这两个方法, 而使用 'attachevent' 和 'detachevent' 方法作为替代方案,opera 两类方法都支持。chrome safari firefox 只支持标准方法。
分析以下代码:
add event listener testremove event listener test

依次点击 'add event listener test' >> 'remove event listener test' >> 'add event listener test',测试各浏览器对这些方法的支持,结果如下:
ie6 ie7 ie8
opera
chrome safari firefox
关于 'addeventlistener' 和 'attachevent' 有几点需要注意:
ie 不支持在捕获阶段触发事件监听器,'attachevent' 方法没有提供参数说明是否响应在捕获阶段触发的事件; 'addeventlistener' 和 'attachevent' 都可以注册多个事件监听器; 在 firefox chrome safari opera 中给同一事件注册同一个事件监听器多次,重复注册的会被丢弃;而在 ie 中重复注册的事件监听器会被重复执行多次; 当给同一元素注册了多个事件监听器的时候,ie6 ie7 的事件监听器执行顺序是随机的,ie8 是倒序的,firefox chrome safari opera 是顺序的; 当元素注册的事件监听器中有非法的事件监听器时(非函数),在 ie firefox 中会抛出异常,而在 chrome safari opera 中则会忽略非法的事件监听器,继续执行其他的事件监听器。 2. 各浏览器获取事件对象的差异dom 2 events 中规定使用事件监听器的第一个参数作为事件对象,而 ie opera chrome safari 还支持通过 window.event 获取事件对象。
分析以下代码:
attacheventaddeventlisteneronclick
info :

以上代码组合不同的事件监听器绑定方式和事件对象获取方法,测试各浏览器的支持程度。
依次点击 'attachevent' >> 'addeventlistener' >> 'onclick',结果如下:
ie6 ie7 ie8
chrome safari
opera
firefox
汇总测试结果如下表:1
事件对象获取方式ie6 ie7 ie8chrome safarioperafirefox
window.eventy y y n
arguments[0]y2 y y y
注1: y 表示支持该事件对象获取方式,n 表示不支持。
注2: 部分支持。
从上表出可以看出:
只有在使用 attachevent 方法注册事件监听器的时候,ie 才支持使用事件监听器传入的第一个参数作为事件对象的方式; chrome safari opera 两种获取事件对象的方式都支持; firefox 只支持获取事件对象的标准方式。 3. 各浏览器中事件对象的属性和方法的差异ie 对事件对象的标准属性和方法支持有限,针对大部分属性和方法,ie 都提供了一套替代非标准的替代方案; 而 firefox chrome safari opera 除了全面支持事件对象的标准属性和方法外,还在不同程度上支持了 ie 提供的非标准替代方案。
以下代码检测了 event、uievent、mouseevent 接口以及事件对象的非标准属性在各浏览器下的支持程度:
interface eventinterface uievent & mouseevent

interface event
interface uievent
&
mouseevent
non-standard
&
click
non-standard
&
mouseover mouseout
non-standard
&
keycode
执行以上测试代码,分别点击 'interface event' 按钮、'interface uievent & mouseevent' 按钮和图片,鼠标移到图片上再移开,在文本框中输入 'a',得到结果整理如下表:3
interface & non-standardproperty & methodie6 ie7 ie8chrome safari operafirefox
interface eventtypey y y
targetn y y
currenttargetn y y
eventphasen y y
bubblesn y y
cancelablen y y
timestampn y y
initeventn y y
preventdefaultn y y
stoppropagationn y y
interface uieventviewn y y
detailn y y
inituieventn y y
interface mouseeventscreenxy y y
screenyy y y
clientxy y y
clientyy y y
ctrlkeyy y y
shiftkeyy y y
altkeyy y y
metakeyn y y
buttony y y
relatedtargetn y y
initmouseeventn y y
non-standardcancelbubbley y y
offsetxy y n
offsetyy y n
returnvaluey y n
srcelementy y n
xy y n
yy y n
fromelementy y n
toelementy y n
keycodey y y
注3: y 代表事件对象支持该属性或方法,n 代表不支持。
从上表中可以看出:
ie 支持事件对象的所有非标准属性,不支持除 'type' 外 event 接口的所有方法属性及方法,不支持 uievent 接口的所有属性和方法,不支持 mouseevent 接口的 'metakey'、'relatedtarget' 属性和 'initmouseevent' 方法; chrome safari opera 支持事件对象的所有标准及非标准的属性和方法; firefox 支持事件对象的所有标准属性和方法,但仅支持非标准属性中的 'cancelbubble' 和 'keycode'。 需要注意的是:
firefox 不支持事件对象的 'returnvalue' 属性,测试样例中虽然显示 'returnvalue' 值为 false,但这仅仅是因为给事件对象添加了 'returnvalue' 属性,并没有起到取消事件默认动作的作用,这从地址栏可以看出,多了 '#' 号,这是 a 标签的 'href' 属性造成的。 各浏览器对 event 接口的 'timestamp' 属性返回值都不同。 关于 ie 实现的事件对象非标准的属性及方法的详细信息,请参考 msdn event object。
关于 firefox 对事件对象实现的详细信息,请参考 mdc event。
解决方案1. 使用特性判断创建无兼容性问题的事件监听器绑定和解绑函数如:
function addevent(elem, type, handler, usecapture){ elem.addeventlistener ? elem.addeventlistener(type, handler, usecapture) : elem.attachevent(on + type, handler);}function removeevent(elem, type, handler, usecapture){ elem.removeeventlistener ? elem.removeeventlistener(type, handler, usecapture) : elem.detachevent(on + type, handler);}
2. 使用特性判断获得有效的事件对象在事件监听器中判断传入的第一个参数或 window.event 是否有效,如:
function handler(e){ e = e || window.event;}
3. 使用特性判断使用与标准对应的非标准方法及属性尽管 ie 对事件对象的标准属性和方法支持有限,但它自己实现的事件模型也基本能够替代或实现标准属性或方法的功能。
下表总结了部分与标准事件对象对应的,或可以实现标准属性或方法功能的非标准属性:
standardnon-standard
target srcelement
preventdefault() returnvalue
stoppropagation() cancelbubble
relatedtarget fromelement toelement
另,标准的 'clientx' 属性和非标准的 'x' 属性作用是相同的,同样 'clienty' 和 'y' 也是。
该用户其它信息

VIP推荐

免费发布信息,免费发布B2B信息网站平台 - 三六零分类信息网 沪ICP备09012988号-2
企业名录 Product