2022-11-22 07:45:36 +00:00
|
|
|
|
'use strict';
|
2024-01-05 14:26:01 +00:00
|
|
|
|
import FileApi from '@/sheep/api/infra/file';
|
2024-09-24 09:16:38 +00:00
|
|
|
|
import CryptoJS from 'crypto-js';
|
2022-11-22 07:45:36 +00:00
|
|
|
|
|
|
|
|
|
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) {
|
2024-09-24 11:14:52 +00:00
|
|
|
|
item.fileType = fileType;
|
2022-11-22 07:45:36 +00:00
|
|
|
|
}
|
|
|
|
|
item.cloudPath = Date.now() + '_' + index + item.name.substring(item.name.lastIndexOf('.'));
|
|
|
|
|
});
|
|
|
|
|
if (!res.tempFilePaths) {
|
|
|
|
|
res.tempFilePaths = res.tempFiles.map((file) => file.path);
|
|
|
|
|
}
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-24 09:16:38 +00:00
|
|
|
|
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}`));
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-22 07:45:36 +00:00
|
|
|
|
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();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-24 09:16:38 +00:00
|
|
|
|
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
|
2022-11-22 07:45:36 +00:00
|
|
|
|
}
|
2024-09-24 09:16:38 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 如果是前端直连上传
|
|
|
|
|
if (UPLOAD_TYPE.CLIENT === import.meta.env.SHOPRO_UPLOAD_TYPE) {
|
|
|
|
|
for (const file of files) {
|
2024-09-25 00:25:20 +00:00
|
|
|
|
// 1.1 获取文件预签名地址
|
2024-09-24 09:16:38 +00:00
|
|
|
|
const { data: presignedInfo } = await FileApi.getFilePresignedUrl(file.name);
|
2024-09-25 00:25:20 +00:00
|
|
|
|
// 1.2 获取二进制文件对象
|
|
|
|
|
const fileBuffer = await convertToArrayBuffer(file);
|
2024-09-24 09:16:38 +00:00
|
|
|
|
// 1.3 上传文件
|
|
|
|
|
await uni.request({
|
|
|
|
|
url: presignedInfo.uploadUrl, // 预签名的上传 URL
|
|
|
|
|
method: 'PUT', // 使用 PUT 方法
|
|
|
|
|
header: {
|
2024-09-24 11:14:52 +00:00
|
|
|
|
'Content-Type': file.fileType + '/' + file.name.substring(file.name.lastIndexOf('.') + 1), // 设置内容类型
|
2024-09-24 09:16:38 +00:00
|
|
|
|
},
|
|
|
|
|
data: fileBuffer, // 文件的路径,适用于小程序
|
|
|
|
|
success: (res) => {
|
|
|
|
|
// 1.4. 记录文件信息到后端(异步)
|
2024-09-25 00:25:20 +00:00
|
|
|
|
createFile(presignedInfo, file);
|
2024-09-24 09:16:38 +00:00
|
|
|
|
// 1.5. 重新赋值
|
|
|
|
|
file.url = presignedInfo.url;
|
|
|
|
|
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;
|
|
|
|
|
}
|
2022-11-22 07:45:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-24 09:16:38 +00:00
|
|
|
|
/**
|
|
|
|
|
* 创建文件信息
|
|
|
|
|
* @param vo 文件预签名信息
|
|
|
|
|
* @param file 文件
|
|
|
|
|
*/
|
2024-09-25 00:25:20 +00:00
|
|
|
|
function createFile(vo, file) {
|
2024-09-24 09:16:38 +00:00
|
|
|
|
const fileVo = {
|
|
|
|
|
configId: vo.configId,
|
|
|
|
|
url: vo.url,
|
2024-09-25 00:25:20 +00:00
|
|
|
|
path: file.name,
|
2024-09-24 09:16:38 +00:00
|
|
|
|
name: file.name,
|
|
|
|
|
type: file.fileType,
|
|
|
|
|
size: file.size,
|
|
|
|
|
};
|
|
|
|
|
FileApi.createFile(fileVo);
|
|
|
|
|
return fileVo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 上传类型
|
|
|
|
|
*/
|
|
|
|
|
const UPLOAD_TYPE = {
|
|
|
|
|
// 客户端直接上传(只支持S3服务)
|
|
|
|
|
CLIENT: 'client',
|
|
|
|
|
// 客户端发送到后端上传
|
|
|
|
|
SERVER: 'server',
|
|
|
|
|
};
|
|
|
|
|
|
2022-11-22 07:45:36 +00:00
|
|
|
|
export { chooseAndUploadFile, uploadCloudFiles };
|