模拟点击,踩到React事件的坑

问题

前两天在掘金大模型子站遇到一个问题,使用 $0.click() 模拟点击弹窗的关闭按钮时,出现报错 click is not a function 且弹窗没有关闭,原因是关闭按钮使用的 svg 元素没有 click 方法。

随后使用 $0.dispatchEvent(new MouseEvent('click')),代码没报错但是弹窗没有关闭。

解决

经过调试分析发现,将 new MouseEvent('click') 的第二个参数设置为 {bubbles: true} 就能关闭弹窗,成功代码: $0.dispatchEvent(new MouseEvent('click',{bubbles: true}))

原因

  • 1、HTMLElement.click() 失败原因:因为 svg 元素没有 click 函数
  • 2、EventTarget.dispatchEvent(new MouseEvent('click')) 失败原因:掘金大模型子站使用 React 编写,模拟点击事件没有冒泡,未能触发 React onClick 事件
    • React 的合成事件区分捕获与冒泡,onClick 只在冒泡阶段被触发,捕获阶段不会触发
    • EventTarget.dispatchEvent(new MouseEvent('click')) 分发的事件没有冒泡阶段,所以没有触发 React onClick 事件
    • 代码修改为 EventTarget.dispatchEvent(new MouseEvent('click',{bubbles:true})),分发的事件既有捕获阶段又有冒泡阶段,可以触发 React onClick 事件

分析要点

搭建最简复现环境

为了解决调试不便和影响因素过多等问题,在 codesandbox 搭建了一个最简复现环境,关键代码如下:

javascript 复制代码
// App.js
export default function App() {
  function clickHandler() {
    console.log("onClick");
  }

  return (
    <div className="App" onClick={clickHandler}>
      <h1>Hello CodeSandbox</h1>
      <h2>Start editing to see some magic happen!</h2>
    </div>
  );
}

对比分析

在上述环境中,HTMLElement.click() 可以正常调用 clickHandler 函数,而 EventTarget.dispatchEvent(new MouseEvent('click')) 没有调用 clickHandler 函数。

如果嗅觉足够灵敏,再搭建一个类似的原生 JavaScript 复现环境,很快就能发现前面这两种方式都能调用 click 事件绑定的函数,只是事件对象属性略有不同,也能很快发现问题出在 new MouseEvent 的第二个参数上。

分析调用栈

为了排查失败的真正原因,可以在 clickHandler 函数中添加断点,对调用栈中的函数逐个排查。

关键源码分析

事件队列不同

经过排查,在 React 代码 react-dom.development.js 文件中的 dispatchEventsForPlugins 函数发现了端倪:

  • 使用 $0.click() 触发点击事件,该函数被执行两次
    • 第一次函数中的变量 dispatchQueue 为空数组,不会调用 clickHandler 函数
    • 第二次函数中的变量 dispatchQueue 不为空,会调用 clickHandler 函数
  • 使用 $0.dispatchEvent(new MouseEvent('click')) 触发点击事件,该函数被执行一次,函数中的 dispatchQueue 为空数组,不会调用 clickHandler 函数

获取监听器的名称与结果不同

从上图可以看到 dispatchQueue[0].listeners[0].listener 的值就是 App.js 中的 clickHandler 函数。这里的 dispatchQueue 可以简单理解为即将执行的事件监听器组成的队列,队列不为空时其中的监听器会逐一执行。

接下来的关键点是搞清楚队列 dispatchQueue 如何被填充,排查发现是被 extractEvents$5 函数所修改与填充,队列中元素的 listener 是通过 getListener 函数获得:

  • 使用 $0.click() 触发点击事件,getListener 函数被执行两次
    • 第一次函数的入参 registrationName 值为 onClickCapture,返回结果为 undefined
    • 第二次函数的入参 registrationName 值为 onClick,返回结果为 App.js 中的 clickHandler 函数
  • 使用 $0.dispatchEvent(new MouseEvent('click')) 触发点击事件,getListener 函数被执行一次,函数的入参 registrationName 值为 onClickCapture,返回结果为 undefined

通过以上分析可以推断出问题的根本原因。

总结

  • 模拟点击事件常用两种方式:HTMLElement.click()EventTarget.dispatchEvent()
  • 使用 EventTarget.dispatchEvent() 时,记得给事件加上冒泡避免踩坑
  • React 的合成事件区分捕获与冒泡,onClickCapture 在捕获阶段触发,onClick 在冒泡阶段触发,其他事件类似

EventTarget.dispatchEvent(new MouseEvent('click'))VueAngular 中可以正常调用

相关资料

相关推荐
电商API&Tina10 分钟前
1688 拍立淘接口(item_search_img)测试与接入实战心得
java·大数据·前端·物联网·oracle·json
不想说话的麋鹿25 分钟前
「性能优化」虚拟列表极致优化实战:从原理到源码,打造丝滑滚动体验
前端·vue.js·面试
ouzz26 分钟前
使用 react-canvas 制作一个 Figma 工具:从画布到编辑器
前端·javascript
万少30 分钟前
AI 智能记账 Skill,基于飞书 CLI + 多维表格构建。
前端
颜酱30 分钟前
语音合成与视觉模型api接入实现
前端·javascript·人工智能
你听得到1132 分钟前
Get 这波之后,我把 Flutter 状态管理重新看了一遍:新项目到底该选谁?
前端·flutter·架构
一天睡25小时1 小时前
做产品前,先别急着写代码:我是怎么判断一个点子值不值得做的
前端
霍理迪1 小时前
TS—函数、类、泛型
前端
cc.ChenLy1 小时前
浏览器缓存机制详解:如何彻底解决前端代码更新后的缓存问题
前端
阿珊和她的猫1 小时前
使用 TypeScript 实现数组类型判断方法
javascript·typescript·状态模式