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

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

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

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

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 属性为空;

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

    <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>
    
  • 4
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Vue实现上传文件夹需要用到HTML5的File API,具体步骤如下: 1. 在模板中添加一个文件上传input标签,并设置multiple和webkitdirectory属性,让用户能够上传文件夹: ```html <template> <div> <input type="file" ref="fileInput" multiple webkitdirectory @change="handleFolderUpload"> </div> </template> ``` 2. 在Vue实例中编写handleFolderUpload方法,通过FileReader API读取文件夹中的所有文件,并将它们添加到一个FormData对象中: ```javascript export default { methods: { handleFolderUpload() { const files = this.$refs.fileInput.files; const formData = new FormData(); const reader = new FileReader(); const folderName = files[0].webkitRelativePath.split('/')[0]; // 文件夹的名称 // 读取文件夹中的所有文件 for (let i = 0; i < files.length; i++) { const file = files[i]; const fileName = file.webkitRelativePath.split('/').pop(); // 文件名称 reader.readAsArrayBuffer(file); reader.onload = e => { const arrayBuffer = e.target.result; const blob = new Blob([arrayBuffer]); formData.append(`${folderName}/${fileName}`, blob, fileName); }; } // 上传FormData对象 axios.post('/upload', formData).then(response => { console.log(response.data); }); } } } ``` 3. 在后端代码中处理文件上传请求,将FormData对象中的所有文件保存到服务器上。 需要注意的是,由于上传文件夹操作可能会上传大量的文件,因此可能会导致上传时间较长或者服务器压力较大,在实际使用中需要进行相关的优化和限制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

风於尘

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值