news 2026/4/16 14:50:54

Vue3文件上传使用@click.stop解决多次触发(事件冒泡)的问题

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Vue3文件上传使用@click.stop解决多次触发(事件冒泡)的问题

关键代码:

// 触发文件选择 const triggerFileSelect = (event?: Event) => { console.log("触发文件选择"); console.log("事件目标:", event?.target); console.log("当前目标:", event?.currentTarget); console.log("事件路径:", event?.composedPath()); if (fileInputRef.value) { fileInputRef.value.click(); } }; <!-- 上传按钮 --> <div v-if="actionLayout.includes(`upload`) && fileSources.length < props.maxCount" class="upload-button-container" title="点击上传文件" @click="triggerFileSelect"> <!-- 上传图标 --> <!-- <el-icon> <Plus /></el-icon> --> <!-- 上传按钮(包含上传图标) --> <el-button type="primary" :icon="Plus" circle size="small" /> <input ref="fileInputRef" type="file" multiple style="display: none" @change="handleFileSelect" /> </div>

点击一下div(class="upload-button-container"),执行两次triggerFileSelect,多次触发(事件冒泡)的应用情况:

这个问题是因为事件冒泡导致的多次触发。当你点击上传按钮时,事件触发了两次:

  1. 第一次是你实际点击div.upload-button-container时触发

  2. 第二次是因为fileInputRef.value.click()触发了 input 的 click 事件,这个事件又冒泡到了父元素

可以修改代码,在调用triggerFileSelect时传递事件对象,并在触发 input 点击前阻止事件冒泡:

vue

<script setup lang="ts"> // ... 其他代码保持不变 ... // 触发文件选择 const triggerFileSelect = (event?: Event) => { console.log("触发文件选择"); console.log("事件目标:", event?.target); console.log("当前目标:", event?.currentTarget); console.log("事件路径:", event?.composedPath()); // 阻止事件冒泡,避免多次触发 if (event) { event.stopPropagation(); } if (fileInputRef.value) { fileInputRef.value.click(); } }; // ... 其他代码保持不变 ... </script> <template> <div class="file-selector-container"> <!-- 文件列表区域 --> <div class="file-list-container"> <!-- ... 其他代码保持不变 ... --> <!-- 上传按钮 --> <div v-if="actionLayout.includes(`upload`) && fileSources.length < props.maxCount" class="upload-button-container" title="点击上传文件" @click="triggerFileSelect($event)"> <!-- 这里传递 $event --> <!-- 上传图标 --> <!-- <el-icon> <Plus /></el-icon> --> <!-- 上传按钮(包含上传图标) --> <el-button type="primary" :icon="Plus" circle size="small" /> <input ref="fileInputRef" type="file" multiple style="display: none" @change="handleFileSelect" /> </div> </div> <!-- ... 其他代码保持不变 ... --> </div> </template>

或者,更简单的解决方案是直接阻止 input 元素的点击事件冒泡:

vue

<template> <div class="file-selector-container"> <!-- 文件列表区域 --> <div class="file-list-container"> <!-- ... 其他代码保持不变 ... --> <!-- 上传按钮 --> <div v-if="actionLayout.includes(`upload`) && fileSources.length < props.maxCount" class="upload-button-container" title="点击上传文件" @click="triggerFileSelect"> <!-- 上传图标 --> <!-- <el-icon> <Plus /></el-icon> --> <!-- 上传按钮(包含上传图标) --> <el-button type="primary" :icon="Plus" circle size="small" /> <!-- 添加 @click.stop 阻止事件冒泡 --> <input ref="fileInputRef" type="file" multiple style="display: none" @change="handleFileSelect" @click.stop /> </div> </div> <!-- ... 其他代码保持不变 ... --> </div> </template>

第二种方法更简洁,直接在 input 元素上使用@click.stop阻止其点击事件冒泡到父元素,这样就不会触发两次了。

选择其中一种方法即可解决问题。推荐第二种方法,因为它更简洁且语义更清晰。


修改后的代码:

// 触发文件选择 const triggerFileSelect = (event?: Event) => { console.log("触发文件选择"); console.log("事件目标:", event?.target); console.log("当前目标:", event?.currentTarget); console.log("事件路径:", event?.composedPath()); if (fileInputRef.value) { fileInputRef.value.click(); } }; <!-- 上传按钮 --> <div v-if="actionLayout.includes(`upload`) && fileSources.length < props.maxCount" class="upload-button-container" title="点击上传文件" @click="triggerFileSelect"> <!-- 上传图标 --> <!-- <el-icon> <Plus /></el-icon> --> <!-- 上传按钮(包含上传图标) --> <el-button type="primary" :icon="Plus" circle size="small" /> <!-- 使用 @click.stop 阻止事件冒泡,从而解决点击div(class="upload-button-container"),执行两次triggerFileSelect --> <input ref="fileInputRef" type="file" multiple style="display: none" @change="handleFileSelect" @click.stop /> </div>

应用效果:

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 12:31:08

EagleEye参数详解:TinyNAS搜索空间配置、FLOPs约束与精度权衡策略

EagleEye参数详解&#xff1a;TinyNAS搜索空间配置、FLOPs约束与精度权衡策略 1. EagleEye是什么&#xff1a;不止是轻量模型&#xff0c;而是可调的视觉决策引擎 EagleEye不是简单把大模型“砍一刀”变小的压缩工具&#xff0c;而是一套面向真实工业场景的可配置目标检测决策…

作者头像 李华
网站建设 2026/4/16 11:00:52

信道损耗与频率响应特性的定量关系解析

以下是对您提供的博文《信道损耗与频率响应特性的定量关系解析》进行 深度润色与专业重构后的版本 。本次优化严格遵循您的全部要求: ✅ 彻底去除AI痕迹,语言自然、专业、有“人味”,像一位深耕SI/PI领域十年的工程师在技术博客中娓娓道来; ✅ 打破模板化结构,取消所有…

作者头像 李华
网站建设 2026/4/16 12:37:32

Qwen-Image-Edit-2511艺术风格迁移,一键变油画风

Qwen-Image-Edit-2511艺术风格迁移&#xff0c;一键变油画风 你有没有试过把一张普通照片&#xff0c;几秒钟内变成伦勃朗笔下的光影杰作&#xff1f;或者让手机随手拍的街景&#xff0c;瞬间拥有梵高《星月夜》般的漩涡笔触和浓烈情绪&#xff1f;这不是后期滤镜的简单叠加&a…

作者头像 李华
网站建设 2026/4/14 6:10:31

MGeo避坑指南:从环境配置到成功推理全记录

MGeo避坑指南&#xff1a;从环境配置到成功推理全记录 1. 引言&#xff1a;中文地址匹配为什么总在“差不多”上栽跟头&#xff1f; 你有没有遇到过这样的情况&#xff1a; 物流系统里&#xff0c;“上海市浦东新区张江路123号”和“上海浦东张江路123号”被当成两个不同地址…

作者头像 李华
网站建设 2026/4/14 3:21:57

5大革新功能让你精通原神:Snap Hutao开源游戏辅助工具全解析

5大革新功能让你精通原神&#xff1a;Snap Hutao开源游戏辅助工具全解析 【免费下载链接】Snap.Hutao 实用的开源多功能原神工具箱 &#x1f9f0; / Multifunctional Open-Source Genshin Impact Toolkit &#x1f9f0; 项目地址: https://gitcode.com/GitHub_Trending/sn/Sn…

作者头像 李华