原址:http://andward.github.io/js/jquery/2014/12/29/how-to-stop-event-bubble.html

用了一段时间 google inbox,赶脚 mail task 的样式很酷炫,于是也模仿着做了一个 task 管理系统。
Google task 有个 behavior 是点击页面别处,当前展开的 mail 就会收起。查了查实现发现这里应该是阻止了 js 事件的冒泡。

关于事件冒泡的官方解释:
After an event triggers on the deepest possible element, it then triggers on parents in nesting order. - 总结一下就是说子元素会逐层往上层查找,直到找到绑定了触发事件的父元素。这儿有个简单的例子:

<div class='class1'>
    <div class='class2'></div>
</div>
<script type='text/javascript'>
$('.class1').click(function(event) {
    /* Act on the event */
    $(this).css('background', 'red');
});
</script>

运行代码,如果点击 class2 的 div,整个 class1 的区域仍然会变红。点击事件从子元素冒泡到了绑定 click 事件的父元素上。

js 的事件冒泡其实有不少好处,例如可以减少同一事件的重复绑定。如果 class1 下面有 10 个 div 都需要触发同一事件而没有事件冒泡,则我们需要写 10 次相同的事件函数。。那是多么的坑爹~

为了实现 google task 的效果,就需要阻止$(document).click()事件的冒泡。很多文章都推荐用 stopPropagation 方法。这个函数可以阻止事件冒泡:

$(document).click(function() {
  // Hide the menus if visible.
});

$('.class2').click(function(event){
  event.stopPropagation();
});

当点击 class2 时 由于其用了 stopPropagation,不会触发到绑定在 $(document) 的点击事件。js 和 jquery 都支持这个方法。

然而有些同学对这种强行阻止冒泡的行为表示了担忧。见帖:dangers-stopping-event-propagation。大意是指这种全局阻止事件冒泡会导致某些 lib 产生 bug(博主举了个 boostrap 的 bug),并推荐一下用法:

$(document).on('click', function(event) {
  if (!$(event.target).closest('.class2').length) {
    // do sth
  }
});

大体思路就是检查一下事件绑定的元素以及父元素是不是想要的,不是就触发方法~

两种方法我试了试在我这里都 work~~最后我用了第二种方法_^


↙↙↙阅读原文可查看相关链接,并与作者交流