深入研究HTML5实现图片压缩上传功能


Posted in HTML / CSS onMarch 25, 2016

上篇文章中提到移动端上传图片,我们知道现在流量还是挺贵的,手机的像素是越来越高,拍个照动不动就是好几M,伤不起。虽然客户端可以轻轻松松实现图片压缩再上传,但是我们的应用还可能在浏览器里面打开,怎么办呢,图片压缩。受以前PC上的开发思维影响,尼玛js哪有权限去操作文件,哪有资格压缩图片啊,搞不了,你们客户端去整吧。只能说自己还是有些井底之蛙了。在HTML5的影响下,前端能干的事情越来越多了,开发的功能逼格也越来越高了,H5万岁!前端的魅力也在这,过去不可能的并不意味现在、以后不可能,努力吧,骚年!

js怎么压缩图片???潜意识里确实一开始是觉得实现不了,后来翻阅资料,研究了下,发现可行!搞起!

先说说H5以前我们怎么上传,一般是借助插件、flash或者干脆一个文件form表单,少操不少心。

自从有了H5,老板再也不担心我的开发了。

上篇文章提到图片上传用到了FileReader,FormData,实际上主要用这两个我们基本能实现图片的预览和上传了。实现图片压缩,我们需要借助canvas,是的,就是canvas!

大致思路是:

1、创建一个图片和一个canvas

XML/HTML Code复制内容到剪贴板
  1. var image = new Image(),   
  2. canvas = document.createElement("canvas"),   
  3. ctx = canvas.getContext('2d');  

2、我们将input中选择的图片地址通过FileReader获取后赋给新建的图片对象,然后将图片对象丢到canvas画布上。

XML/HTML Code复制内容到剪贴板
  1. var file = obj.files[0];   
  2.                         var reader = new FileReader();//读取客户端上的文件   
  3.                         reader.onload = function() {   
  4.                             var url = reader.result;//读取到的文件内容.这个属性只在读取操作完成之后才有效,并且数据的格式取决于读取操作是由哪个方法发起的.所以必须使用reader.onload,   
  5.                             image.src=url;//reader读取的文件内容是base64,利用这个url就能实现上传前预览图片   
  6.                             ...   
  7.                         };   
  8.                         image.onload = function() {   
  9.                             var w = image.naturalWidth,   
  10.                                 h = image.naturalHeight;   
  11.                             canvas.width = w;   
  12.                             canvas.height = h;   
  13.                             ctx.drawImage(image, 0, 0, w, h, 0, 0, w, h);   
  14.                             fileUpload();   
  15.                         };   
  16.                         reader.readAsDataURL(file);  

这里需要注意的是,canvas将图片画到画布上的时候需要确定canvas的尺寸,同时设定好drawImage的参数,具体如下:

XML/HTML Code复制内容到剪贴板
  1. void ctx.drawImage(image, sx, sy, sWidth, sHeight, dx, dy, dWidth, dHeight);  

深入研究HTML5实现图片压缩上传功能

dx源图像的左上角在目标canvas上 X 轴的位置。

dy源图像的左上角在目标canvas上 Y 轴的位置。

dWidth在目标canvas上绘制图像的宽度。 允许对绘制的图像进行缩放。 如果不说明, 在绘制时图片宽度不会缩放。

dHeight在目标canvas上绘制图像的高度。 允许对绘制的图像进行缩放。 如果不说明, 在绘制时图片高度不会缩放。

sx需要绘制到目标上下文中的,源图像的矩形选择框的左上角 X 坐标。

sy需要绘制到目标上下文中的,源图像的矩形选择框的左上角 Y 坐标。

sWidth需要绘制到目标上下文中的,源图像的矩形选择框的宽度。如果不说明,整个矩形从坐标的sx和sy开始,到图像的右下角结束。

sHeight需要绘制到目标上下文中的,源图像的矩形选择框的高度。

为了上传完整的图片,这里dx,dy必须设置为0,dWidth和dHeight必须设置为原始图片的宽度和高度。这就是为什么我们需要等image对象下载完毕后获取其原始尺寸,这很关键!

3、图片上传

XML/HTML Code复制内容到剪贴板
  1. function fileUpload() {   
  2.      var data = canvas.toDataURL("image/jpeg", quality);   
  3.      //dataURL 的格式为 “data:image/png;base64,****”,逗号之前都是一些说明性的文字,我们只需要逗号之后的就行了   
  4.     datadata = data.split(',')[1];   
  5.     data = window.atob(data);   
  6.     var ia = new Uint8Array(data.length);   
  7.     for (var i = 0; i < data.length; i++) {   
  8.           ia[i] = data.charCodeAt(i);   
  9.     };   
  10.      //canvas.toDataURL 返回的默认格式就是 image/png   
  11.     var blob = new Blob([ia], {   
  12.      type: "image/jpeg"   
  13.     });   
  14.     var fd = new FormData();   
  15.         fd.append('myFile', blob);   
  16.     var xhr = new XMLHttpRequest();   
  17.     xhr.addEventListener("load", opts.success, false);   
  18.     xhr.addEventListener("error", opts.error, false);   
  19.     xhr.open("POST", opts.url);   
  20.     xhr.send(fd);   
  21.  }  

这里用的关键方法是canvas.toDataURL

XML/HTML Code复制内容到剪贴板
  1. canvas.toDataURL(type, encoderOptions);  

官方的说明是The HTMLCanvasElement.toDataURL() method returns a data URI containing a representation of the image in the format specified by the type parameter (defaults to PNG). The returned image is in a resolution of 96 dpi.实际上就是读取canvas画布上图片的数据。其默认是png格式,如果第一个参数type是image/jpeg的话,第二个参数encoderOptions就可以用来设置图片的压缩质量,经过测试,如果是png格式,100%的宽高经过该方法还有可能使图片变大~~~~适得其反,所以我们可以在canvas.drawImage的时候适当设置sWidth和sHeight,比如同比例缩小1.5倍等,图片质量其实并不太影响查看,尤其对尺寸比较大的图片来说。

上面还有比较陌生的方法atob,其作用是做解码,因为图片格式的base64.

XML/HTML Code复制内容到剪贴板
  1. var encodedData = window.btoa("Hello, world"); // encode a string   
  2. var decodedData = window.atob(encodedData); // decode the string  

该方法解码出来可能是一堆乱码,Uint8Array返回的是8进制整型数组。

Blob是存储二进制文件的容器,典型的Blob对象是一个图片或者声音文件,其默认是PNG格式。

XML/HTML Code复制内容到剪贴板
  1. var blob = new Blob([ia], {   
  2.      type: "image/jpeg"   
  3.     });  

最后通过ajax将Blob对象发送到server即可。

整个流程大致如上,但是~~~实现以后测试跑来说:“你不是说图片压缩了吗,为什么图片还是上传那么慢!”,哥拿起手机对妹纸演示了一下,明明很快嘛,于是反道“是你手机不行或者网络不好吧,你下载图片看明明变小了,比之前肯定快,你看我秒传”。呵呵,说归说,还是偷偷检查代码,在浏览器中打时间log,对比没压缩之前的,尼玛!!!居然才快了几百毫秒!!折腾了半天,之前的代码也重构了,玩我呢。

细心的大神看了上面的代码估计能猜出问题在哪,没错,获取本地图片长宽尺寸的时候出了问题。

深入研究HTML5实现图片压缩上传功能

我去,获取本地4M大小的图片尺寸花了3174ms!!,图片越大时间也越久~

JavaScript Code复制内容到剪贴板
  1. image.onload = function() {   
  2.         var w = image.naturalWidth,   
  3.           h = image.naturalHeight;   
  4.         canvas.width = w / 1.5;   
  5.         canvas.height = h / 1.5;   
  6.         ctx.drawImage(image, 0, 0, w, h, 0, 0, w / 1.5, h / 1.5);   
  7.         Upload.fileUpload(type);   
  8. };  

浏览器在本地取图片的时候是没法直接像file.size一样获取其长宽的,只能通过FileReader拿到内容后赋值给新建的image对象,新建的image对象下载需要时间!怎么破?不就是获取本地图片的尺寸吗,难道没有别的办法了?

于是想到了之前研究过的快速获取图片长宽的博文,点击进入 ,demo地址:http://jsbin.com/jivugadure/edit?html,js,output,定时去查询图片加载过程中的高度或者宽度,不用等整个图片加载完毕。

测了下,还是不行,因为定时查询这种方法对常规的server返回的图片有作用,这里图片地址是base64,貌似时间还更久了~哭。

小结一下:

1、用HTML5来压缩图片上传是可行的,在移动端我们不用依赖客户端或者插件,目前主流浏览器支持程度已经很高了。

2、压缩图片一方面是想减少用户上传等待的时间,另外也减少用户为此牺牲的流量,从整体时间来看,因为获取图片尺寸导致多一次下载需要耗时,其实压不压缩时间差别并不是特别大。除非大神们找到合适的方法能够直接获取图片的尺寸,麻烦也告知我一声,万分感谢;

3、既然时间成本差不多,但是我们压缩了图片,减少了图片的大小,减少了流量的消耗,存储空间以及下次获取该图片的时间,所以还是值得的。

 补充源代码:

JavaScript Code复制内容到剪贴板
  1. (function($) {   
  2.     $.extend($.fn, {   
  3.         fileUpload: function(opts) {   
  4.             this.each(function() {   
  5.                 var $self = $(this);   
  6.                 var quality = opts.quality ? opts.quality / 100 : 0.2;   
  7.                 var dom = {   
  8.                     "fileToUpload": $self.find(".fileToUpload"),   
  9.                     "thumb": $self.find(".thumb"),   
  10.                     "progress": $self.find(".upload-progress")   
  11.                 };   
  12.                 var image = new Image(),   
  13.                     canvas = document.createElement("canvas"),   
  14.                     ctx = canvas.getContext('2d');   
  15.                 var funs = {   
  16.                     setImageUrl: function(url) {   
  17.                         image.src = url;   
  18.                     },   
  19.                     bindEvent: function() {   
  20.                         console.log(dom.fileToUpload)   
  21.                         dom.fileToUpload.on("change"function() {   
  22.                             funs.fileSelect(this);   
  23.                         });   
  24.                     },   
  25.                     fileSelect: function(obj) {   
  26.                         var file = obj.files[0];   
  27.                         var reader = new FileReader();   
  28.                         reader.onload = function() {   
  29.                             var url = reader.result;   
  30.                             funs.setImageUrl(url);   
  31.                             dom.thumb.html(image);   
  32.                         };   
  33.                         image.onload = function() {   
  34.                             var w = image.naturalWidth,   
  35.                                 h = image.naturalHeight;   
  36.                             canvas.width = w;   
  37.                             canvas.height = h;   
  38.                             ctx.drawImage(image, 0, 0, w, h, 0, 0, w, h);   
  39.                             funs.fileUpload();   
  40.                         };   
  41.                         reader.readAsDataURL(file);   
  42.                     },   
  43.                     fileUpload: function() {   
  44.                         var data = canvas.toDataURL("image/jpeg", quality);   
  45.                         //dataURL 的格式为 “data:image/png;base64,****”,逗号之前都是一些说明性的文字,我们只需要逗号之后的就行了   
  46.                         data = data.split(',')[1];   
  47.                         data = window.atob(data);   
  48.                         var ia = new Uint8Array(data.length);   
  49.                         for (var i = 0; i < data.length; i++) {   
  50.                             ia[i] = data.charCodeAt(i);   
  51.                         };   
  52.                         //canvas.toDataURL 返回的默认格式就是 image/png   
  53.                         var blob = new Blob([ia], {   
  54.                             type: "image/jpeg"  
  55.                         });   
  56.                         var fd = new FormData();   
  57.                         fd.append('myFile', blob);   
  58.                         var xhr = new XMLHttpRequest();   
  59.                         xhr.addEventListener("load", opts.success, false);   
  60.                         xhr.addEventListener("error", opts.error, false);   
  61.                         xhr.open("POST", opts.url);   
  62.                         xhr.send(fd);   
  63.                     }   
  64.                 };   
  65.                 funs.bindEvent();   
  66.             });   
  67.         }   
  68.     });   
  69. })(Zepto);  

调用方式:

JavaScript Code复制内容到剪贴板
  1. $(".fileUpload").fileUpload({   
  2.                 "url""savetofile.php",   
  3.                 "file""myFile",   
  4.                 "success":function(evt){   
  5.                     console.log(evt.target.responseText)   
  6.                 }   
  7. });  

希望大家能找到更好的办法,多多交流!感谢!

以上这篇深入研究HTML5实现图片压缩上传功能就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

原文地址:http://www.cnblogs.com/hutuzhu/p/5265023.html

HTML / CSS 相关文章推荐
CSS3教程:background-clip和background-origin
Oct 17 HTML / CSS
纯CSS打造(无图像无js)的非常流行的讲话(语音)气泡效果
Dec 28 HTML / CSS
纯CSS改变webkit内核浏览器的滚动条样式
Apr 17 HTML / CSS
推荐一些比较有用的css3新属性
Nov 11 HTML / CSS
CSS3 2D模拟实现摩天轮旋转效果
Nov 16 HTML / CSS
CSS3 filter(滤镜)实现网页灰色或者黑色模式的示例代码
Feb 24 HTML / CSS
浅谈cookie和localStorage那些事
Aug 27 HTML / CSS
HTML5视频支持检测(检查浏览器是否支持视频播放)
Jun 08 HTML / CSS
HTML5+Canvas+CSS3实现齐天大圣孙悟空腾云驾雾效果
Apr 26 HTML / CSS
Canvas系列之滤镜效果
Feb 12 HTML / CSS
H5页面适配iPhoneX(就是那么简单)
Dec 02 HTML / CSS
HTML5中在title标题标签里设置小图标的方法
Jun 23 HTML / CSS
移动web模拟客户端实现多方框输入密码效果【附代码】
Mar 25 #HTML / CSS
HTML5打开本地app应用的方法
Mar 31 #HTML / CSS
如何在网站上添加谷歌定位信息
Apr 16 #HTML / CSS
HTML5中使用postMessage实现Ajax跨域请求的方法
Apr 19 #HTML / CSS
HTML5的Geolocation地理位置定位API使用教程
May 12 #HTML / CSS
HTML5的video标签的浏览器兼容性增强方案分享
May 19 #HTML / CSS
HTML5中的postMessage API基本使用教程
May 20 #HTML / CSS
You might like
PHP新手上路(九)
2006/10/09 PHP
PHP SPL标准库之文件操作(SplFileInfo和SplFileObject)实例
2015/05/11 PHP
PHP中Notice错误常见解决方法
2017/04/28 PHP
PHP异步进程助手async-helper
2018/02/05 PHP
js+CSS 图片等比缩小并垂直居中实现代码
2008/12/01 Javascript
JavaScript 对象成员的可见性说明
2009/10/16 Javascript
JavaScript 自动分号插入(JavaScript synat:auto semicolon insertion)
2009/11/04 Javascript
js实现checkbox全选和反选示例
2014/05/01 Javascript
Jquery对select的增、删、改、查操作
2015/02/06 Javascript
Javascript中的arguments与重载介绍
2015/03/15 Javascript
基于JS实现新闻列表无缝向上滚动实例代码
2016/01/22 Javascript
EasyUI Pagination 分页的两种做法小结
2016/07/09 Javascript
jquery判断页面网址是否有效的两种方法
2016/12/11 Javascript
Angular 输入框实现自定义验证功能
2017/02/19 Javascript
js实现导航吸顶效果
2017/02/24 Javascript
完美解决UI-Grid表格元素中多个空格显示为一个空格的问题
2017/04/25 Javascript
微信小程序上滑加载下拉刷新(onscrollLower)分批加载数据(二)
2017/05/11 Javascript
vue中各组件之间传递数据的方法示例
2017/07/27 Javascript
微信小程序实现验证码获取倒计时效果
2018/02/08 Javascript
vue axios请求拦截实例代码
2018/03/29 Javascript
简单说说如何使用vue-router插件的方法
2019/04/08 Javascript
微信小程序 可搜索的地址选择实现详解
2019/08/28 Javascript
Vue项目中使用jsonp抓取跨域数据的方法
2019/11/10 Javascript
vue 移动端记录页面浏览位置的方法
2020/03/11 Javascript
微信小程序获取当前时间及星期几的实例代码
2020/09/20 Javascript
[01:27:30]LGD vs Newbee 2019国际邀请赛小组赛 BO2 第二场 8.16
2019/08/19 DOTA
[00:59]DOTA2英雄背景故事——上古巨神
2020/06/28 DOTA
Python中处理字符串的相关的len()方法的使用简介
2015/05/19 Python
Python 循环语句之 while,for语句详解
2018/04/23 Python
python 提取文件指定列的方法示例
2019/08/07 Python
通过Python实现一个简单的html页面
2020/05/16 Python
手工制作的音乐盒:Music Box Attic
2019/09/05 全球购物
家长对学生的评语
2014/04/18 职场文书
健康状况证明模板
2014/10/23 职场文书
该怎么书写道歉信?
2019/07/03 职场文书
Python爬虫入门案例之回车桌面壁纸网美女图片采集
2021/10/16 Python