javascript事件机制与jQuery.bind的补充说明
2010-05-10 17:36 BAsil 阅读(2594) 评论(1) 编辑 收藏 举报在之前的文章javascript 事件机制 与 jQuery.Bind中,为了说明冒泡阶段中Event Handler Function的表现,我使用了event.data来记录触发function的次数。并且提出了一个问题,就是在jQuery.bind方式中,event.data无法正确记录触发的次数。后来经过测试和查阅网上的相关的资料,得出了一个结论,就是我之前关于event.data的使用方式是错误的,或者说对于跨浏览器的支持是困难的。同时我也意识到,由于event.data在w3c dom level 2文档中,并不是作为event的标准属性出现的,所以jQuery对event进行了fix,使其能够兼容各个浏览器。
在我纠正误用event.data的方式之前,再描述一下我对event的理解。在我查看jQuery(1.3.2)源代码的时候,jQuery.event的add方法中有如下代码
// Init the element's event structure var events = jQuery.data(elem, "events") || jQuery.data(elem, "events", {}), handle = jQuery.data(elem, "handle") || jQuery.data(elem, "handle", function() { // Handle the second event of a trigger and when // an event is called after a page has unloaded return typeof jQuery !== "undefined" && !jQuery.event.triggered ? jQuery.event.handle.apply(arguments.callee.elem, arguments) : undefined; });
其中的jQuery.event.handle中进行event的fix
event = arguments[0] = jQuery.event.fix(event || window.event);
这里按照网上的资料大多是说ie下使用window.event,而firefox下使用arguments[0]也就是传递过来的函数参数event。可我在测试中发现ie6,ie7(未测试),ie8在fix之前event并不为空,也就是说在fix的时候并没有使用window.event。
看一下这段代码
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml" lang="zh-CN"> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <title></title> <style> #panelGrandPa,#panelPaPa,#panelSon { border:1px solid #320213;} </style> </head> <body> <div id="panelGrandPa" style="width:300px;height:200px;" > <div id="panelPaPa" style="width:200px;height:100px;" > <div id="panelSon" style="width:100px;height:50px;" > </div> </div> </div> <script> function click() { alert(event.srcElement.id); event.data = event.data || 1; alert("click function has fired " + event.data + " times"); event.data = parseInt(event.data) + 1; } function clickSon() { alert("I am son"); click(); } function clickGrandPa() { alert("I am GrandPa"); click(); } document.getElementById("panelGrandPa").onclick = clickGrandPa; document.getElementById("panelSon").onclick = clickSon; </script> </body> </html>
只能在ie8下正常工作,在ie6和ie7下都报event.data undefined错误。当然我们这里使用的是window.event也就是页面维护的event相当于全局变量,那我们再试一下事件方法的event参数(之前阅读jQuery源代码提到的ie中除却window.event另外的event)
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml" lang="zh-CN"> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <title></title> <style> #panelGrandPa,#panelPaPa,#panelSon { border:1px solid #320213;} </style> </head> <body> <div id="panelGrandPa" style="width:300px;height:200px;" > <div id="panelPaPa" style="width:200px;height:100px;" > <div id="panelSon" style="width:100px;height:50px;" > </div> </div> </div> <script> function click(e) { alert(event.srcElement.id); e.data = e.data || 1; alert("click function has fired " + e.data + " times"); e.data = parseInt(e.data) + 1; } function clickSon() { alert("I am son"); click(arguments[0]); } function clickGrandPa() { alert("I am GrandPa"); click(arguments[0]); } document.getElementById("panelGrandPa").attachEvent("onclick", clickGrandPa); document.getElementById("panelSon").attachEvent("onclick", clickSon); </script> </body> </html>
注意必须用attachEvent我们才能得到区别于window.event的“另一个”event。以上代码在ie8下正常工作,在ie6和ie7下event.data始终为1
对于window.event,在ie6和ie7下,通过调试工具发现其中并没有event.data的属性。而attachEvent的得到的event.data 在ie6和ie7下不能正确计数,始终为1,但ie8下正确。
鉴于event在各浏览器下的差异以及我们对书写跨浏览器脚本的良好愿望,我个人认为不应该在多个事件方法中传递event.data。
当然jQuery在这方面做得更好经过fix后的event 使得event.data在各种浏览器下表现一致
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml" lang="zh-CN"> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <script type="text/javascript" src="http://jqueryjs.googlecode.com/files/jquery-1.3.2.js"></script> <title></title> <style> #panelGrandPa,#panelPaPa,#panelSon { border:1px solid #320213;} </style> <head> <body> <div id="panelGrandPa" style="width:300px;height:200px;" > <div id="panelPaPa" style="width:200px;height:100px;" > <div id="panelSon" style="width:100px;height:50px;" > </div> </div> </div> <script> function click(e) { alert(e.target.id); e.data = e.data || 1; alert("click function has fired " + e.data + " times"); e.data = parseInt(e.data) + 1; } function clickSon(e) { alert("I am son"); click(e); } function clickGrandPa(e) { alert("I am GrandPa"); click(e); } $("#panelGrandPa").bind("click", clickGrandPa); $("#panelSon").bind("click", clickSon); </script> </body> </html>
上述代码在任何浏览器下都不能正常计数,显示event.data为1(这下ie8也不行了)。
我们分析一下原因,这也是我对上一篇提出问题的一个回答
jQuery.event.add方法封装了attachEvent/addEventListener并且给每一个handler方法附加了data
add:function(elem, types, handler, data) { //省略部分代码 // if data is passed, bind to handler if (data !== undefined) { // Create temporary function pointer to original handler var fn = handler; // Create unique handler function, wrapped around original handler handler = this.proxy(fn); // Store data in unique handler handler.data = data; } //省略部分代码 }
在jQuery.event的fix方法中,我们看到其将传入的event复制了一份(包括data),注意不是引用(这意味着每次方法的event都是不同的)
fix: function(event) { if (event[expando]) return event; // store a copy of the original event object // and "clone" to set read-only properties var originalEvent = event; event = jQuery.Event(originalEvent); for (var i = this.props.length, prop; i; ) { prop = this.props[--i]; event[prop] = originalEvent[prop]; } //以下省略 }那么这样一来,我们肯定不能在多个事件方法中用上述方法传递data了。