用的canvas。这个问题测试妹子反馈了好几次bug,解决了好多次,虽然用了比较僵硬的办法,但总算最终解决了。

因为php的同事说,页面上的图片要直接调用七牛的接口上传到七牛,所以后端那边不能处理,必须前端这边把图片处理到2m以下。可是我感觉用之前的办法只是把宽高变小,并不能保证压缩后一定就小于2m。所以没办法吧,还是得搞。

/**
* 将图片转化为base64
*/
function imgBase64(file) {
var self = this;
// 看支持不支持FileReader
if (!file || !window.FileReader) return;
// 创建一个 Image 对象
var image = new Image();
// 绑定 load 事件处理器,加载完成后执行
image.onload = function(){
// 获取 canvas DOM 对象
var canvas = document.createElement('canvas')
// 返回一个用于在画布上绘图的环境, '2d' 指定了您想要在画布上绘制的类型
var ctx = canvas.getContext('2d')
// 如果高度超标 // 参数,最大高度
var MAX_HEIGHT = 9000;
if(image.height > MAX_HEIGHT) {
// 宽度等比例缩放 *=
image.width *= MAX_HEIGHT / image.height;
image.height = MAX_HEIGHT;
}
// 获取 canvas的 2d 环境对象,
// 可以理解Context是管理员,canvas是房子
// canvas清屏
console.log('canvas.width:', canvas.width);
ctx.clearRect(0, 0, canvas.width, canvas.height);
// 重置canvas宽高
canvas.width = image.width;
canvas.height = image.height;
// 将图像绘制到canvas上
ctx.drawImage(image, 0, 0, image.width, image.height);
// !!! 注意,image 没有加入到 dom之中
console.log(file.type);
// console.log(canvas.toDataURL('image/jpeg',0.5));
//----------//
var maxSize = 2*1024; // 2M
var fileSize = file.size/1024; // 图片大小
if(fileSize > maxSize) { // 如果图片大小大于2m,进行压缩
console.log(maxSize,fileSize, maxSize/fileSize );
uploadSrc = canvas.toDataURL(file.type, maxSize/fileSize);
document.getElementById('previewImage').src = uploadSrc;
uploadFile = convertBase64UrlToFile(uploadSrc, file.name.split('.')[0]); // 转成file文件
} else {
uploadSrc = canvas.toDataURL(file.type, 0.5);
document.getElementById('previewImage').src = uploadSrc;
uploadFile = file;
}
//--------//
};
if (/^image/.test(file.type)) {
// 创建一个reader
var reader = new FileReader();
// 将图片将转成 base64 格式
reader.readAsDataURL(file);
// 读取成功后的回调
reader.onload = function () {
// 设置src属性,浏览器会自动加载。
// 记住必须先绑定事件,才能设置src属性,否则会出同步问题。
image.src = this.result;
}
}
}

这段代码是找的之前别人写过的代码,然后自己加了点东西凑合用的。
之前没怎么弄过压缩图片,直觉觉得不能把宽高压缩的太小,怕把图片给压缩模糊了,于是傻逼的把最大高度设置了9000。

var MAX_HEIGHT = 9000;

压缩的图片质量也设置了一个奇奇怪怪的值。

uploadSrc = canvas.toDataURL(file.type, maxSize/fileSize); 

测试了一波上线了,结果果然出问题了。测试同学反馈用户的问题:压缩后会大于2m!于是脑子进水的改了一波。

var maxSize = 1.5*1024; // 1.5M

结果又测出了问题:小于2m的图片压缩后会出现大于2m的情况!才忽然想到一个问题,大于1.5m的图片转base64之后是可能大于2m的,自己就是瞎改。

于是又想着把最大高度改了,试试。顺便把图片质量降低。

var MAX_HEIGHT = 4000;
...
uploadSrc = canvas.toDataURL(file.type, 0.3); // 把0.5改成了0.3

结果测试妹子测完后心虚的问,完全没问题了么?心里也比较虚的不得了,干脆加了一个判断,压缩后还是大于2m的话,就提示重新上传图片。当时都上线了,结果过了几天测试妹子一琢磨,这么提示还是不友好,不合理,又开始敦促优化。

给改了方案,压缩之后如果大于2m,就再进行压缩,心想着这应该万无一失了,果然很久测试和用户都没再反映,至少通过了一段比较长时间的实践检验。
其实把最大高度还改小了一点,改成了3000,但感觉也并什么用。

/**
* 校验图片转换后大小并上传
*/
function checkAndHandleUpload(file) {
imgBase64(file, function (image, canvas) {
var maxSize = 2*1024; // 2M
var fileSize = file.size/1024; // 图片大小 if(fileSize > maxSize) { // 如果图片大小大于2m,进行压缩
console.log(maxSize,fileSize, maxSize/fileSize );
uploadSrc = canvas.toDataURL(file.type, maxSize/fileSize);
uploadFile = convertBase64UrlToFile(uploadSrc, file.name.split('.')[0]); // 转成file文件
} else {
uploadSrc = image.src; //canvas.toDataURL(file.type,0.5);
uploadFile = file;
} var compressedSize = uploadFile.size / 1024 / 1024;
if(compressedSize.toFixed(2) > 2.00) {
checkAndHandleUpload(uploadFile);
} else {
document.getElementById('previewImage').src = uploadSrc;
}
});
} /**
* 将图片转化为base64
*/
function imgBase64(file, callback) {
var self = this;
// 看支持不支持FileReader
if (!file || !window.FileReader) return;
// 创建一个 Image 对象
var image = new Image();
// 绑定 load 事件处理器,加载完成后执行
image.onload = function(){
// 获取 canvas DOM 对象
var canvas = document.createElement('canvas')
// 返回一个用于在画布上绘图的环境, '2d' 指定了您想要在画布上绘制的类型
var ctx = canvas.getContext('2d')
// 如果高度超标 // 参数,最大高度
var MAX_HEIGHT = 3000;
if(image.height > MAX_HEIGHT) {
// 宽度等比例缩放 *=
image.width *= MAX_HEIGHT / image.height;
image.height = MAX_HEIGHT;
}
// 获取 canvas的 2d 环境对象,
// 可以理解Context是管理员,canvas是房子
// canvas清屏
console.log('canvas.width:', canvas.width);
ctx.clearRect(0, 0, canvas.width, canvas.height);
// 重置canvas宽高
canvas.width = image.width;
canvas.height = image.height;
// 将图像绘制到canvas上
ctx.drawImage(image, 0, 0, image.width, image.height);
// !!! 注意,image 没有加入到 dom之中
console.log(file.type);
// console.log(canvas.toDataURL('image/jpeg',0.5));
//----------//
callback(image, canvas);
//--------//
};
if (/^image/.test(file.type)) {
// 创建一个reader
var reader = new FileReader();
// 将图片将转成 base64 格式
reader.readAsDataURL(file);
// 读取成功后的回调
reader.onload = function () {
// self.imgUrls.push(this.result);
// 设置src属性,浏览器会自动加载。
// 记住必须先绑定事件,才能设置src属性,否则会出同步问题。
image.src = this.result;
}
}
}

js压缩图片到2m以下的更多相关文章

  1. js压缩图片base64长度

    var myCanvas=$('.img-container > img').cropper('getCroppedCanvas'); (function (base64){ var image ...

  2. JS压缩图片(canvas),返回base64码

    上传图片时总会遇到图片过大上传不上去的问题,本方法是在网上搜的压缩图片的例子,我测试过了,确实能用,但是照搬别人的代码,发现压缩后图片会失真,不清晰,现经修改图片清晰度还可以,不仔细看差别不大,so, ...

  3. js 压缩图片(只缩小体积,不更改图片尺寸)

      1.情景展示 如上图所示,点击上传图片按钮,调用手机摄像头拍照功能. <input onchange="javascript:imgFun.uploadPicture();&quo ...

  4. js 压缩图片 上传

    感谢,参考了以下作者的绝大部分内容 https://blog.csdn.net/tangxiujiang/article/details/78755292 https://blog.csdn.net/ ...

  5. 了解JS压缩图片,这一篇就够了

    前言 公司的移动端业务需要在用户上传图片是由前端压缩图片大小,再上传到服务器,这样可以减少移动端上行流量,减少用户上传等待时长,优化用户体验. 插播一下,本文案例已整理成插件,已上传npm ,可通过 ...

  6. base64之js压缩图片

    在日常的一些项目中会有上传图片之类的接口,如果图片过大了再上传的时候是非常的耗时以及占用资源,在这里就给大家分享一下如何在js中把大的图片压缩成小的图片,我这里的功能是用户点击按钮调用相机或者选择文件 ...

  7. js压缩图片上传插件localResizeIMG

    示例 /** * 本地图片压缩后上传 */ $("#vfile").change(function(){ var _this = $(this); lrz(this.files[0 ...

  8. js压缩图片并上传,不失真,保证图片清晰度

    <!DOCTYPE HTML> <html lang="zh-CN"> <head> <meta charset="UTF-8& ...

  9. html5+js压缩图片上传

    最近在折腾移动站的开发,涉及到了一个手机里面上传图片.于是经过N久的折腾,找到一个插件,用法如下: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 ...

随机推荐

  1. 非关系型数据库redis

    Redis初始 redis中不区分字母的大小写 查看端口 ss tnlp 安装 编译安装(需要先配置epel源) yum install -y redis 编译安装 wget https://down ...

  2. LeetCode-031-下一个排列

    下一个排列 题目描述:实现获取 下一个排列 的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列. 如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列). 必须 原地 修 ...

  3. LeetCode-082-删除排序链表中的重复元素 II

    删除排序链表中的重复元素 II 题目描述:存在一个按升序排列的链表,给你这个链表的头节点 head ,请你删除链表中所有存在数字重复情况的节点,只保留原始链表中 没有重复出现 的数字. 返回同样按升序 ...

  4. HarmonyOS UI组件在线预览,程序员直呼“不要太方便~”

    一.介绍 以往大家如果想查看组件的使用效果,需要打开DevEco Studio构建工程.现在为了便于大家高效开发,文档上线了JS UI组件在线预览功能,无需本地构建工程,在线即可修改组件样式等参数.一 ...

  5. python安装各种插件

    http://www.lfd.uci.edu/~gohlke/pythonlibs/#pip 感受:如果编辑pip真的一直出问题,考虑降成32位的进行安装.毕竟合理搭配比木桶突出有用.

  6. tp5 ajax批量删除(自写)

    html代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="U ...

  7. 当.Net撞上BI可视化,这3种“套路”你必须知道

    最近葡萄在做技术支持,又遇到了客户给我们出的新问题. 事情是这样的. 这次客户使用的是.Net项目,直接做BI大屏过于复杂,所以想直接集成使用BI数据可视化分析大屏. 所以,这次我们就从--Wyn出发 ...

  8. Go 循环语句

    Go 循环语句 一.概述 在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句. 循环程序的流程图: Go 语言提供了以下几种类型循环处理语句: 循环类型 描述 for 循环 ...

  9. kubernetes如何强制删除namespace

    K8S如何强制删除namespace 先运行kubectl get namespace ingress-nginx -o json > nginx.json,拿到当前namespace描述,然后 ...

  10. 6月21日 Django ORM那些相关操作(表关联、聚合查询和分组查询)

    一.ForeignKey操作 正向查找 对象查找(跨表) 语法: 对象.关联字段.字段   示例: book_obj = models.Book.objects.first() # 第一本书对象 pr ...