Vue 图片压缩并上传至服务器功能
本文主要讲解基于 Vue + Vant ,实现移动端图片选择,并用 Canvas 压缩图片,上传至服务器。还会封装一个工具类,方便直接调用。
一、工具类封装
废话不多说先上代码,封装一个 CompressImageUtils 工具类
图片压缩工具类 最大高度和最大宽度都为 500,如果超出大小将等比例缩放。 注意可能出现压缩后比原图更大的情况,在调用的地方自己判断大小并决定上传压缩前或压缩后的图到服务器。 / // 将base64转换为blob export function convertBase64UrlToBlob(urlData) { let arr = urlData.split(',') let mime = arr[0].match(/:(.?);/)[1] let bstr = atob(arr[1]) let n = bstr.length let u8arr = new Uint8Array(n) while (n--) { u8arr[n] = bstr.charCodeAt(n) } return new Blob([u8arr], {type: mime}) } // 压缩图片 export function pressImage(path) { //最大高度 const maxHeight = 500; //最大宽度 const maxWidth = 500; return new Promise((resolve, reject) => { let img = new Image(); img.src = path; img.onload = function () { const originHeight = img.height; const originWidth = img.width; let pressedWidth = img.height; let pressedHeight = img.width; if ((originWidth > maxWidth) && (originHeight > maxHeight)) { // 更宽更高, if ((originHeight / originWidth) > (maxHeight / maxWidth)) { // 更加严重的高窄型,确定最大高,压缩宽度 pressedHeight = maxHeight pressedWidth = maxHeight (originWidth / originHeight) } else { //更加严重的矮宽型, 确定最大宽,压缩高度 pressedWidth = maxWidth pressedHeight = maxWidth (originHeight / originWidth) } } else if (originWidth > maxWidth && originHeight <= maxHeight) { // 更宽,但比较矮,以maxWidth作为基准 pressedWidth = maxWidth pressedHeight = maxWidth (originHeight / originWidth) } else if (originWidth <= maxWidth && originHeight > maxHeight) { // 比较窄,但很高,取maxHight为基准 pressedHeight = maxHeight pressedWidth = maxHeight (originWidth / originHeight) } else { // 符合宽高限制,不做压缩 } // 生成canvas let canvas = document.createElement('canvas'); let context = canvas.getContext('2d'); canvas.height = pressedHeight; canvas.width = pressedWidth; context.clearRect(0, 0, pressedWidth, pressedHeight); context.drawImage(img, 0, 0, pressedWidth, pressedHeight); let base64 = canvas.toDataURL('image/', 0.8); let blob = convertBase64UrlToBlob(base64); // 回调函数返回blob的值。也可根据自己的需求返回base64的值 resolve(blob) } }) }
定义的最大宽度和最大高度均为 500,如果图片的宽高至少有一个超出了 500,都会被 等比例 压缩,不用担心变形。可以根据自己项目需要改变 maxWidth 和 maxHeight 。
这里直接把压缩的最大高度和最大宽度写死为 500 了,没有在调用时传。因为一个项目压缩的逻辑和大小一般都一致的,没必要在每次调用的时候传。如果想写的灵活一点,可以在 pressImage 方法里再把 maxWidth 、 maxHeight 和压缩质量传上。
pressImage 方法返回的是 blob 值,根据服务端接口需要可以改为返回 base64,只需将 resolve(blob) 改为 resolve(base64) 即可。
注意一点,对于有些宽高没到 500,且分辨率很小的图片,压缩之后可能比之前还大。猜测可能是 canvas 生成的图片分辨率要比原来高一些,所以最终的图片比压缩前更大。可以在调用的地方加个判断,如果压缩完的大小比原图小,就上传压缩后的图片;如果如果压缩完的大小比原图大,就上传原图。
二、如何使用
将 CompressImageUtils 引入到目标文件,然后调用 pressImage 方法,即可在回调里获得压缩后的结果。注意 pressImage 方法返回的是 Promise。
省略其他无关代码,只保留跟压缩图片和上传相关的
<template> <div> <van-uploader v-model="fileList" :after-read="afterRead" /> </div> </template> <script> import {pressImage} from '../../utils/CompressImageUtils' export default { ponents: {}, methods: { //读取完图片后 afterRead(file) { console.log('afterRead------', file); this._pressAndUploadFile(file); }, //压缩图片上传 _pressAndUploadFile(file) { pressImage(file.content).then(result => { console.log('压缩后的结果', result); // result即为压缩后的结果 console.log('压缩前大小', file.file.size); console.log('压缩后大小', result.size); if (result.size > file.file.size){ console.log('上传原图'); //压缩后比原来更大,则将原图上传 this._uploadFile(file.file, file.file.name); } else { //压缩后比原来小,上传压缩后的 console.log('上传压缩图'); this._uploadFile(result, file.file.name) } }) }, //上传图片 _uploadFile(file, filename) { let params = new FormData(); params.append("file", file, filename); this.$api.uploadImage(params).then(res => { console.log('uploadImage', res); //上传成功,写自己的逻辑 }).catch(err => { console.log('err', err); }); }, } } </script>
在返回结果中加了层判断,压缩后比原来更大,则将原图上传;压缩后比原来小,上传压缩后的。解决压缩后比原图更大的情况。
this.$api.uploadImage(params) 是调用封装的 api 方法,如下
//上传图片 uploadImage(params){ return axios.post(`${base}/api/v1/file`, params, { headers: {'content-type': 'multipart/form-data'} }) },
三、使用效果
先上传一个非常大的,尺寸为 6016 × 4016,16.8M 的大图,看输出日志,压缩后大小仅为 260k 左右。此时判断压缩后比压缩前小,上传压缩图到服务器。
再看个尺寸 300 × 300,12k 的小图,压缩前大小是 11252,压缩后大小是 93656,大了很多。此时判断压缩后比压缩前更大,上传的是原图。
这个工具类对大图的压缩效果很明显,不管多大的图,压缩之后基本不会超过 300k。但对某些小图可能出现压缩完反而更大的情况。在调用的地方加层压缩后和压缩前大小的比较判断,会完美解决这个问题。
也可以在工具类内部判断,但个人觉得跟业务逻辑相关的代码还是不要放在公用的工具类比较好。
以上所述是长沙网络推广给大家介绍的Vue 图片压缩并上传至服务器功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,长沙网络推广会及时回复大家的。在此也非常感谢大家对狼蚁SEO网站的支持!
如果你觉得本文对你有帮助,欢迎网络推广网站推广转载,烦请注明出处,谢谢!
编程语言
- 甘肃哪有关键词排名优化购买方式有哪些
- 甘肃SEO如何做网站优化
- 河南seo关键词优化怎么做电话营销
- 北京SEO优化如何做QQ群营销
- 来宾百度关键词排名:提升您网站曝光率的关键
- 卢龙关键词优化:提升您网站排名的策略与技巧
- 山东网站优化的注意事项有哪些
- 四川整站优化怎样提升在搜索引擎中的排名
- 疏附整站优化:提升网站性能与用户体验的全新
- 海南seo主要做什么工作售后服务要做到哪些
- 荣昌百度网站优化:提升您网站的搜索引擎排名
- 河北seo网站排名关键词优化如何做SEO
- 江西优化关键词排名推广售后保障一般有哪些
- 古浪SEO优化:提升你的网站可见性
- 西藏网站排名优化怎么把网站排名在百度首页
- 如何提升阳东百度快照排名:详尽指南