文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Vue如何实现封装一个切片上传组件

2023-07-05 12:39

关注

今天小编给大家分享一下Vue如何实现封装一个切片上传组件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

组件效果

单文件切片上传

Vue如何实现封装一个切片上传组件

多文件切片上传

Vue如何实现封装一个切片上传组件

组件使用案例

<template>  <div id="app">    <div class="upload-wrap">      <UploadSlice        :action="uploadInfoSlice.actionChunk"        :headers="uploadInfoSlice.headers"        :limit="uploadInfoSlice.limit"        :accept="uploadInfoSlice.accept"        :show-file-list="false"        cancelable        :on-success="handleSuccess"        :on-remove="handleRemove"        :on-cancel="handleCancel"        :on-upload-pre="handleUploadPre"        :on-upload-merge="handleUploadMerge"        :on-form-data="genFormData"      />    </div>  </div></template><script>import UploadSlice from './components/UploadSlice.vue'import { uploadPre, uploadMerge } from '@/api/upload-slice'export default {  name: 'App',  components: {    UploadSlice  },  data() {    return {      // 上传部分      uploadInfoSlice: {        actionChunk: process.env.VUE_APP_BASE_API + '/storage/file/v3/chunk', // 切片请求上传路径        headers: { 'Authorization': 'Bearer XXX' }      }    }  },  methods: {    // 分片预请求    async handleUploadPre(file) {      const form = new FormData()      form.append('fileSource', 'APPLICATION')      form.append('originFileName', file.name)      let res = ''      try {        res = await uploadPre(form)      } catch (error) {        throw new Error(error)      }    },    // 构造分片参数    genFormData(chunks, uid) {      const prepareId = this.getCurrentPrepareId(uid)      return chunks.map(chunk => {        const form = new FormData()        form.append('chunk', chunk.file)        form.append('uploadId', prepareId)        form.append('partNumber', chunk.index)        return form      })    },    // 合并请求    async handleUploadMerge(file, uid) {      const prepareId = this.getCurrentPrepareId(uid)      const form = new FormData()      form.append('fileSource', 'APPLICATION')      form.append('hash', prepareId)      form.append('filename', file.name)      // return 建议使用, 用于handleSuccess获取数据      try {        const res = await uploadMerge(form)        return res      } catch (error) {        return error      }    },    // 判断当前处理prepareId    getCurrentPrepareId(uid) {      for (const item of this.progressFileList) {        if (item.uid === uid) {          return item.prepareId        }      }    }  }}</script>

使用文档

Attribute

标红色部分为二次封装处理过的功能,其他为el-upload自带属性

参数说明类型可选值默认值备注
action必选参数,分片上传的地址,预请求和合并请求在组件外操作String--
headers设置上传的请求头部String--
multiple是否支持多选文件boolean-

accept可上传文件类型,多种类型用","分隔 (格式不符合自动提示)String--
on-remove文件列表移除文件时的钩子function(file, fileList)&mdash;&mdash;
on-success文件上传成功时的钩子function(response, file, fileList)&mdash;&mdash;
on-error文件上传失败时的钩子function(err, file, fileList)&mdash;&mdash;
on-progress文件上传时的钩子function(event, file, fileList)&mdash;&mdash;
on-change文件状态改变时的钩子,添加文件、上传成功和上传失败时都会被调用function(file, fileList)&mdash;&mdash;
on-exceed文件超出个数限制时的钩子function(files, fileList)&mdash;&mdash;
list-type文件列表的类型stringtext/picture/picture-cardtext
show-file-list是否显示已上传文件列表(文件分片上传时建议设置false,否则会有两个进度条)boolean&mdash;true
file-list上传的文件列表, 例如: [{name: 'food.jpg', url: 'xxx.cdn.com/xxx.jpg'}]array&mdash;[]
disabled是否禁用boolean&mdash;false
cancelable是否支持取消boolean&mdash;false
limit最多允许上传个数(超出个数自动提示)number&mdash;&mdash;
size限制大小String&mdash;&mdash;
hideBtn是否在上传过程中隐藏上传按钮boolean&mdash;&mdash;false

Slot

插槽名说明
trigger触发文件选择框的内容
tip提示说明文字
more-tips在默认提示后补充说明

封装过程

切片上传组件是基于el-upload进行的二次封装,文章开头组件效果演示可以看到上传一个文件会发送三个请求:prepare,chunk, merge,也就是整个上传过程,主要分为三步:1.预请求 2.分片请求 3.合并请求,预请求和合并请求就是我们正常的http请求,主要处理的是分片请求,分片请求主要的步骤是:

1. 文件切片

el-upload上传后, 在on-change属性的回调里可以获取文件file,通过file.raw.slice对文件进行切片,目前的切片规则是:1.小于10M 固定一片 2.小于50M 文件10%为一片 3.大于50M 固定5M 一片(可以根据自己的需求进行修改)

genFileChunks(file) {  const chunks = []  let cur = 0  // 小于10M 固定一片  if (file.size < (10 * 1024 * 1024)) {    chunks.push({      index: cur,      file: file.raw.slice(cur, file.size),      originFilename: file.name    })    return chunks  }  // 小于50M 文件10%为一片  if (file.size < (50 * 1024 * 1024)) {    const chunkSize = parseInt(file.size * 0.1)    while (cur < file.size) {      chunks.push({        index: cur,        file: file.raw.slice(cur, cur + chunkSize),        originFilename: file.name      })      cur += chunkSize    }    return chunks  }  // 大于50M 固定5M 一片  const chunkSize = parseInt(5 * 1024 * 1024)  while (cur < file.size) {    chunks.push({      index: cur,      file: file.raw.slice(cur, cur + chunkSize),      originFilename: file.name    })    cur += chunkSize  }  return chunks},

一个32M的文件按照10%切一片,构造好的切片数据是这样的

Vue如何实现封装一个切片上传组件

2. 构造切片请求参数

切片请求不同业务的参数是变化的,所以参数部分可以抛出给父组件处理,增加组件的复用性

父组件

<template>  <UploadSlice    :action="uploadInfoSlice.actionChunk"    :headers="uploadInfoSlice.headers"    :on-form-data="genFormData"  /></template><script>  methods: {    // 构造分片参数    genFormData(chunks, uid) {      const prepareId = this.getCurrentPrepareId(uid)      return chunks.map(chunk => {        const form = new FormData()        form.append('chunk', chunk.file)        form.append('uploadId', prepareId)        form.append('partNumber', chunk.index)        return form      })    },  },</script>

子组件

<template>  <el-upload      action=""      :accept="accept"  ></template><script>  props: {    onFormData: {      type: Function,      default: () => {}    },  },  methods: {    async uploadChunks(uid) {      // 预请求      // ---------------      // 上传切片      const requests = this._genRequest(this._genUploadData(uid), uid)      // 控制并发      await this.sendRequest(requests)            // 合并请求      // ---------------    },        // 构造分片参数    _genUploadData(uid) {      const chunks = this.getCurrentChunks(uid)      return this.onFormData(chunks, uid)    },        // 生成调用请求:[Promise, Promise]    _genRequest(uploadData, uid) {      console.log('uploadData', uploadData)      const file = this.getCurrentFile(uid)      const chunks = this.getCurrentChunks(uid)      return uploadData.map((form, index) => {        const options = {          headers: this.$attrs.headers,          file: file,          data: form,          action: this.action,          onProgress: progress => {            chunks[index].progress = Number(              ((progress.loaded / progress.total) * 100).toFixed(2)            )            this.handleProgress(progress, file, uid)          }        }        return options      })    },  },</script>

3. 控制分片请求的并发

切片上传如果不控制并发,在分片很多时,就会同时发送很多个http请求,导致线程阻塞,影响页面其他请求的操作,所以控制并发是需要的。我设置的是最多允许3个并发请求。

    sendRequest(requests, limit = 3) {      return new Promise((resolve, reject) => {        const len = requests.length        let counter = 0        let isTips = false // 只提示一次失败        let isStop = false // 如果一个片段失败超过三次 认为当前网洛有问题 停止全部上传        const startRequest = async() => {          if (isStop) return          const task = requests.shift()          if (task && task.file.status !== 'cancel') {            // 利用try...catch捕获错误            try {              // 具体的接口  抽离出去了              await ajax(task)              if (counter === len - 1) { // 最后一个任务                resolve()              } else { // 否则接着执行                counter++                startRequest() // 启动下一个任务              }            } catch (error) {              // 网络异常              if (error === 'NETWORK_ERROR' && !isTips) {                Message.error('网络异常,文件上传失败')                this.upLoading = false                this.preLoading = false                isTips = true                this.handleRemove('', [])              }              // 接口报错重试,限制为3次              if (task.error < 3) {                task.error++                requests.unshift(task)                startRequest()              } else {                isStop = true                reject(error)              }            }          }        }        // 启动任务        while (limit > 0) {          // 模拟不同大小启动          setTimeout(() => {            startRequest()          }, Math.random() * 2000)          limit--        }      })    }  }

以上就是“Vue如何实现封装一个切片上传组件”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯