| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 | <template>	<view class="compress" v-if="canvasId">		<canvas :canvas-id="canvasId" :style="{ width: canvasSize.width,height: canvasSize.height}"></canvas>	</view></template><script>	export default {		data() {			return {				pic:'',				canvasSize: {					width: 0,					height: 0				},				canvasId:""			}		},		mounted() {			if(!uni || !uni._helang_compress_canvas){				uni._helang_compress_canvas = 1;			}else{				uni._helang_compress_canvas++;			}			this.canvasId = `compress-canvas${uni._helang_compress_canvas}`;		},		methods: {			// 压缩			compressFun(params) {				return new Promise(async (resolve, reject) => {					// 等待图片信息					let info = await this.getImageInfo(params.src).then(info=>info).catch(()=>null);										if(!info){						reject('获取图片信息异常');						return;					}										// 设置最大 & 最小 尺寸					const maxSize = params.maxSize || 1080;					const minSize = params.minSize || 640;										// 当前图片尺寸					let {width,height} = info;										// 非 H5 平台进行最小尺寸校验					// #ifndef H5					if(width <= minSize && height <= minSize){						resolve(params.src);						return;					}					// #endif										// 最大尺寸计算					if (width > maxSize || height > maxSize) {						if (width > height) {							height = Math.floor(height / (width / maxSize));							width = maxSize;						} else {							width = Math.floor(width / (height / maxSize));							height = maxSize;						}					}										// 设置画布尺寸					this.$set(this,"canvasSize",{						width: `${width}px`,						height: `${height}px`					});															// Vue.nextTick 回调在 App 有异常,则使用 setTimeout 等待DOM更新					setTimeout(() => {						const ctx = uni.createCanvasContext(this.canvasId, this);						ctx.clearRect(0,0,width, height)						ctx.drawImage(info.path, 0, 0, width, height);						ctx.draw(false, () => {							uni.canvasToTempFilePath({								x: 0,								y: 0,								width: width,								height: height,								destWidth: width,								destHeight: height,								canvasId: this.canvasId,								fileType: params.fileType || 'png',								quality: params.quality || 0.9,								success: (res) => {																		// 在H5平台下,tempFilePath 为 base64									resolve(res.tempFilePath);								},								fail:(err)=>{									reject(null);								}							},this);						});					}, 300);				});			},			// 获取图片信息			getImageInfo(src){				return new Promise((resolve, reject)=>{					uni.getImageInfo({						src,						success: (info)=> {							resolve(info);						},						fail: () => {							reject(null);						}					});				});			},			// 批量压缩			compress(params){				// index:进度,done:成功,fail:失败				let [index,done,fail] = [0,0,0];				// 压缩完成的路径集合				let paths = [];				// 待压缩的图片				let waitList = [];				if(typeof params.src == 'string'){					waitList = [params.src];				}else{					waitList = params.src;				}				// 批量压缩方法				let batch = ()=>{					return new Promise((resolve, reject)=>{						// 开始						let start = async ()=>{							// 等待图片压缩方法返回							let path = await next().catch(()=>null);							if(path){								done++;								paths.push(path);							}else{								fail++;							}														params.progress && params.progress({								done,								fail,								count:waitList.length							});														index++;							// 压缩完成							if(index >= waitList.length){								resolve(true);							}else{								start();							}						}						start();					});				}				// 依次调用压缩方法				let next = ()=>{					return this.compressFun({						src:waitList[index],						maxSize:params.maxSize,						fileType:params.fileType,						quality:params.quality,						minSize:params.minSize					})				}								// 全部压缩完成后调用				return new Promise(async (resolve, reject)=>{					// 批量压缩方法回调					let res = await batch();					if(res){						if(typeof params.src == 'string'){							resolve(paths[0]);						}else{							resolve(paths);						}					}else{						reject(null);					}				});			}		}	}</script><style lang="scss" scoped>	.compress{		position: fixed;		width: 12px;		height: 12px;		overflow: hidden;		top: -99999px;		left: 0;	}</style>
 |