'use strict'; import FileApi from '@/sheep/api/infra/file'; import CryptoJS from 'crypto-js'; const ERR_MSG_OK = 'chooseAndUploadFile:ok'; const ERR_MSG_FAIL = 'chooseAndUploadFile:fail'; function chooseImage(opts) { const { count, sizeType = ['original', 'compressed'], sourceType = ['album', 'camera'], extension, } = opts; return new Promise((resolve, reject) => { uni.chooseImage({ count, sizeType, sourceType, extension, success(res) { resolve(normalizeChooseAndUploadFileRes(res, 'image')); }, fail(res) { reject({ errMsg: res.errMsg.replace('chooseImage:fail', ERR_MSG_FAIL), }); }, }); }); } function chooseVideo(opts) { const { camera, compressed, maxDuration, sourceType = ['album', 'camera'], extension } = opts; return new Promise((resolve, reject) => { uni.chooseVideo({ camera, compressed, maxDuration, sourceType, extension, success(res) { const { tempFilePath, duration, size, height, width } = res; resolve( normalizeChooseAndUploadFileRes( { errMsg: 'chooseVideo:ok', tempFilePaths: [tempFilePath], tempFiles: [ { name: (res.tempFile && res.tempFile.name) || '', path: tempFilePath, size, type: (res.tempFile && res.tempFile.type) || '', width, height, duration, fileType: 'video', cloudPath: '', }, ], }, 'video', ), ); }, fail(res) { reject({ errMsg: res.errMsg.replace('chooseVideo:fail', ERR_MSG_FAIL), }); }, }); }); } function chooseAll(opts) { const { count, extension } = opts; return new Promise((resolve, reject) => { let chooseFile = uni.chooseFile; if (typeof wx !== 'undefined' && typeof wx.chooseMessageFile === 'function') { chooseFile = wx.chooseMessageFile; } if (typeof chooseFile !== 'function') { return reject({ errMsg: ERR_MSG_FAIL + ' 请指定 type 类型,该平台仅支持选择 image 或 video。', }); } chooseFile({ type: 'all', count, extension, success(res) { resolve(normalizeChooseAndUploadFileRes(res)); }, fail(res) { reject({ errMsg: res.errMsg.replace('chooseFile:fail', ERR_MSG_FAIL), }); }, }); }); } function normalizeChooseAndUploadFileRes(res, fileType) { res.tempFiles.forEach((item, index) => { if (!item.name) { item.name = item.path.substring(item.path.lastIndexOf('/') + 1); } if (fileType) { item.fileType = fileType; } item.cloudPath = Date.now() + '_' + index + item.name.substring(item.name.lastIndexOf('.')); }); if (!res.tempFilePaths) { res.tempFilePaths = res.tempFiles.map((file) => file.path); } return res; } function convertToArrayBuffer(uniFile) { return new Promise((resolve, reject) => { const fs = uni.getFileSystemManager(); fs.readFile({ filePath: uniFile.path, // 确保路径正确 success: (fileRes) => { try { // 将读取的内容转换为 ArrayBuffer const arrayBuffer = new Uint8Array(fileRes.data).buffer; resolve(arrayBuffer); } catch (error) { reject(new Error(`转换为 ArrayBuffer 失败: ${error.message}`)); } }, fail: (error) => { reject(new Error(`读取文件失败: ${error.errMsg}`)); }, }); }); } function uploadCloudFiles(files, max = 5, onUploadProgress) { files = JSON.parse(JSON.stringify(files)); const len = files.length; let count = 0; let self = this; return new Promise((resolve) => { while (count < max) { next(); } function next() { let cur = count++; if (cur >= len) { !files.find((item) => !item.url && !item.errMsg) && resolve(files); return; } const fileItem = files[cur]; const index = self.files.findIndex((v) => v.uuid === fileItem.uuid); fileItem.url = ''; delete fileItem.errMsg; uniCloud .uploadFile({ filePath: fileItem.path, cloudPath: fileItem.cloudPath, fileType: fileItem.fileType, onUploadProgress: (res) => { res.index = index; onUploadProgress && onUploadProgress(res); }, }) .then((res) => { fileItem.url = res.fileID; fileItem.index = index; if (cur < len) { next(); } }) .catch((res) => { fileItem.errMsg = res.errMsg || res.message; fileItem.index = index; if (cur < len) { next(); } }); } }); } async function uploadFiles(choosePromise, { onChooseFile, onUploadProgress }) { // 获取选择的文件 const res = await choosePromise; // 处理文件选择回调 let files = res.tempFiles || []; if (onChooseFile) { const customChooseRes = onChooseFile(res); if (typeof customChooseRes !== 'undefined') { files = await Promise.resolve(customChooseRes); if (typeof files === 'undefined') { files = res.tempFiles || []; // Fallback } } } // 如果是前端直连上传 if (UPLOAD_TYPE.CLIENT === import.meta.env.SHOPRO_UPLOAD_TYPE) { for (const file of files) { // 获取二进制文件对象 const fileBuffer = await convertToArrayBuffer(file); // 1.1 生成文件名称 const fileName = await generateFileName(fileBuffer, file.name); // 1.2 获取文件预签名地址 const { data: presignedInfo } = await FileApi.getFilePresignedUrl(file.name); // 1.3 上传文件 await uni.request({ url: presignedInfo.uploadUrl, // 预签名的上传 URL method: 'PUT', // 使用 PUT 方法 header: { 'Content-Type': file.fileType + '/' + file.name.substring(file.name.lastIndexOf('.') + 1), // 设置内容类型 }, data: fileBuffer, // 文件的路径,适用于小程序 success: (res) => { // 1.4. 记录文件信息到后端(异步) createFile(presignedInfo, fileName, file); // 1.5. 重新赋值 file.url = presignedInfo.url; file.name = fileName; console.log('上传成功:', res); }, fail: (err) => { console.error('上传失败:', err); }, }); } return files; } else { // 后端上传 for (let file of files) { const { data } = await FileApi.uploadFile(file.path); file.url = data; } return files; } } function chooseAndUploadFile( opts = { type: 'all', }, ) { if (opts.type === 'image') { return uploadFiles(chooseImage(opts), opts); } else if (opts.type === 'video') { return uploadFiles(chooseVideo(opts), opts); } return uploadFiles(chooseAll(opts), opts); } /** * 生成文件名称(使用算法SHA256) * @param arrayBuffer 二进制文件对象 * @param fileName 文件名称 */ async function generateFileName(arrayBuffer, fileName) { return new Promise((resolve, reject) => { try { // 创建 WordArray const wordArray = CryptoJS.lib.WordArray.create(new Uint8Array(arrayBuffer)); // 计算SHA256 const sha256 = CryptoJS.SHA256(wordArray).toString(); // 拼接后缀 const ext = fileName.substring(fileName.lastIndexOf('.')); resolve(`${sha256}${ext}`); } catch (error) { reject(new Error('计算SHA256失败: ' + error.message)); } }); } /** * 创建文件信息 * @param vo 文件预签名信息 * @param name 文件名称 * @param file 文件 */ function createFile(vo, name, file) { const fileVo = { configId: vo.configId, url: vo.url, path: name, name: file.name, type: file.fileType, size: file.size, }; FileApi.createFile(fileVo); return fileVo; } /** * 上传类型 */ const UPLOAD_TYPE = { // 客户端直接上传(只支持S3服务) CLIENT: 'client', // 客户端发送到后端上传 SERVER: 'server', }; export { chooseAndUploadFile, uploadCloudFiles };