JavaScript事件冒泡应用实例

在一些传统的小型WEB应用开发过程中,JavaScript通常只是拿来做表单验证而以,所以你很少会遇到因为JavaScript事件冒泡而影响功能的实现情况,又或者事件冒泡对最终实现效果影响不大,可忽略。然而,在今天一些大型的WEB交互项目中,比如大型的WebGame项目,JavaScript事件冒泡影响是值得重视的。本文通过一个简单的例子来讲解JavaScript事件冒泡及使用注意事项。

如果你对JavaScript事件冒泡还没有什么印象,不妨先看看我之前写的一篇博客《JavaScript事件冒泡简介及应用》。本文讲究实用,不在对JavaScript事件冒泡基础知识进行过多的阐述。

在文章开始之前,现在先看看下面这样一个需求:下面HTML假设描述的是一个WebGame项目包裹栏(玩过网游的人应该知道什么是包裹栏或物品栏)的外框架,拖动这个包裹标题栏可以拖动这个包裹到页面上任何位置,而点击标题栏右侧的“×”关闭按钮可以关闭这个包裹栏的显示。你通过观察HTML结构,可以发现那个关闭按钮其实是一个A链接,而且是作为标题栏H5的子元素存在。要想拖动一个元素,我们会想到向拖动的句柄元素注册mousedown事件,而点击或“单击”关闭按钮就关闭这个包裹。根据这个需求,我们很快得到下面这样的代码。

JavaScript事件冒泡应用实例

<! 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" >
< head >
< meta  http-equiv ="Content-Type"  content ="text/html; charset=utf-8"   />
< title > JavaScript事件冒泡实例一 </ title >
< style  type ="text/css" >
#box 
{ width : 200px ;  height : 100px ;  border : 2px solid blue }
#box h5 
{ margin : 0 ;  padding : 2px 5px ;  font-size : 18px ;  text-align : right ;  background : blue ;  cursor : move }
#box h5 a 
{ text-decoration : none ;  color : #FFF }
</ style >
</ head >

< body >
< div  id ="box" >
< h5  onmousedown ="startDrag();" >< onclick ="closeBox();"  href ="javascript:void(0);" > × </ a ></ h5 >
< div  id ="testInfo" ></ div >
</ div >
< script  type ="text/javascript" >
function  startDrag()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' startDrag<br/> ' ;
}
function  closeBox()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' closeBox<br/> ' ;
}
</ script >
</ body >
</ html >

在上面的例子中,你发现点击关闭按钮的时候,标题栏的mousedown事件也触发了,显然这是在你意料之中的,因为你知道这正是JavaScript事件冒泡在起作用。其实你真正要的效果是点关闭按钮的时候,不要执行标题H5注册的mousedown事件,于是你想到了阻止事件冒泡,接着代码修改成下面这样:

< div  id ="box" >
< h5  onmousedown ="startDrag();" >< onclick ="closeBox(event);"  href ="javascript:void(0);" > × </ a ></ h5 >
< div  id ="testInfo" ></ div >
</ div >
< script  type ="text/javascript" >
function  startDrag()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' startDrag<br/> ' ;
}
function  closeBox(e)
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' closeBox<br/> ' ;
    stopBubble(e);
}

// 阻止事件冒泡函数
function  stopBubble(e)
{
    
if  (e  &&  e.stopPropagation)
        e.stopPropagation();
    
else
        window.event.cancelBubble
= true ;
}
</ script

结果你发现单击关闭按钮的时候,标题H5注册的mousedown事件还是执行了,这是怎么回事呢?其实细心的你可能发现,H5标题和A链接注册的事件不是一样的,在上面的代码中,我们在A链接注册的click事件中调用阻止事件方法,这只是意味着其父元素注册的“同类事件”不会执行,意思是说如果H5标题也注册了click事件,这个click事件不会执行,而这里的mousedown就继续执行了。这里的mousedown执行不是因为你点中标题栏而发生的,是由于你在单击click的时候伴随产生mousedown事件,然后又由于JavaScript事件冒泡机制的存在,事件向父级广播并被H5标题mousedown注册方法捕获。关于对这点的理解,可以参考我另外一篇博客《当onmousedown、onmouseup、onclick同时应用于同一个标签节点Element》。现在,通过分析,你应该知道怎么做了,小改上面的代码,只要把A链接的click事件改成与H5标题一样的mousedown事件,你想要的效果就实现了。

相关话题:

现在讲讲如何在使用jQuery开发时轻松阻止事件冒泡。jQuery作为一个优秀的脚本框架,对事件的封装及浏览器兼容处理自然也是很出众的。想了解更多也可阅读我的另外一篇博客《利用jQuery的$.event.fix函数统一浏览器event处理》。

使用jQuery要想阻止事件冒泡方法有二:

1、利用jQuery做过兼容处理的event对象,直接使用event.stopPropagation(),举例代码如下:

< script  type ="text/javascript"  src ="http://ajax.googleapis.com/ajax/libs/jquery/1.3/jquery.min.js" ></ script >
< div  id ="box" >
< h5  onmousedown ="startDrag();" >< onmousedown ="closeBox(event);"  href ="javascript:void(0);" > × </ a ></ h5 >
< div  id ="testInfo" ></ div >
</ div >
< script  type ="text/javascript" >
function  startDrag()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' startDrag<br/> ' ;
}
function  closeBox(e)
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' closeBox<br/> ' ;
    
var  event  =  $.event.fix(e);
    event.stopPropagation();
}
</ script >

2、在jQuery绑定的函数中返回false,即return false。注意:不是使用jQuery绑定的方法返回false是没有用的。代码如下:

< script  type ="text/javascript"  src ="http://ajax.googleapis.com/ajax/libs/jquery/1.3/jquery.min.js" ></ script >
< div  id ="box" >
< h5  onmousedown ="startDrag();" >< href ="javascript:void(0);" > × </ a ></ h5 >
< div  id ="testInfo" ></ div >
</ div >
< script  type ="text/javascript" >
function  startDrag()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' startDrag<br/> ' ;
}
function  closeBox()
{
    document.getElementById(
' testInfo ' ).innerHTML  +=   ' closeBox<br/> ' ;
    
return   false ;
}

$(
' #box a ' ).bind( ' mousedown ' ,closeBox);
</ script >

最后还要说明一下,利用第二个方法阻止事件冒泡,同时也阻止了浏览器的默认行为,在jQuery事件处理的源代码的handle方法中(jQuery JavaScript Library v1.3.2非压缩代码2700行)我们可以看到向下面这样的处理,其中event.preventDefault()是用于阻止浏览器默认行为。

handle :  function (event)
{
    
// other code......
     if  (ret  ===   false
    {
        event.preventDefault();
        event.stopPropagation();
    }
    
// other code......
}

你可能感兴趣的:(JavaScript)