uniapp 多图上传,加水印功能(全平台通用)

多图上传和水印都是比较难得,特别是有的api只支持在小程序用,h5不给用

效果图
模拟插件
没加水印

加水印

普通的多图上传

// 多图上传
// count:最大数量
export function headerUploads0(count = 9, orderNumber = '', watermarkInfo) {return new Promise((resolve, reject) => {let UploadList = []uni.chooseImage({// 最多可以选择的图片总数count,success: async res => {let processedFiles = res.tempFilePathsif (watermarkInfo) {// 创建列式水印文本数组const watermarkLines = [`订单号: ${watermarkInfo.number}`,`买  家: ${watermarkInfo.traderName}`,`卖  家: ${watermarkInfo.sellerTraderName}`,`资金方: ${watermarkInfo.capitalTraderName}`,`时  间: ${format(new Date(), false)}`];// 批量添加水印}//启动上传等待中...  uni.showLoading({title: '上传中',});await Promise.all(processedFiles.map((item, index) => {return new Promise((resolve1, reject1) => {console.log('上传');uni.uploadFile({// 上传地址url: BASE_URL + VUE_APP_BASE_API +'/AppBusiness/AppCommon/UploadFile',name: 'file',filePath: processedFiles[index],formData: {FileNameType: 3,FileDir: orderNumber || '',ClassifyType: orderNumber ? 'order' : ''},header: {"Authorization": (store.state.token ||uni.getStorageSync('token')) ?'Bearer ' + (store.state.token ||uni.getStorageSync('token')) :''},success: (resz) => {console.log('后端返回', (JSON.parse(resz.data).data));uni.hideLoading()UploadList.push(JSON.parse(resz.data).data)resolve1()},fail: (resz) => {console.log('失败返回', resz);uni.hideLoading()reject1()}})});})).then(() => {console.log('循环后', UploadList);resolve(UploadList)}).catch((error) => {console.log('循环后', UploadList);resolve(UploadList)})},complete: compRes => {}});})
}

带水印的多图上传

在组件页面那需要加个空画布用来操作

// vue页面上
<!-- 隐藏的Canvas,用于绘制水印(全端兼容)-->
<canvas canvas-id="watermarkCanvas":style="{position: 'absolute', top: '0', left: '-1000vw', width: canvasWidth + 'px', height: canvasHeight + 'px'}"></canvas>// 引入下面的函数tools.js是我的封装js,别搞错了
import {headerUploads, // 多图上传
} from "@/utils/tools.js"// data数据中
canvasWidth: 500, // 动态绑定Canvas宽高
canvasHeight: 500// js中调用 
// 参数 多少张图片3张,携带的订单号(不重要我这边上传后端要) Y12345, 订单信息 this.watermarkInfo, this 用来设置canvasWidth的
headerUploads(3, 'Y12345', this.watermarkInfo, this)

函数–封装的tools.js中


/*** 跨平台图片压缩(兼容小程序和H5)* @param {string} filePath 原始图片路径* @param {number} [quality=0.7] 压缩质量(0-1)* @param {number} [maxWidth=1024] 最大宽度* @param {number} [maxHeight=1024] 最大高度* @return {Promise<string>} 压缩后的图片路径*/
export function compressImage(filePath, quality = 0.8, maxWidth = 1024, maxHeight = 1024) {return new Promise((resolve, reject) => {// 平台判断// #ifdef MP-WEIXIN// 微信小程序使用官方APIwx.compressImage({src: filePath,quality: Math.floor(quality * 100), // 微信使用0-100整数success: (res) => resolve(res.tempFilePath),fail: reject});// #endif// #ifdef H5// H5使用Canvas压缩const img = new Image();img.crossOrigin = 'anonymous';img.src = filePath;img.onload = () => {const canvas = document.createElement('canvas');const ctx = canvas.getContext('2d');// 计算压缩尺寸let width = img.width;let height = img.height;if (width > maxWidth) {height *= maxWidth / width;width = maxWidth;}if (height > maxHeight) {width *= maxHeight / height;height = maxHeight;}canvas.width = width;canvas.height = height;// 绘制压缩图片ctx.drawImage(img, 0, 0, width, height);console.log(666);// 获取压缩结果canvas.toBlob((blob) => {const reader = new FileReader();reader.onload = () => resolve(reader.result);reader.readAsDataURL(blob);},'image/jpeg',quality);};img.onerror = reject;// #endif// 其他平台(如App)或未处理平台返回原图// #ifndef MP-WEIXIN, H5resolve(filePath);// #endif});
}// 多图上传(带水印功能)
export function headerUploads(count = 9, orderNumber = '', watermarkInfo, pageContext) {return new Promise((resolve, reject) => {uni.chooseImage({count,success: async res => {try {const tempFiles = res.tempFilePaths;let processedFiles = tempFiles;// 如果需要加水印if (watermarkInfo && watermarkInfo.number) {const watermarkLines = [`时  间: ${format(new Date(), false)}`,`资金方: ${watermarkInfo.capitalTraderName || ''}`,`卖  家: ${watermarkInfo.sellerTraderName || ''}`,`买  家: ${watermarkInfo.traderName || ''}`,`订单号: ${watermarkInfo.number}`];// 串行处理水印(确保画布状态独立)processedFiles = [];for (const imgPath of tempFiles) {try {const watermarked = await addWatermarkByContext(imgPath, watermarkLines, pageContext);processedFiles.push(watermarked);} catch (e) {console.error('水印处理失败,使用原图:', e);processedFiles.push(imgPath); // 失败时使用原图}}}// 上传所有处理后的文件const UploadList = await uploadAllFiles(processedFiles, orderNumber);resolve(UploadList);} catch (e) {reject(e);}},fail: reject});});
}// 重置画布状态(关键解决画布污染问题)
function resetCanvasContext(pageContext) {if (!pageContext) return;// 重置画布尺寸(避免上一张图片的尺寸影响)pageContext.canvasWidth = 0;pageContext.canvasHeight = 0;// 清除画布内容const ctx = uni.createCanvasContext('watermarkCanvas', pageContext);ctx.clearRect(0, 0, pageContext.canvasWidth || 500, pageContext.canvasHeight || 500); // 清除超大区域确保干净ctx.draw(); // 立即执行清除
}// 具体的加水印
function addWatermarkByContext(imgPath, watermarkLines, pageContext) {// 无需水印直接返回原图if (!watermarkLines || watermarkLines.length === 0 || !pageContext) {return Promise.resolve(imgPath);}return new Promise((resolve, reject) => {// 1. 先重置画布状态(关键步骤)resetCanvasContext(pageContext);// 2. 获取原图信息uni.getImageInfo({src: imgPath,success: (imgInfo) => {const { width: imgWidth, height: imgHeight } = imgInfo;const dpr = 1// uni.getSystemInfoSync().pixelRatio;let dpr2 = uni.getSystemInfoSync().pixelRatio// 3. 设置画布尺寸(使用原图尺寸)pageContext.canvasWidth = imgWidth * dpr;pageContext.canvasHeight = imgHeight * dpr;const maxCanvasSize = 4096; // 大多数设备的限制if (imgWidth * dpr > maxCanvasSize || imgHeight * dpr > maxCanvasSize) {const scale = Math.min(maxCanvasSize/(imgWidth*dpr), maxCanvasSize/(imgHeight*dpr));pageContext.canvasWidth = imgWidth * dpr * scale;pageContext.canvasHeight = imgHeight * dpr * scale;}// 4. 创建画布上下文const ctx = uni.createCanvasContext('watermarkCanvas', pageContext);ctx.scale(dpr, dpr);// 5. 绘制原图ctx.drawImage(imgPath, 0, 0, imgWidth, imgHeight);// 6. 绘制水印const fontSize = 28 * dpr2;const lineHeight = fontSize + 8;const margin = 20 * dpr2;ctx.setFontSize(fontSize);ctx.setFillStyle('#3975e2');ctx.setTextAlign('left');ctx.setTextBaseline('bottom');console.log(777);watermarkLines.forEach((line, index) => {const y = imgHeight - margin - (index * lineHeight);ctx.fillText(line, margin, y);});// 7. 延迟确保绘制完成setTimeout(() => {ctx.draw(false, () => {uni.canvasToTempFilePath({canvasId: 'watermarkCanvas',destWidth: imgWidth,destHeight: imgHeight,fileType: 'jpg',quality: 0.9,success: (res) => {// 8. 再次重置画布(为下一张图准备)resetCanvasContext(pageContext);resolve(res.tempFilePath);},fail: (err) => {resetCanvasContext(pageContext);reject(new Error(`Canvas转图片失败: ${JSON.stringify(err)}`));}}, pageContext);});}, 300); // 适当延迟确保绘制完成},fail: (err) => {reject(new Error(`获取图片信息失败: ${JSON.stringify(err)}`));}});});
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/910506.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/910506.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【appium】5. Appium WebDriver 支持的常用方法汇总

下面是一个完整的 Appium WebDriver 支持的常用方法汇总&#xff0c;并附上典型用法示例。 一、元素查找方法/元素操作方法 ✅ 使用 find_element() 和 find_elements() from appium import webdriver from appium.webdriver.common.appiumby import AppiumBy# 单个元素查找 …

FreeRTOS 介绍、使用方法及应用场景

一、FreeRTOS 概述 FreeRTOS 是一款广泛应用于嵌入式系统的实时操作系统&#xff08;RTOS&#xff09;&#xff0c;具有开源、可移植、可裁剪、轻量级等显著特点。它最初由 Richard Barry 开发&#xff0c;如今已成为全球开发者在物联网、工业控制、消费电子等领域的热门选择&a…

深度解析 Caffeine:高性能 Java 缓存库

1. Caffeine 简介 Caffeine 是一个基于 Java 8 的高性能本地缓存库&#xff0c;由 Ben Manes 开发&#xff0c;旨在替代 Google Guava Cache&#xff0c;提供更优的缓存策略、更高的吞吐量和更灵活的配置。 核心优势 ✅ 卓越的性能&#xff1a;采用优化的数据结构&#xff0…

创客匠人赋能创始人 IP 打造:健康行业知识变现案例深度解析

在知识服务行业蓬勃发展的当下&#xff0c;创始人 IP 打造已成为知识变现的核心驱动力。创客匠人近期披露的陪跑案例显示&#xff0c;通过系统化的线上线下联动运营&#xff0c;传统行业从业者可高效实现 IP 价值转化。以亓黄中医科技创始人吴丰言老师为例&#xff0c;其在创客…

64、最小路径和

题目&#xff1a; 解答&#xff1a; 简单dp。 定义&#xff1a;dp[i][j]为到达(i,j)所需要的最短路程 初始化&#xff1a;dp[0][0]grid[0][0]&#xff0c;同时对第一行和第一列的&#xff0c;第i个就是前i个之和加上自身 递归&#xff1a;dp[i][j]min(dp[i-1][j],dp[i][j-1…

获取连接通义千问大语言模型配置信息的步骤:api_key、api_url

一、注册并开通通义千问API服务 1. 注册阿里云账号 访问 阿里云官网点击右上角"免费注册"&#xff0c;按指引完成账号注册和实名认证 2. 开通通义千问API服务 进入 通义千问API产品页点击"立即开通"&#xff0c;按提示完成服务开通&#xff08;部分服务…

汽车加气站操作工考试题库含答案【最新】

1.天然气的主要成分是&#xff08;&#xff09;。 A. 乙烷 B. 乙烯 C. 甲烷 D. 乙炔 答案&#xff1a;C 2.CNG 加气站中&#xff0c;加气机的加气软管应&#xff08;&#xff09;进行检查。 A. 每天 B. 每周 C. 每月 D. 每季度 答案&#xff1a;A 3.储气罐的安全阀应&#xf…

显示任何结构的数组对象数据【向上自动滚动】

显示任何结构的数组对象数据 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>地图编辑软件 - 数…

GPIO模式详解

一、GPIO的八种模式 GPIO支持4种输入模式&#xff08;浮空输入、上拉输入、下拉输入、模拟输入&#xff09;和4种输出模式&#xff08;开漏输出、开漏复用输出、推挽输出、推挽复用输出&#xff09;。 GPIO_Mode_AIN模拟输入GPIO_Mode_IN_FLOATING浮空输入GPIO_Mode_IPD下拉输…

django rest_framework 自定义403 Forbidden错误页面

django本来有是可以很方便自定义HTTP错误页面的&#xff0c;网上资料一大把。核心是在项目的urls代码中增加handler403的定义&#xff0c;比如&#xff1a; handler403 "app.views.your_custom_view" 404&#xff0c;500都是一样的&#xff0c;重新定义handler404…

Kafka Streams架构深度解析:从并行处理到容错机制的全链路实践

在流处理技术领域&#xff0c;Kafka Streams以其轻量级架构与Kafka生态的深度整合能力脱颖而出。作为构建在Kafka生产者/消费者库之上的流处理框架&#xff0c;它通过利用Kafka原生的分区、副本与协调机制&#xff0c;实现了数据并行处理、分布式协调与容错能力的无缝集成。本文…

【嵌入式硬件实例】-555定时器控制舵机/伺服电机

555定时器控制舵机/伺服电机 文章目录 555定时器控制舵机/伺服电机1、555定时器介绍2、舵机/伺服电机介绍3、硬件准备与接线使用 555 定时器 IC 的伺服电机控制器和测试仪电路是一个简单的电路,可用于生成操作伺服电机所需的控制信号。该电路允许我们通过按下按钮手动驱动/控制…

国产麒麟 安装可视化数据库软件DBeaver(图解)

目录 ​​​​​​​​编辑DBeaver介绍 官网 通过强制使用 Ubuntu 模板来修复 add-apt-repository 重新添加 PPA 撤销更改&#xff08;可选&#xff09; 官网直接下载 DBeaver CE 下载好后安装软件 启动方式一 启动方式二 启动成功 在左侧右击新建连接 安装驱动 测…

线程池 JMM 内存模型

线程池 & JMM 内存模型 文章目录 线程池 & JMM 内存模型线程池线程池的创建ThreadPoolExecutor 七大参数饱和策略ExecutorService 提交线程任务对象执行的方法&#xff1a;ExecutorService 关闭线程池的方法&#xff1a;线程池最大线程数如何确定&#xff1f; volatile…

[论文阅读] 软件工程 + 教学 | 软件工程项目管理课程改革:从传统教学到以学生为中心的混合式学习实践

软件工程项目管理课程改革&#xff1a;从传统教学到以学生为中心的混合式学习实践 论文信息 arXiv:2506.14369 Agile and Student-Centred Teaching of Agile/Scrum Concepts Maria Spichkova Comments: Preprint. Accepted to the 29th International Conference on Knowledg…

Windows系统提示“mfc140u.dll丢失”?详细修复指南,一键恢复程序运行!

当你兴致勃勃地打开某个游戏或专业软件时&#xff0c;突然弹出一条错误提示——“MFC140u.dll丢失”&#xff0c;程序直接闪退&#xff0c;让人无比沮丧。别担心&#xff01;这个问题并不复杂&#xff0c;通常只需重新安装运行库或修复系统文件即可解决。本文将为你提供详细的修…

云XR(AR/VR)算力底座关键特征与技术路径

云XR&#xff08;AR/VR&#xff09;算力底座是支撑扩展现实技术规模化落地的核心基础设施&#xff0c;当前发展呈现以下关键特征与技术路径&#xff1a; 一、算力架构&#xff1a;云边端协同异构融合 分布式部署模式‌ 云端‌&#xff1a;承担高复杂度渲染与大数据处理&#x…

Android开发常用adb合集

Android开发常用adb合集 Android开发常用adb合集crash日志导出 Android开发常用adb合集 crash日志导出 bugreport: adb bugreportdropbox: adb shell dumpsys dropbox --print > desktop/full_dropbox_logs.txt

LTspice仿真4——exp指数函数波形

参数设置 Vinitial&#xff1a;初始电压值 Vpulsed&#xff1a;脉冲达到值 Rise Delay&#xff1a;上升延迟时间 Rise Tau&#xff1a;上升指数系数tau Fall Delay&#xff1a;下降延迟时间 Fall Tau&#xff1a;下降指数系数tau tau决定指数波形下降或者上升快慢&#x…

[Java 基础]集合框架

在 Java 中&#xff0c;我们经常需要存储和操作一组数据&#xff0c;而集合框架就是为此而生。它提供了一套统一的接口和类&#xff0c;帮助我们高效地管理各种数据集合。 常用的集合框架中的类只有 ArrayList、LinkedList、HashSet、HashMap 这 4 个&#xff0c;这些类的继承…