这里有新鲜出炉的 Javascript 教程,程序狗速度看过来!
Javascript 是一种由 Netscape 的 LiveScript 发展而来的原型化继承的基于对象的动态类型的区分大小写的客户端脚本语言,主要目的是为了解决服务器端语言,比如 Perl,遗留的速度问题,为客户提供更流畅的浏览效果。
然而,在今天一些大型的 web 交互项目中,比如大型的 WebGame 项目,JavaScript 事件冒泡影响是值得重视的。本文通过一个简单的例子来讲解 JavaScript 事件冒泡及使用注意事项。 如果你对 JavaScript 事件冒泡还没有什么印象,不妨先看看我之前写的一篇博客《JavaScript 事件冒泡简介及应用》。本文讲究实用,不在对 JavaScript 事件冒泡基础知识进行过多的阐述。 在文章开始之前,现在先看看下面这样一个需求:下面 html 假设描述的是一个 WebGame 项目包裹栏(玩过网游的人应该知道什么是包裹栏或物品栏)的外框架,拖动这个包裹标题栏可以拖动这个包裹到页面上任何位置,而点击标题栏右侧的 "×" 关闭按钮可以关闭这个包裹栏的显示。你通过观察 HTML 结构,可以发现那个关闭按钮其实是一个 A 链接,而且是作为标题栏 H5 的子元素存在。要想拖动一个元素,我们会想到向拖动的句柄元素注册 mousedown 事件,而点击或 "单击" 关闭按钮就关闭这个包裹。根据这个需求,我们很快得到下面这样的代码。
在一些传统的小型 WEB 应用开发过程中,JavaScript 通常只是拿来做表单验证而以,所以你很少会遇到因为 JavaScript 事件冒泡而影响功能的实现情况,又或者事件冒泡对最终实现效果影响不大,可忽略。
[Ctrl+A 全选 注:
- <!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>
- <h5 onmousedown="startDrag();">
- <a href="javascript:void(0);">
- ×
- </a>
- </h5>
- <div>
- </div>
- </div>
- <script type="text/javascript">
- function startDrag() {
- document.getElementById('testInfo').innerHTML += 'startDrag
- ';
- }
- function closeBox() {
- document.getElementById('testInfo').innerHTML += 'closeBox
- ';
- }
- </script>
- </body>
- </html>
如需引入外部 Js 需刷新才能执行
]在上面的例子中,你发现点击关闭按钮的时候,标题栏的 mousedown 事件也触发了,显然这是在你意料之中的,因为你知道这正是 JavaScript 事件冒泡在起作用。其实你真正要的效果是点关闭按钮的时候,不要执行标题 H5 注册的 mousedown 事件,于是你想到了阻止事件冒泡,接着代码修改成下面这样:
- <div id="box">
- <h5 onmousedown="startDrag();"><a 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.preventDefault(),举例代码如下:
- <script type="text/javascript" src="http://img.phperz.com/jslib/jquery/jquery.js">
- </script>
- <div id="box">
- <h5 onmousedown="startDrag();">
- <a 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://img.phperz.com/jslib/jquery/jquery.js">
- </script>
- <div id="box">
- <h5 onmousedown="startDrag();">
- <a 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......
- }
作者:WebFlash
来源: http://www.phperz.com/article/17/0427/285787.html