preventDefault, stopPropagation, return false -JS事件处理中的坑
我们以一个文件上传ui重设计为例子来探讨这几个函数的区别:

其中的html代码如下:
<div class="file-upload">
<input type="file" name="upload-file" class="file-upload__input" style="display: none;" /> <div class="file-upload__drop-zone">
<span class="file-upload__drop-zone-text">Drop files here</span>
<a href="#" class="file-upload__btn--upload">Upload files</a>
</div>
</div>
HTML包含有三部分内容:
1. 一个inpyout空间来处理文件上传的对话.在这里,我们设置为hidden,因为我们并不想使用浏览器默认给出的控件;
2. 一个class为file-upload__dropzone的div元素,它作为主要的"drop zone"拖拽区,如果有代码支持的话,可以直接拖拽文件到这个区域。
3. 一个a元素,它具有file-upload__btn--upload作为其css类,它作为实际的"upload files"按钮,当我们点击它时,就能够打开文件选择对话框
Javascript部分:
function fileUpload() {
document.querySelector('.file-upload__input').click();
}
const dropzone = document.querySelector('.file-upload__drop-zone');
const button = document.querySelector('.file-upload__btn--upload');
dropzone.addEventListener('click', fileUpload);
button.addEventListener('click', fileUpload);
js也有三部分组成:
1. 一个fileUpload函数用于触发input元素的click事件
2.div和a元素获取并赋予变量
3.给这几个元素添加事件侦听,当click时就调用fileUpload函数以模拟触发file input的点击
如果我们直接实验,我们会发现很多诡异的行为:当第一个对话框打开并选择文件后,又有第二个会打开
event.preventDefault()
该函数用于阻止浏览器的默认行为(比如当我们点击a标签的时候,自然就打开a中的链接,这个行为就是默认的行为),但是并不会阻止事件的bubbling,也就是说其父亲元素依然会接受到对应该元素的事件
在我们的例子中,当我们点击upload files按钮时,就会调用fileUpload函数,这和我们的预期时一致的。
而作为a标签,其默认行为是引导浏览器nav到href所指定的url中,在这里,我们设置为#,大多数浏览器解释为跳到页面的top处。
而跳转到页面的top处可能并不是我们希望的行为,因此,我们可以通过使用preventDefault方法来阻止这个行为。
通过修改我们的js代码,我们可以阻止a标签的默认行为,但是依然会调用fileUpload函数:
dropzone.addEventListener('click', fileUpload);
button.addEventListener('click', (event) => {
event.preventDefault();
fileUpload();
});
这时,虽然点击a标签不再跳转到页面top处,但是文件对话框依然会弹出两次。。。
需要注意的是:
1.preventDefault有继承性,如果在父亲元素的事件处理函数中调用了event.preventDefault,那么子元素对应的event默认行为也将丢失
2.如果希望在子组件中要覆盖这个行为,我们可以通过在子组件的event handler中最后执行 return true; 来恢复!
调试evenyout.preventDefault
有的时候,浏览器的行为非常奇怪,往往就是因为在某些组件上调用过preventDefault函数,而我们又无法知晓是哪个元素身上调用的,这时需要调试。
一个比较好的聪敏的办法是使用类似proxy代理wrapper将event的该函数重新包装:
var oldEPD = Event.prototype.preventDefault;
Event.prototype.preventDefault = function() {
debugger;
oldEPD.call(this);
};
这样的话就会在有该调用时代码停止供我们查找
event.stopPropagation()
这个函数可以阻止发生在子元素上的事件继续向上冒泡,但是不会阻止浏览器的默认行为.
上面的例子中,由于我们点击a元素后,a元素的click handler中会触发对话框,而随后a的click事件继续bubble到上级元素,也就是dropzone,而该dropzone元素也有监听click事件因此就又会被调用fileUpload,这就是为什么文件对话框弹出两次的原因。我们可以通过stopPropagation来阻止事件继续网上冒泡。
我们来看终级解决方案:
dropzone.addEventListener('click', fileUpload);
button.addEventListener('click', (event) => {
event.preventDefault();
event.stopPropagation();
fileUpload();
});
return false
通常return false这个代码仅仅在jQuery的事件处理函数中有效,对于原生的js代码并无任何影响。
而在jquery的代码中return false又有两个效果:
1. preventDefault
2. stopPropagation
const dropzone = $('.file-upload__drop-zone');
const button = $('.file-upload__btn--upload');
$(dropzone).on('click', fileUpload);
$(button).on('click', (event) => {
fileUpload();
return false;
});
preventDefault, stopPropagation, return false -JS事件处理中的坑的更多相关文章
- event.stopPropagation()、event.preventDefault()与return false的区别
做小demo时经常用到return false来取消默认事件,但一直不是很懂它和preventDefault()等的区别,今天查了查文档和大神们的博客,在这里对相关知识点做一个总结 首先开门见山,总结 ...
- 【转】stopPropagation, preventDefault 和 return false 的区别
因为有父, 子节点同在, 因为有监听事件和浏览器默认动作之分. 使用 JavaScript 时为了达到预期效果经常需要阻止事件和动作执行. 一般我们会用到三种方法, 分别是 stopPropagat ...
- stopPropagation, preventDefault 和 return false 的区别
因为有父, 子节点同在, 因为有监听事件和浏览器默认动作之分. 使用 JavaScript 时为了达到预期效果经常需要阻止事件和动作执行. 一般我们会用到三种方法, 分别是 stopPropagati ...
- stopPropagation, preventDefault 和 return false
e.stopPropagation()此方法用于阻止事件冒泡或者事件捕获.IE8及以下中没有此方法,使用e.cancelBubble=false 来阻止事件冒泡. 当标准W3C中,事件包括捕获阶段和冒 ...
- event.stopPropagation(),event.preventDefault()和return false的区别
event.stopPropagation(),event.preventDefault()和return false的区别 1.event.stopPropagation()方法 这是阻止事件的冒泡 ...
- event.preventDefault() vs. return false
使用jquery方式的话,以下是等效的 return false === event.stopPropagation + event.preventDefault() //1. event.preve ...
- js的stopPropagation()、cancelBubble、preventDefault()、return false的分析
个人笔记,如有错误,望指出. 事件冒泡,举个列子: <li> <a href='http://www.baidu.com'>点击a</a> </li> ...
- JS 之 阻止事件冒泡,阻止默认事件,event.stopPropagation()和event.preventDefault(),return false的区别
在前端开发中,有时我们需要阻止冒泡和阻止默认事件的发生. 一.event.stopPropagation() 阻止事件的冒泡,不让事件向documen上蔓延,但是默认事件任然会执行,当调用这个方法的时 ...
- 阻止事件冒泡,阻止默认事件,event.stopPropagation()和event.preventDefault(),return false的区别
1.event.stopPropagation()方法 这是阻止事件的冒泡方法,不让事件向documen上蔓延,但是默认事件任然会执行,当你掉用这个方法的时候,如果点击一个连接,这个连接仍然会被打开, ...
随机推荐
- Zuul 1.x 的工作原理
Zuul简介 Zuul在微服务架构中,可以作为提供动态路由,监控,弹性,安全等边缘服务的框架.在Netflix,被用作所有请求到达streaming application的前门.Zuul使用一系列不 ...
- Vue.js 源码分析(六) 基础篇 计算属性 computed 属性详解
模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的.在模板中放入太多的逻辑会让模板过重且难以维护,比如: <div id="example">{{ messag ...
- Microsoft Office自制安装指南 —Nusen_Liu
Microsoft Word 2010 正版下载安装步骤 版权来自:Nusen_Liu 1. 解压文件(推荐解压到当前文件夹,大神也可以自定义的)下载地址在第16步 (*^__^*) 2. 解 ...
- iOS 禁用`URL Scheme`和`Universal Link`(通用链接)
为什么要禁用URL Scheme和Universal Link(通用链接) 通常我们APP中都会嵌套一些web页面,有时我们的web页面会被DNS劫持从而跳转到其他APP中:或者是某些APP的Univ ...
- 安装quickLook插件以及解决如何不能读取offic问题
目录 @(安装quickLook插件) quickLook插件是Mac上的快速浏览的一个功能,现在win10系统上也能安装插件,这个插件可以快速浏览txt,doc,图片,表格等文件如下图: 我认为最方 ...
- web网页利用JavaScript实现对摄像头的调用
实现效果: 代码如下: <!DOCTYPE html> <html lang="zh"> <head> <meta charset=&qu ...
- I2C硬件与模拟的区别
硬件I2C对应芯片上的I2C外设,有相应I2C驱动电路,其所使用的I2C管脚也是专用的,因而效率要远高于软件模拟的I2C:一般也较为稳定,但是程序较为繁琐. 硬件(固件)I2C是直接调用内部寄存器进行 ...
- 初学JavaScript正则表达式(六)
JavaScript预定义类 ab+数字+任意字符 ab[0-9][^\r\n] 等价于 ab\d. '@123@abc@'.replace(/@./g,'Q') Q23Qbc@ 将"@加任 ...
- 201871010111-刘佳华《面向对象程序设计(java)》第八周学习总结
201871010111-刘佳华<面向对象程序设计(java)>第八周学习总结 实验七 接口的定义与使用 实验时间 2019-10-18 第一部分:知识总结 接口的概念: ①java为了克 ...
- (转)新建maven项目时报错Error:Maven Resources Compiler: Maven project configuration required for module 'XX'解决方法
转载地址:https://blog.csdn.net/qq784515681/article/details/85070195 在新建maven项目时,Problems中报错: Error:Maven ...