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 相关文章推荐
由prototype_1.3.1进入javascript殿堂-类的初探
Nov 06 Javascript
js中function()使用方法
Dec 24 Javascript
js 触发select onchange事件代码
Mar 20 Javascript
JS+CSS实现鼠标滑过时动态翻滚的导航条效果
Sep 24 Javascript
AngularJS中如何使用$parse或$eval在运行时对Scope变量赋值
Jan 25 Javascript
javascript移动开发中touch触摸事件详解
Mar 18 Javascript
AngularJS基础 ng-class-odd 指令示例
Aug 01 Javascript
微信小程序 图片上传实例详解
May 05 Javascript
JavaScript基础之流程控制语句的用法
Aug 31 Javascript
VueJs组件prop验证简单介绍
Sep 12 Javascript
vue2中的keep-alive使用总结及注意事项
Dec 21 Javascript
JS使用队列对数组排列,基数排序算法示例
Mar 02 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
PHPShop存在多个安全漏洞
2006/10/09 PHP
setAttribute 与 class冲突解决
2008/02/17 Javascript
JavaScript获取XML数据附示例截图
2014/03/05 Javascript
div失去焦点事件实现思路
2014/04/22 Javascript
js事件监听机制(事件捕获)总结
2014/08/08 Javascript
jQuery后代选择器用法实例
2014/12/23 Javascript
JS实现超简单的仿QQ折叠菜单效果
2015/09/21 Javascript
javascript省市级联功能实现方法实例详解
2015/10/20 Javascript
JavaScript中的编码和解码函数
2017/02/15 Javascript
ES6中module模块化开发实例浅析
2017/04/06 Javascript
js实现日期显示的一些操作(实例讲解)
2017/07/27 Javascript
canvas轨迹回放功能实现
2017/12/20 Javascript
nodejs npm错误Error:UNKNOWN:unknown error,mkdir 'D:\Develop\nodejs\node_global'at Error
2019/03/02 NodeJs
vue-router跳转时打开新页面的两种方法
2019/07/29 Javascript
vue监听用户输入和点击功能
2019/09/27 Javascript
Element Steps步骤条的使用方法
2020/07/26 Javascript
Element-ui el-tree新增和删除节点后如何刷新tree的实例
2020/08/31 Javascript
python通过scapy获取局域网所有主机mac地址示例
2014/05/04 Python
9种python web 程序的部署方式小结
2014/06/30 Python
简单谈谈python中的多进程
2016/11/06 Python
python中正则的使用指南
2016/12/04 Python
python版本坑:md5例子(python2与python3中md5区别)
2017/06/20 Python
详解flask入门模板引擎
2018/07/18 Python
Python实现堡垒机模式下远程命令执行操作示例
2019/05/09 Python
Django集成搜索引擎Elasticserach的方法示例
2019/06/04 Python
Python tornado上传文件的功能
2020/03/26 Python
HTML5如何为形状图上颜色怎么绘制具有颜色和透明度的矩形
2014/06/23 HTML / CSS
世界上最大的各式箱包网络零售店:eBag
2016/07/21 全球购物
英国Lookfantastic中文网站:护肤品美妆美发购物(英国直邮)
2020/04/27 全球购物
护理专业本科生自荐信
2013/10/01 职场文书
工商管理本科毕业生求职信范文
2013/10/05 职场文书
大学毕业生求职自荐信
2014/02/20 职场文书
部门优秀员工推荐信
2015/03/24 职场文书
杨善洲电影观后感
2015/06/04 职场文书
办公室规章制度范本
2015/08/04 职场文书
使用feign服务调用添加Header参数
2021/06/23 Java/Android