vue3实现文件夹上传及存在问题解决

需求是使用 VUE3 实现上传文件夹,其功能主要依赖于 <input> 标签的属性 webkitdirectory

html 复制代码
<input
  ref="uploadFolderRef"
  type="file"
  style="display: none"
  @change="submitUpload"
  webkitdirectory
  multiple
/>
<el-button @click="openUploadDialog">上传文件夹</el-button>

将 input 标签的 display 属性设为 none 使其隐藏起来,后续通过点击按钮触发上传框的点击事件:

javascript 复制代码
import { ref } from 'vue'
import { ElMessage } from 'element-plus'

const uploadFolderRef = ref()

const openUploadDialog = () => {
  uploadFolderRef.value.click()
}

const submitUpload = (e: any) => {
  const files = Array.from(e.target.files)
  
  Promise.all(
    files.map((file: any) => {
      const form = new FormData()
      form.append('file', file)
      return UploadApi.upload(form)
    }),
  )
    .then(() => {
      ElMessage.success('上传成功')
    })
    .catch(() => {
      ElMessage.error('上传失败')
    })
}

存在问题:重复上传相同文件夹,只有第一次有效,后续选择文件夹无法完成上传。

原因 是上传框的提交上传是 onchange 事件,选择相同的文件夹的 value 属性值是一样的,因此无法触发;

解决方案

  1. 手动修改上传框的 value 属性为空;

    javascript 复制代码
    const uploadFolderRef = ref()
    /* 打开上传框之前,先将上传框的value属性设为空 */
    const openUploadDialog = () => {
      uploadFolderRef.value.value = ''
      uploadFolderRef.value.click()
    }
  2. 采用 JavaScript 生成上传框,这样将每次都生成新的上传框;

    html 复制代码
    <template>
        <el-button @click="createFileInput">上传文件夹</el-button>
    </template>
    
    <script setup lang="ts">
    const createFileInput = () => {
      const input = document.createElement('input')
      input.style.display = 'none'
      input.type = 'file'
      input.multiple = true
      input.webkitdirectory = true
      input.onchange = submitUpload
      document.body.appendChild(input)
      input.click()
    }
    </script>
相关推荐
橙序员小站2 小时前
Agent Skill 是什么?一文讲透 Agent Skill 的设计与实现
前端·后端
炫饭第一名4 小时前
速通Canvas指北🦮——基础入门篇
前端·javascript·程序员
王晓枫5 小时前
flutter接入三方库运行报错:Error running pod install
前端·flutter
符方昊5 小时前
React 19 对比 React 16 新特性解析
前端·react.js
ssshooter5 小时前
又被 Safari 差异坑了:textContent 拿到的值居然没换行?
前端
曲折5 小时前
Cesium-气象要素PNG色斑图叠加
前端·cesium
Forever7_5 小时前
Electron 淘汰!新的桌面端框架 更强大、更轻量化
前端·vue.js
不会敲代码15 小时前
前端组件化样式隔离实战:React CSS Modules、styled-components 与 Vue scoped 对比
css·vue.js·react.js
Angelial5 小时前
Vue3 嵌套路由 KeepAlive:动态缓存与反向配置方案
前端·vue.js
jiayu6 小时前
Angular学习笔记24:Angular 响应式表单 FormArray 与 FormGroup 相互嵌套
前端