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
}

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

Logo

新一代开源开发者平台 GitCode,通过集成代码托管服务、代码仓库以及可信赖的开源组件库,让开发者可以在云端进行代码托管和开发。旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐