查看完整版本: 关于DOM事件模型的两件事

snowfox373 2007-11-26 09:52

关于DOM事件模型的两件事

[b]DOM事件模型的两件事:事件捕捉(Event Capture)的实现问题以及IE的高级事件处理模型的问题。[/b]
JGs2y/pi9Q   [b]一、事件捕捉(Event Capture)的实现问题[/b]
F"?b*S*e,H/V   3C DOM Level2的事件模型规范中,事件在DOM树中的传播过程(从根节点到目标节点)被分为了两个阶段:捕捉(Capture)和冒泡(Bubbling)。下面这个图能大概的说明整个过程:wt:N ^7v |d![;W
  w7?o AN"d PZ%x
[img]http://homepage.yesky.com/imagelist/2007/329/wvr66gst3g95.png[/img] V)H_l-_ FO
  (from W3C)\WXY1RG lQ1s2s%r
  如果想创建一个捕捉事件,在支持W3C 事件模型的浏览器中,将addEventListener的第三个参数设为true就好了。例如:
m} ~'l1Z8i}   document.getElementById('foo').addEventListener('click',function(){alert('Hello, world!');},true);Bw9j$l[7Q
  在Firefox 2、Safari 3 on Windows和Opera 9上实践了事件捕捉(当然,因为IE不支持事件捕捉,所以…),实验的原理见下图:9C]8sx]O
  $y9B3q0c-jp3r%dG+eJ
[img]http://homepage.yesky.com/imagelist/2007/329/4sx3x902mvlj.png[/img]
FJ5Y+X?L,T7j)w@   ID为div1和div2的两个元素都被委派了捕捉阶段的事件处理函数,这样:
MEC6H)GZM)x4k   当点击#div1(蓝色区域)时,应该会alert出”div1″r}!^z'a0w2Ig
  当点击#div2(黄色区域)时,应该会先alert出”div1″,再alert出”div2″,因为在事件捕捉阶段,事件是从根元素向下传播的,#div1是#div2的父元素,自然绑定在#div1上的click事件也会先于#div2上的click事件被执行。
GS"LpCn:U h   然而,以上的设想只试用于Firefox 2和Safari 3 on Windows,在Opera 9中,事情会变成这样:
|1|.R^ve1qv Y   当点击#div1(蓝色区域)时,什么都不会发生
"ARtT)["[   当点击#div2(黄色区域)时,会alert出”div1″,随后什么都不会再发生*wW[.kd)iD6B IA
  可以看出,在Opera 9中,目标元素(TargetElement)的click事件没有被执行。DOM规范中陈述了捕捉型的事件不应该在目标元素上被执行,因为捕捉型事件的用意就是为了监测到达目标元素之前的事件。Firefox和Safari的实现都是带有bug的。
Som[mMr   再来看看W3C的DOM Events规范中的原话:C1R?r(g^!?yr7U*tO
  A capturing EventListener will not be triggered by events dispatched directly to the EventTarget upon which it is registered.cRJ-h7G;x%LK
  所以,在整个事件传播中,被执行的顺序是:L!L\s&ci
  父元素中所有的捕捉型事件(如果有)自上而下地执行
P T`9^oO J8fP   目标元素的冒泡型事件(如果有)
$jAWH;N(Y!MX,_   父元素中所有的冒泡型事件(如果有)自下而上地执行,n.h4MX8o#H5hU(B
  在了解了这些后,也许还是不要使用事件捕捉为妙,至少暂时不要。-P:\j1mVv5I
  [b]二、IE的高级事件处理模型的问题[/b]BZQJ6W9x
  [b]重复绑定[/b]
7C]K7Ly"U&_+t   IE下没有addEventListener,但是也有自己的attachEvent,即所谓的Microsoft Model。二者的实现基本相同只是attachEvent的第一个参数(事件类型)需要加”on”,而addEventListener不用,另外attachEvent因为不支持事件捕捉,所以也没有第三个参数。
@5s eW8zRT7h   然而,attachEvent还有一个很要命的问题:重复绑定事件。
9m9H] zm   一个例子:[code]function sayHello(){ sae,` @&}m
alert('Hello, world!'); {S ?~ IB'g
} 3`icow0lgm1BcL
// W3C Model bp*n S$EE)gI
$('div1').addEventListener('click', sayHello, false);
|0z(hHv1q%w4_| $('div1').addEventListener('click', sayHello, false); #}PE]W4f
// Microsoft Model
vo)UmN-? $('div1').attachEvent('onclick', sayHello);
c;a{'H.Qc#b)t $('div1').attachEvent('onclick', sayHello); [/code]  在W3C模型中,相同事件处理函数的绑定会被忽略,也就是说第二个$('div1').addEventListener('click', sayHello, false);会被忽略。
YU&YXT*c])C$H   而在Microsoft模型中,第二个$('div1').attachEvent('onclick', sayHello);同样会被执行,所以,当你点击#div1的时候,alert框会弹出来两次。更有甚者,在detachEvent时候,也同样要detachEvent两次才能彻底把sayHello从#div1的click事件中删除。C"`:x#XG@k)U"e R ^
  为什么不继续使用alertID()了?
?3T3H [H&` g!V   这是因为IE的事件模型的另一个缺陷,在alertID中,使用了this关键字来指代被委派了该事件处理函数的元素,这样,在W3C模型中,alertID中的this指代了#div1或者#div2。
1U,T#B'^+H,`2or   但是在Microsoft模型中,缺少了对this的支持后,this.id就会变为undefined,因为这时候this指代了window对象。
页: [1]
查看完整版本: 关于DOM事件模型的两件事
查看完整版本: 关于DOM事件模型的两件事