news 2026/5/10 21:09:43

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/5/8 20:54:02

零基础M3U8视频下载避坑指南:从问题诊断到高效下载的完整方案

零基础M3U8视频下载避坑指南&#xff1a;从问题诊断到高效下载的完整方案 【免费下载链接】m3u8-downloader 一个M3U8 视频下载(M3U8 downloader)工具。跨平台: 提供windows、linux、mac三大平台可执行文件,方便直接使用。 项目地址: https://gitcode.com/gh_mirrors/m3u8d/…

作者头像 李华
网站建设 2026/5/11 7:14:47

PCBA叠层设计图解说明:四层板堆叠结构解析

以下是对您提供的博文《PCBA叠层设计图解说明:四层板堆叠结构解析》的 深度润色与专业重构版本 。本次优化严格遵循您的全部要求: ✅ 彻底去除AI痕迹(无模板化表达、无空洞术语堆砌、无机械连接词) ✅ 摒弃“引言/概述/总结”等程式化结构,全文以 真实工程师视角+项目…

作者头像 李华
网站建设 2026/5/1 14:33:28

DCT-Net人像卡通化镜像快速上手:10秒加载+WebUI一键转换保姆级步骤

DCT-Net人像卡通化镜像快速上手&#xff1a;10秒加载WebUI一键转换保姆级步骤 你是不是也试过在各种AI工具里反复上传照片、调整参数、等半天才出图&#xff0c;结果还发现卡通效果生硬、五官变形、背景糊成一团&#xff1f;别折腾了——这次我们直接用一个专为人像优化的GPU镜…

作者头像 李华
网站建设 2026/5/8 10:10:40

DDColor实测:一张黑白照如何变成彩色艺术品

DDColor实测&#xff1a;一张黑白照如何变成彩色艺术品 你有没有翻过家里的老相册&#xff1f;泛黄的纸页间&#xff0c;祖父穿着笔挺军装站在梧桐树下&#xff0c;祖母抱着襁褓中的父亲站在石库门门口——可所有画面都是灰白的。我们看得清他们的轮廓、表情、衣褶&#xff0c…

作者头像 李华