图片压缩

  1 <!DOCTYPE html>
  2 <html>
  3 <head>
  4     <meta charset="UTF-8">
  5     <title>XMLHttpRequest上传文件</title>
  6     <script type="text/javascript">
  7         /*
  8         三个参数
  9         file:一个是文件(类型是图片格式),
 10         w:一个是文件压缩的后宽度,宽度越小,字节越小
 11         objDiv:一个是容器或者回调函数
 12         photoCompress()
 13          */
 14         function photoCompress(file,w,objDiv){
 15             var ready=new FileReader();
 16             /*开始读取指定的Blob对象或File对象中的内容. 当读取操作完成时,readyState属性的值会成为DONE,如果设置了onloadend事件处理程序,则调用之.同时,result属性中将包含一个data: URL格式的字符串以表示所读取文件的内容.*/
 17             ready.readAsDataURL(file);
 18             ready.onload=function(){
 19                 var re=this.result;
 20                 canvasDataURL(re,w,objDiv)
 21             }
 22         }
 23         function canvasDataURL(path, obj, callback){
 24             var img = new Image();
 25             img.src = path;
 26             img.onload = function(){
 27                 var that = this;
 28                 // 默认按比例压缩
 29                 var w = that.width,
 30                     h = that.height,
 31                     scale = w / h;
 32                 w = obj.width || w;
 33                 h = obj.height || (w / scale);
 34                 var quality = 0.7;  // 默认图片质量为0.7
 35                 //生成canvas
 36                 var canvas = document.createElement('canvas');
 37                 var ctx = canvas.getContext('2d');
 38                 // 创建属性节点
 39                 var anw = document.createAttribute("width");
 40                 anw.nodeValue = w;
 41                 var anh = document.createAttribute("height");
 42                 anh.nodeValue = h;
 43                 canvas.setAttributeNode(anw);
 44                 canvas.setAttributeNode(anh);
 45                 ctx.drawImage(that, 0, 0, w, h);
 46                 // 图像质量
 47                 if(obj.quality && obj.quality <= 1 && obj.quality > 0){
 48                     quality = obj.quality;
 49                 }
 50                 // quality值越小,所绘制出的图像越模糊
 51                 var base64 = canvas.toDataURL('image/jpeg', quality);
 52                 // 回调函数返回base64的值
 53                 callback(base64);
 54             }
 55         }
 56         /**
 57          * 将以base64的图片url数据转换为Blob
 58          * @param urlData
 59          *            用url方式表示的base64图片数据
 60          */
 61         function convertBase64UrlToBlob(urlData){
 62             var arr = urlData.split(','), mime = arr[0].match(/:(.*?);/)[1],
 63                 bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n);
 64             while(n--){
 65                 u8arr[n] = bstr.charCodeAt(n);
 66             }
 67             return new Blob([u8arr], {type:mime});
 68         }
 69 
 70 
 71         var xhr;
 72         //上传文件方法
 73         function UpladFile() {
 74             var fileObj = document.getElementById("file").files[0]; // js 获取文件对象
 75             var url = "后台图片上传接口"; // 接收上传文件的后台地址 
 76 
 77             var form = new FormData(); // FormData 对象
 78 
 79             if(fileObj.size/1024 > 1025) { //大于1M,进行压缩上传
 80                 photoCompress(fileObj, {
 81                     quality: 0.2
 82                 }, function(base64Codes){
 83                     //console.log("压缩后:" + base.length / 1024 + " " + base);
 84                     var bl = convertBase64UrlToBlob(base64Codes);
 85                     form.append("file", bl, "file_"+Date.parse(new Date())+".jpg"); // 文件对象,重要
 86                     xhr = new XMLHttpRequest();  // XMLHttpRequest 对象
 87                     xhr.open("post", url, true); //post方式,url为服务器请求地址,true 该参数规定请求是否异步处理。
 88                     xhr.onload = uploadComplete; //请求完成
 89                     xhr.onerror =  uploadFailed; //请求失败
 90 
 91                     xhr.upload.onprogress = progressFunction;//【上传进度调用方法实现】
 92                     xhr.upload.onloadstart = function(){//上传开始执行方法
 93                         ot = new Date().getTime();   //设置上传开始时间
 94                         oloaded = 0;//设置上传开始时,以上传的文件大小为0
 95                     };
 96 
 97                     xhr.send(form); //开始上传,发送form数据
 98                 });
 99             }else{ //小于等于1M 原图上传
100                 form.append("file", fileObj); // 文件对象
101                 xhr = new XMLHttpRequest();  // XMLHttpRequest 对象
102                 xhr.open("post", url, true); //post方式,url为服务器请求地址,true 该参数规定请求是否异步处理。
103                 xhr.onload = uploadComplete; //请求完成
104                 xhr.onerror =  uploadFailed; //请求失败
105 
106                 xhr.upload.onprogress = progressFunction;//【上传进度调用方法实现】
107                 xhr.upload.onloadstart = function(){//上传开始执行方法
108                     ot = new Date().getTime();   //设置上传开始时间
109                     oloaded = 0;//设置上传开始时,以上传的文件大小为0
110                 };
111 
112                 xhr.send(form); //开始上传,发送form数据
113             }
114         }
115 
116         //上传成功响应
117         function uploadComplete(evt) {
118             //服务断接收完文件返回的结果
119 
120             var data = JSON.parse(evt.target.responseText);
121             if(data.success) {
122                 alert("上传成功!");
123             }else{
124                 alert("上传失败!");
125             }
126 
127         }
128         //上传失败
129         function uploadFailed(evt) {
130             alert("上传失败!");
131         }
132         //取消上传
133         function cancleUploadFile(){
134             xhr.abort();
135         }
136 
137         //上传进度实现方法,上传过程中会频繁调用该方法
138         function progressFunction(evt) {
139             var progressBar = document.getElementById("progressBar");
140             var percentageDiv = document.getElementById("percentage");
141             // event.total是需要传输的总字节,event.loaded是已经传输的字节。如果event.lengthComputable不为真,则event.total等于0
142             if (evt.lengthComputable) {//
143                 progressBar.max = evt.total;
144                 progressBar.value = evt.loaded;
145                 percentageDiv.innerHTML = Math.round(evt.loaded / evt.total * 100) + "%";
146             }
147             var time = document.getElementById("time");
148             var nt = new Date().getTime();//获取当前时间
149             var pertime = (nt-ot)/1000; //计算出上次调用该方法时到现在的时间差,单位为s
150             ot = new Date().getTime(); //重新赋值时间,用于下次计算
151             var perload = evt.loaded - oloaded; //计算该分段上传的文件大小,单位b
152             oloaded = evt.loaded;//重新赋值已上传文件大小,用以下次计算
153             //上传速度计算
154             var speed = perload/pertime;//单位b/s
155             var bspeed = speed;
156             var units = 'b/s';//单位名称
157             if(speed/1024>1){
158                 speed = speed/1024;
159                 units = 'k/s';
160             }
161             if(speed/1024>1){
162                 speed = speed/1024;
163                 units = 'M/s';
164             }
165             speed = speed.toFixed(1);
166             //剩余时间
167             var resttime = ((evt.total-evt.loaded)/bspeed).toFixed(1);
168             time.innerHTML = ',速度:'+speed+units+',剩余时间:'+resttime+'s';
169             if(bspeed==0) time.innerHTML = '上传已取消';
170         }
171     </script>
172 </head>
173 <body>
174 <progress id="progressBar" value="0" max="100" style=" 300px;"></progress>
175 <span id="percentage"></span><span id="time"></span>
176 <br /><br />
177 <input type="file" id="file" name="myfile" accept="image/x-png, image/jpg, image/jpeg, image/gif"/>
178 <input type="button" onclick="UpladFile()" value="上传" />
179 <input type="button" onclick="cancleUploadFile()" value="取消" />
180 </body>
181 </html>

转自https://www.cnblogs.com/xiangsj/p/8932525.html

原文地址:https://www.cnblogs.com/qihang0/p/13844821.html