uni-app 同时上传多个文件

uni-app 同时上传多个文件

最近在用uni-app写项目的时候 有业务需要同时上传多张图片给后台,但查了官方文档,发现uni.uploadFile() 虽然可以通过 files 参数上传多个文件,但是只支持APP和H5。

在这里插入图片描述
我的项目主要针对的是微信小程序,所以看了很多文章,找到如下两种方式,可以参考一下

方案一:使用 uni.uploadFile() 轮询上传

使用 uni.chooseMedia 成功后会返回 本地临时文件列表 可以参考官方文档 uni.chooseMedia()

uni.chooseMedia({
  mediaType: ['image'],
  sourceType: ['album'],
  success: (res) => {
    const { tempFiles } = res
    	// uploadFile 方法为自己封装的上传方法
    	uploadFile(tempFiles)
    },
  fail: () => {
      console.log('fail')
    },
  complete: () => {
      // console.log('complete')
    },
  })

成功后调用自己写的 uploadFile

const uploadFile = (files:any[]) => {
  const uploadTasks = files.map((file: any, index: number) => {
     return new Promise((resolve, reject) => {
     	const uploadTask = uni.uploadFile({
        url: 'https://www.xxx.cn/v1/wxInvoice/upload', // 接口地址
        filePath: file.tempFilePath, // 临时文件路径
        name: 'files[]', // 服务器接收的文件字段名(这个地方很重要要和后端沟通一下)
        header: {
        	Authorization: 'Bearer ' + token.value,
          	'Content-Type': 'multipart/form-data',
        },
        formData: {
          // 可以在这里添加额外的formData参数
         },
        success: function (res) {
           resolve(res.data)
         },
         fail: function (err) {
         	reject(err)
         },
     })
	  
	 // 这里可以根据需要显示进度条
     uploadTask.onProgressUpdate((res) => {
     console.log('上传进度', res.progress)
     console.log('已经上传的数据长度', res.totalBytesSent)
     console.log('预期需要上传的数据总长度', res.totalBytesExpectedToSend)

    })
  })
})

Promise.all(uploadTasks)
     .then((res) => {
      console.log('上传成功', res)
       // 上传成功后的操作
     })
     .catch((err) => {
       console.log('上传失败', err)
      // 上传失败后的操作
     })
}

方案二

使用 wx-formdata 库,当然我这里是uni-app + ts 项目,所以我把这个库里的 formData.jsmimeMap.js 库进行了修改,文章结尾看(不要介意代码质量,能跑就行)

使用

uni.chooseMedia({
  mediaType: ['image'],
  sourceType: ['album'],
  success: (res) => {
    const { tempFiles } = res
    	
    	const formData = new FormData()
		
		// 非文件用 append()方法
		// 文件用 appendFile()方法
  		// 需要遍历将文件加入到formData中(这里是我的需求,可自定义)
  		for (let i = 0; i < tempFiles .length; i++) {
    		formData.appendFile('files[' + i + ']', tempFiles [i].tempFilePath)
  		}
  		const data = formData.getData()
		
		// 调用接口传递数据
		uni.request({
			url: '接口地址',
    		method: 'post',
    		data: data.buffer,
    		header: { 'Content-Type': data.contentType },
    		success:() => {
    			//...
    		}
		})
    },
  fail: () => {
      console.log('fail')
    },
  complete: () => {
      // console.log('complete')
    },
})

以下是修改过的文件
formData.ts

import mimeMap from './mimeMap'

interface CommonFormData {
  fileManager: any
  data: any
  files: any

  append(name: string, value: any): boolean
  appendFile(name: string, path: any, fileName?: any): boolean
  getData(): any
}

class FormData implements CommonFormData {
  fileManager: any
  data: any
  files: any

  constructor() {
    this.fileManager = uni.getFileSystemManager()
    this.data = {}
    this.files = []
  }

  append(name: string, value: any) {
    this.data[name] = value
    return true
  }

  appendFile(name: string, path: any, fileName?: any) {
    let buffer = this.fileManager.readFileSync(path)
    if (Object.prototype.toString.call(buffer).indexOf('ArrayBuffer') < 0) {
      return false
    }
    if (!fileName) {
      fileName = getFileNameFromPath(path)
    }
    this.files.push({
      name: name,
      buffer: buffer,
      fileName: fileName,
    })
    return true
  }

  getData() {
    return convert(this.data, this.files)
  }
}

function getFileNameFromPath(path: any) {
  let idx = path.lastIndexOf('/')
  return path.substr(idx + 1)
}

function convert(data: any, files: any) {
  let boundaryKey = 'unimpFormBoundary' + randString() // 数据分割符,一般是随机的字符串
  let boundary = '--' + boundaryKey
  let endBoundary = boundary + '--'

  let postArray: any = []
  //拼接参数
  if (data && Object.prototype.toString.call(data) == '[object Object]') {
    for (let key in data) {
      postArray = postArray.concat(formDataArray(boundary, key, data[key]))
    }
  }
  //拼接文件
  if (files && Object.prototype.toString.call(files) == '[object Array]') {
    for (let i in files) {
      let file = files[i]
      postArray = postArray.concat(formDataArray(boundary, file.name, file.buffer, file.fileName))
    }
  }

  //结尾
  let endBoundaryArray = []
  endBoundaryArray.push(...endBoundary.toUtf8Bytes())
  postArray = postArray.concat(endBoundaryArray)

  return {
    contentType: 'multipart/form-data; boundary=' + boundaryKey,
    buffer: new Uint8Array(postArray).buffer,
  }
}

// 获取随机字符串
function randString() {
  var result = ''
  var chars = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
  for (var i = 17; i > 0; --i) result += chars[Math.floor(Math.random() * chars.length)]
  return result
}

function formDataArray(boundary: any, name: string, value: any, fileName?: string) {
  let dataString: any = ''
  let isFile = !!fileName

  dataString += boundary + '\r\n'
  dataString += 'Content-Disposition: form-data; name="' + name + '"'
  if (isFile) {
    dataString += '; filename="' + fileName + '"' + '\r\n'
    dataString += 'Content-Type: ' + getFileMime(fileName || '') + '\r\n\r\n'
  } else {
    dataString += '\r\n\r\n'
    dataString += value
  }

  var dataArray = []
  dataArray.push(...dataString.toUtf8Bytes())

  if (isFile) {
    let fileArray = new Uint8Array(value)
    dataArray = dataArray.concat(Array.prototype.slice.call(fileArray))
  }
  dataArray.push(...'\r'.toUtf8Bytes())
  dataArray.push(...'\n'.toUtf8Bytes())

  return dataArray
}

function getFileMime(fileName: string) {
  let idx = fileName.lastIndexOf('.')
  let index: string = fileName.substr(idx)
  let mime = mimeMap[index]
  return mime ? mime : 'application/octet-stream'
}

function stringToUtf8(string: string) {
  let encoder = new TextEncoder()
  return encoder.encode(string)
}

String.prototype.toUtf8Bytes = function () {
  var str: any = this
  var bytes = []
  for (var i = 0; i < str.length; i++) {
    bytes.push(...str.utf8CodeAt(i))
    if (str.codePointAt(i) > 0xffff) {
      i++
    }
  }
  return bytes
}

String.prototype.utf8CodeAt = function (i: any) {
  var str = this
  var out = [],
    p = 0
  var c = str.charCodeAt(i)
  if (c < 128) {
    out[p++] = c
  } else if (c < 2048) {
    out[p++] = (c >> 6) | 192
    out[p++] = (c & 63) | 128
  } else if ((c & 0xfc00) == 0xd800 && i + 1 < str.length && (str.charCodeAt(i + 1) & 0xfc00) == 0xdc00) {
    // Surrogate Pair
    c = 0x10000 + ((c & 0x03ff) << 10) + (str.charCodeAt(++i) & 0x03ff)
    out[p++] = (c >> 18) | 240
    out[p++] = ((c >> 12) & 63) | 128
    out[p++] = ((c >> 6) & 63) | 128
    out[p++] = (c & 63) | 128
  } else {
    out[p++] = (c >> 12) | 224
    out[p++] = ((c >> 6) & 63) | 128
    out[p++] = (c & 63) | 128
  }
  return out
}

export default FormData

mimeMap.ts

const mimeMap:{[key:string]:any = {
	// 此处省略...
	// 把一些重复的处理一下
}

formData.d.ts

interface String {
  toUtf8Bytes(): any
  utf8CodeAt(i:any): any
}

以上就是我的分享,有问题留言

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
uni-app中,你可以使用uni.uploadFile方法来上传多个文件。首先,你需要将需要上传文件逐个进行遍历,并通过uni.uploadFile方法进行上传。 以下是一个示例代码: ```javascript // 定义一个数组,用于存储上传成功的文件路径 let uploadedFiles = []; // 遍历需要上传文件列表 for (let i = 0; i < fileList.length; i++) { let file = fileList[i]; // 使用uni.uploadFile方法进行文件上传 uni.uploadFile({ url: 'your_upload_url', // 替换为实际的上传接口地址 filePath: file.path, // 文件路径 name: 'file', // 对应后端接口中接收文件的字段名 formData: { // 其他参数 }, success: (res) => { // 上传成功,返回的数据在res.data中,可以根据需要进行处理 uploadedFiles.push(res.data); // 如果所有文件上传完成 if (uploadedFiles.length === fileList.length) { // 所有文件上传完成后的操作 console.log('所有文件上传完成'); console.log(uploadedFiles); } }, fail: (err) => { // 上传失败的处理逻辑 console.log('文件上传失败'); console.log(err); } }); } ``` 上述代码中,fileList是一个包含需要上传文件信息的数组,可以根据具体需求进行修改。在每次上传成功后,将返回的文件路径存储到uploadedFiles数组中。当所有文件上传完成后,可以在相应的位置进行处理。 请注意替换代码中的'your_upload_url'为实际的上传接口地址,并根据后端接口要求修改其他参数。 希望对你有所帮助!如果你还有其他问题,请继续提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值