el-drawer注册全局点击事件无效;el-dialog注册全局点击事件无效

提示:el-drawer或el-dialog注册全局点击事件无效,即抽屉或弹框外点击会触发事件,但抽屉和弹框内点击无反应

目前通过方案2:使用捕获阶段(推荐)解决


文章目录


这是因为 事件冒泡被阻止 了。el-drawer 组件内部的事件默认不会冒泡到 window,这可能是因为:

  1. el-drawer阻止了事件冒泡 (事件修饰符 .stop
  2. 事件目标被判断为不在 window
  3. el-drawer的层叠上下文导致事件被拦截

解决方案

方案1:将事件监听器改为 document

javascript 复制代码
onMounted(() => {
  document.addEventListener("click", clearFlag)
})

onBeforeUnmount(() => {
  document.removeEventListener("click", clearFlag)
})

方案2:使用捕获阶段(推荐)

javascript 复制代码
onMounted(() => {
  document.addEventListener("click", clearFlag, true) // 第三个参数 true 表示捕获阶段
})

onBeforeUnmount(() => {
  document.removeEventListener("click", clearFlag, true)
})

方案3:检查事件路径

如果上面的方法无效,可以检查事件是否被阻止:

javascript 复制代码
const clearFlag = (event) => {
  console.log('事件触发', event.target)
  console.log('事件路径', event.composedPath())
  // 检查是否被阻止
  console.log('defaultPrevented', event.defaultPrevented)
}

方案4:针对 el-drawer 的特殊处理

javascript 复制代码
import { nextTick } from 'vue'

onMounted(() => {
  nextTick(() => {
    // 获取 drawer 的 DOM
    const drawer = document.querySelector('.el-drawer')
    if (drawer) {
      drawer.addEventListener('click', clearFlag)
    }
    
    // 同时监听 document 作为后备
    document.addEventListener('click', clearFlag)
  })
})

onBeforeUnmount(() => {
  const drawer = document.querySelector('.el-drawer')
  if (drawer) {
    drawer.removeEventListener('click', clearFlag)
  }
  document.removeEventListener('click', clearFlag)
})

方案5:使用 Vue 的自定义事件(如果 drawer 是子组件)

vue 复制代码
<!-- 父组件 -->
<template>
  <ChildDrawer @drawer-click="clearFlag" />
</template>

<!-- 子组件 drawer 内部 -->
<template>
  <el-drawer>
    <div @click="$emit('drawer-click')">
      <!-- 内容 -->
    </div>
  </el-drawer>
</template>

调试建议

javascript 复制代码
const clearFlag = (event) => {
  console.log('事件触发', {
    target: event.target,
    currentTarget: event.currentTarget,
    eventPhase: event.eventPhase, // 1: 捕获, 2: 目标, 3: 冒泡
    bubbles: event.bubbles, // 是否冒泡
    defaultPrevented: event.defaultPrevented,
    composed: event.composed, // 是否能跨越 Shadow DOM
  })
}

推荐先用方案1或方案2 ,大多数情况下将 window 改为 document 并使用捕获阶段就能解决问题。

相关推荐
踩着两条虫29 分钟前
VTJ.PRO 核心架构全公开!从设计稿到代码,揭秘AI智能体如何“听懂人话”
前端·vue.js·ai编程
用头发抵命2 小时前
Vue 3 中优雅地集成 Video.js 播放器:从组件封装到功能定制
开发语言·javascript·ecmascript
蓝冰凌2 小时前
Vue 3 中 defineExpose 的行为【defineExpose暴露ref变量】详解:自动解包、响应性与实际使用
前端·javascript·vue.js
奔跑的呱呱牛2 小时前
generate-route-vue基于文件系统的 Vue Router 动态路由生成工具
前端·javascript·vue.js
sp42a3 小时前
在 NativeScript-Vue 中实现流畅的共享元素转场动画
vue.js·nativescript·app 开发
柳杉3 小时前
从动漫水面到赛博飞船:这位开发者的Three.js作品太惊艳了
前端·javascript·数据可视化
TON_G-T4 小时前
day.js和 Moment.js
开发语言·javascript·ecmascript
Irene19914 小时前
JavaScript 中 this 指向总结和箭头函数的作用域说明(附:call / apply / bind 对比总结)
javascript·this·箭头函数
2501_921930834 小时前
ReactNative项目OpenHarmony三方库集成实战:react-native-appearance(更推荐自带的Appearance)
javascript·react native·react.js