html5使用canvas压缩图片的示例代码


Posted in HTML / CSS onSeptember 11, 2018

前俩天做了一个图片转base64上传的功能,发现如果图片的base64过大的话,请求会变的很慢,严重的直接超时了,所以想到了在上传前压缩一下图片,然后再上传到后台,这样可以大大的提高效率,在这里记录一下利用 canvas 压缩图片遇到的几个坑。完整代码会在文末给出。

第一个坑,在压缩图片的时候没获取图片本身的宽高,给了一个 600*480 的定宽定高,因为是手机端的,在上传图片的时候都是几兆的图片,所以这块没任何问题。出问题的地方在 修改头像的时候,测试的时候上传的图片都是小图片,然后就出现了 压缩后的图片显示不完全,大部分都是空白的现象,这就是因为在压缩的时候没有考虑图片原本的宽高的情况。

第二个坑,解决第一个坑的办法就是在图片加载完成后(onload),获取图片本身的宽高,然后赋值给 canvas ,这样进行操作,但是这有个坑就是,图片加载是异步的,在你 return 的时候,返回的可能是 undefined 而不是你需要的 压缩后的 base64。这里的解决方法是,新建一个 Promise ,然后把结果 resolve() 返回去,在调用的时候 .then() 得到结果。

知识点:

  • canvas 的 toDataURL('image/png', 0.9) ; 把 canvas 画的图片转换为 base64,第一个参数表示的是图片的类型,第二个参数表示的是图片的清晰度。
  • 规定一个最大尺寸,如果图片本身的宽高大于这个尺寸,按照最大的一个边进行缩放,另一个根据图片的 比例 进行设置,然后设置给 canvas .

miniImage.js

export default async function miniSize(imgData, maxSize = 200*1024){
    // const maxSize = 200 * 1024;

    if(imgData && imgData.files && imgData.files.size < maxSize) {
        return imgData.url;
    }else{
      console.log('----------------压缩图片-------------------');
      const canvas = document.createElement('canvas');
      let img = new Image();
      img.src = imgData.url;
      let ctx = canvas.getContext('2d');
      return new Promise((resolve =>{
        img.addEventListener('load', function(){
          //图片原始尺寸
          let originWidth = this.width;
          let originHeight = this.height;
          // 最大尺寸限制
          let maxWidth = 400, maxHeight = 400;
          // 目标尺寸
          let targetWidth = originWidth, targetHeight = originHeight;
          // 图片尺寸超过400x400的限制
          if (originWidth > maxWidth || originHeight > maxHeight) {
            if (originWidth / originHeight > maxWidth / maxHeight) {
              // 更宽,按照宽度限定尺寸
              targetWidth = maxWidth;
              targetHeight = Math.round(maxWidth * (originHeight / originWidth));
            } else {
              targetHeight = maxHeight;
              targetWidth = Math.round(maxHeight * (originWidth / originHeight));
            }
          }
          canvas.width = targetWidth;
          canvas.height = targetHeight;
          ctx.drawImage(img, 0, 0, targetWidth, targetHeight);
          let base64 = canvas.toDataURL('image/png', 0.9);
          resolve(base64);
        }, false);
      }))
    }
}

调用:

test.js

onChangeImg = async (files, type, index) => {
    let previous = this.props.imagePicker.files;
    if(type === "add") {
      let result = miniSize(files[files.length-1]);
      //使用 .then() 调用获得结果
      await result.then(res => {
         previous.push({url: res});
      });
    }else if(type === "remove") {
        previous.splice(index,1);
    }
    await this.props.dispatch({
      type: 'imagePicker/saveImage',
      payload: {
        files: previous
      }
    })
  }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

HTML / CSS 相关文章推荐
CSS3对图片照片进行边缘模糊处理的实现
Aug 08 HTML / CSS
浅析与CSS3的loading动画加载相关的transition优化
May 18 HTML / CSS
详解CSS3中border-image的使用
Jul 18 HTML / CSS
详解CSS3中nth-child与nth-of-type的区别
Jan 05 HTML / CSS
利用CSS3实现进度条的两种姿势详解
Mar 21 HTML / CSS
使用纯 CSS 创作一个脉动 loader效果的源码
Sep 28 HTML / CSS
CSS3实现水平居中、垂直居中、水平垂直居中的实例代码
Feb 27 HTML / CSS
HTML5新增form控件和表单属性实例代码详解
May 15 HTML / CSS
html5 video全屏播放/自动播放的实现示例
Aug 06 HTML / CSS
使用HTML+Css+transform实现3D导航栏的示例代码
Mar 31 HTML / CSS
用position:sticky完美解决小程序吸顶问题的实现方法
Apr 24 HTML / CSS
聊聊CSS粘性定位sticky案例解析
Jun 01 HTML / CSS
canvas像素点操作之视频绿幕抠图
Sep 11 #HTML / CSS
HTML5中使用json对象的实例代码
Sep 10 #HTML / CSS
html5 figure和figcaption的使用方法
Sep 10 #HTML / CSS
用canvas画心电图的示例代码
Sep 10 #HTML / CSS
Html5 canvas实现粒子时钟的示例代码
Sep 06 #HTML / CSS
canvas离屏技术与放大镜实现代码示例
Aug 31 #HTML / CSS
使用PDF.JS插件在HTML中预览PDF文件的方法
Aug 29 #HTML / CSS
You might like
php XMLWriter类的简单示例代码(RSS输出)
2011/09/30 PHP
Yii学习总结之数据访问对象 (DAO)
2015/02/22 PHP
Alliance vs AM BO3 第一场2.13
2021/03/10 DOTA
JS IE和FF兼容性问题汇总
2009/02/09 Javascript
URL地址中的#符号使用说明
2011/02/12 Javascript
javascript 基础篇2 数据类型,语句,函数
2012/03/14 Javascript
jQuery实现鼠标经过图片变亮其他变暗效果
2015/05/08 Javascript
js实现简单div拖拽功能实例
2015/05/12 Javascript
JQuery实现网页右侧随动广告特效
2016/01/17 Javascript
微信jssdk用法汇总
2016/07/16 Javascript
JS获取及验证开始结束日期的方法
2016/08/20 Javascript
使用vue.js2.0 + ElementUI开发后台管理系统详细教程(一)
2017/01/21 Javascript
CodeMirror js代码加亮使用总结
2017/03/25 Javascript
jQuery实现鼠标滑过预览图片大图效果的方法
2017/04/26 jQuery
JavaScript之class继承_动力节点Java学院整理
2017/07/03 Javascript
解决vue无法设置滚动位置的问题
2018/10/07 Javascript
vue中的inject学习教程
2019/04/24 Javascript
flexible.js实现移动端rem适配方案
2020/04/07 Javascript
javascript使用canvas实现饼状图效果
2020/09/08 Javascript
js对象属性名驼峰式转下划线的实例代码
2020/09/17 Javascript
[54:41]2018DOTA2亚洲邀请赛3月30日 小组赛B组 VGJ.T VS paiN
2018/03/31 DOTA
[51:17]Mineski vs Secret 2019国际邀请赛淘汰赛 败者组 BO3 第一场 8.22
2019/09/05 DOTA
python自动化测试之从命令行运行测试用例with verbosity
2014/09/28 Python
浅谈flask源码之请求过程
2018/07/26 Python
Pandas操作CSV文件的读写实现方法
2019/11/13 Python
python读取ini配置文件过程示范
2019/12/23 Python
python实现翻译word表格小程序
2020/02/27 Python
Pycharm打开已有项目配置python环境的方法
2020/07/03 Python
python 获取域名到期时间的方法步骤
2021/02/10 Python
乌克兰的第一家手表店:Deka
2020/03/05 全球购物
视图的作用
2014/12/19 面试题
《充气雨衣》教学反思
2014/04/07 职场文书
2014年母亲节演讲稿范文
2014/05/07 职场文书
运动会致辞稿
2015/07/29 职场文书
2016小学新学期寄语
2015/12/04 职场文书
职场中的你,辞职信写对了吗?
2019/06/26 职场文书