React冒泡和阻止冒泡的应用详解


Posted in Javascript onAugust 18, 2020

阻止事件冒泡分三种:

1:阻止合成事件往最外层document上的事件冒泡,用e.nativeEvent.stopImmediatePropagation();

2: 合成事件间的冒泡,使用 e.stopPropagation();

3:阻止合成事件,往处理document上的其他原生事件冒泡,需要通过e.target来判断,示例代码如下。

import React,{ Component } from 'react';
import ReactDOM,{findDOMNode} from 'react-dom';

class Counter extends Component{
constructor(props){
super(props);

this.state = {
count:0,
    }
  }

handleClick(e){
this.setState({count:++this.state.count});
  }
render(){
return(
<div ref="test">
<p>{this.state.count}</p>
<a ref="update" onClick={(e)=>this.handleClick(e)}>更新</a>
</div>
    )
  }

componentDidMount() {
document.body.addEventListener('click',e=>{
// 通过e.target判断阻止冒泡
      if(e.target&&e.target.matches('a')){
return;
      }
console.log('body');
    })
  }
}

var div1 = document.getElementById('content');

ReactDOM.render(<Counter/>,div1,()=>{});

需求

最近在写react的项目,需要手写一个自定义的菜单,和antd的menu不同,需要点击一级菜单后弹出类似一个Drawer展示二级和三级菜单,且菜单样式自定义,都在一个Drawer里展示。

难点

其中难点在于点击一级菜单时弹出Drawer,点击除Drawer和一级菜单项之外的dom,Drawer收起。

问题

乍一想就是给document添加一个点击事件监听,给Drawer和一级菜单添加阻止冒泡,思路确实如此,后面实现中发现:
react为提高性能,有自己的一套事件处理机制,相当于将事件代理到全局进行处理,也就是说监听函数并未绑定到Dom上。 因此阻止react的事件冒泡e.stopPropagation(),就不发阻止原生事件的冒泡,表现为点击Drawer也会收起Drawer;禁用原生事件冒泡e.nativeEvent.stopPropagation(),React的监听函数就调用不到,表现为点击Drawer以外dom,Drawer不会收起。这些都不是我们想要的。

解决方案

正确的姿势应该是判断event.target对象是否是目标对象或包含目标对象或被包含目标对象,以此来决定是否触发事件

componentDidMount() {
  document.addEventListener('click', this.hideDrawer);
 }

 componentWillUnmount() {
  document.removeEventListener('click', this.hideDrawer);
 }

 hideDrawer = e => {
  const { closeDrawer } = this.props;
  // 找到不需要关闭的dom 一级菜单
  const tabContent = document.querySelectorAll('.ant-menu-submenu-vertical');
  // 找到不需要关闭的dom   Drawer
  const drawerContent = document.querySelector('#menuDrawer');
  // 判断当前点击的dom对象有没有包含在目标dom中
  const isHave = Array.from(tabContent).some(item => item.contains(e.target));
  // 不包含则关闭Drawer 包含就走其他的业务逻辑了
  if (tabContent !== null && !(isHave || drawerContent.contains(e.target))) {
   closeDrawer();
  }
 };

到此这篇关于React冒泡和阻止冒泡的应用详解的文章就介绍到这了,更多相关React冒泡和阻止冒泡内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木! 

Javascript 相关文章推荐
javascript判断机器是否联网的2种方法
Aug 09 Javascript
使用iframe window的scroll方法控制iframe页面滚动
Mar 05 Javascript
jquery JSON的解析方式示例介绍
Jul 27 Javascript
浅谈Unicode与JavaScript的发展史
Jan 19 Javascript
jquery实现两边飘浮可关闭的对联广告
Nov 27 Javascript
JavaScript中Number对象的toFixed() 方法详解
Sep 02 Javascript
jquery+ajax实现省市区三级联动效果简单示例
Jan 04 Javascript
对存在JavaScript隐式类型转换的四种情况的总结(必看篇)
Aug 31 Javascript
jquery 获取索引值在一定范围的列表方法
Jan 25 jQuery
js实现盒子拖拽动画效果
Aug 09 Javascript
Vue包大小优化的实现(从1.72M到94K)
Feb 18 Vue.js
ztree+ajax实现文件树下载功能
May 18 Javascript
JavaScript数组排序的六种常见算法总结
Aug 18 #Javascript
js实现简单扫雷
Nov 27 #Javascript
基于JavaScript实现大文件上传后端代码实例
Aug 18 #Javascript
javascript实现扫雷简易版
Aug 18 #Javascript
详解Vue的组件中data选项为什么必须是函数
Aug 17 #Javascript
Openlayers实现扩散的动态点(水纹效果)
Aug 17 #Javascript
openLayer4实现动态改变标注图标
Aug 17 #Javascript
You might like
PHP实现文件安全下载
2006/10/09 PHP
具有时效性的php加密解密函数代码
2013/06/19 PHP
thinkphp实现多语言功能(语言包)
2014/03/04 PHP
ThinkPHP的URL重写问题
2014/06/22 PHP
php修改指定文件后缀的方法
2014/09/11 PHP
用正则xmlHttp实现的偷(转)
2007/01/22 Javascript
菜鸟学习JavaScript小实验之函数引用
2010/11/17 Javascript
jquery实现兼容浏览器的图片上传本地预览功能
2013/10/14 Javascript
纯js和css实现渐变色包括静态渐变和动态渐变
2014/05/29 Javascript
Javascript遍历Html Table示例(包括内容和属性值)
2014/07/08 Javascript
jQuery实现带滚动线条导航效果的方法
2015/01/30 Javascript
JavaScript使用cookie实现记住账号密码功能
2015/04/27 Javascript
javascript实现下拉提示选择框
2015/12/29 Javascript
javascript随机抽取0-100之间不重复的10个数
2016/02/25 Javascript
JS实现复制内容到剪贴板功能兼容所有浏览器(推荐)
2016/06/17 Javascript
JavaScript表单焦点自动切换代码
2016/07/24 Javascript
Bootstrap分页插件之Bootstrap Paginator实例详解
2016/10/15 Javascript
ajax实现动态下拉框示例
2017/01/10 Javascript
Vue Element 分组+多选+可搜索Select选择器实现示例
2018/07/23 Javascript
一文快速详解前端框架 Vue 最强大的功能
2019/05/21 Javascript
js实现表格单列按字母排序
2020/08/12 Javascript
python实现爬虫统计学校BBS男女比例(一)
2015/12/31 Python
python实现多张图片拼接成大图
2019/01/15 Python
python基础梳理(一)(推荐)
2019/04/06 Python
Python列表元素常见操作简单示例
2019/10/25 Python
python多线程案例之多任务copy文件完整实例
2019/10/29 Python
印度尼西亚最大和最全面的网络商城:Blibli.com
2017/10/04 全球购物
NYX Professional Makeup官方网站:专业彩妆和美容产品
2019/10/29 全球购物
澳大利亚床上用品、浴巾和家居用品购物网站:Bambury
2020/04/16 全球购物
C#面试题
2016/05/06 面试题
测量实习生自我鉴定
2013/09/19 职场文书
甜品店创业计划书
2014/09/21 职场文书
2015年保险业务员工作总结
2015/05/27 职场文书
给校长的建议书作文500字
2015/09/14 职场文书
三严三实·严以修身心得体会
2016/01/15 职场文书
JavaScript利用html5新方法操作元素类名详解
2021/11/27 Javascript