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实现的文字亮光特效的代码演示
Nov 27 HTML / CSS
使用CSS3滤镜的filter:blur属性制作毛玻璃模糊效果的方法
Jul 08 HTML / CSS
CSS3 mask 遮罩的具体使用方法
Nov 03 HTML / CSS
CSS3实现背景透明文字不透明的示例代码
Jun 25 HTML / CSS
css3针对移动端卡顿问题的解决(动画性能优化)
Feb 14 HTML / CSS
Javascript 高级手势使用介绍
Apr 21 HTML / CSS
html5 CSS过度-webkit-transition使用介绍
Jul 02 HTML / CSS
HTML5 Canvas渐进填充与透明实现图像的Mask效果
Jul 11 HTML / CSS
html5 迷宫游戏(碰撞检测)实例一
Jul 25 HTML / CSS
详解HTML5中div和section以及article的区别
Jul 14 HTML / CSS
详解如何解决H5开发使用wx.hideMenuItems无效果不生效
Jan 20 HTML / CSS
CSS几步实现赛博朋克2077风格视觉效果
Jun 16 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 读取文件内容代码(txt,js等)
2009/12/06 PHP
PHP采用自定义函数实现遍历目录下所有文件的方法
2014/08/19 PHP
php实现最简单的MVC框架实例教程
2014/09/08 PHP
浅谈ThinkPHP的URL重写
2014/11/25 PHP
PHP文件及文件夹操作之创建、删除、移动、复制
2016/07/13 PHP
PHP屏蔽关键字实现方法
2016/11/17 PHP
JavaScript 全角转半角部分
2009/10/28 Javascript
javascript预加载图片、css、js的方法示例介绍
2013/10/14 Javascript
js获取url中的参数且参数为中文时通过js解码
2014/03/19 Javascript
js获取ajax返回值代码
2014/04/30 Javascript
jQuery事件之键盘事件(ctrl+Enter回车键提交表单等)
2014/05/11 Javascript
jQuery结合ajax实现动态加载文本内容
2015/05/19 Javascript
解决angular的$http.post()提交数据时后台接收不到参数值问题的方法
2015/12/10 Javascript
JavaScript事件学习小结(三)js事件对象
2016/06/09 Javascript
vue用addRoutes实现动态路由的示例
2017/09/15 Javascript
vue.js分页中单击页码更换页面内容的方法(配合spring springmvc)
2018/02/10 Javascript
js中数组常用方法总结(推荐)
2019/04/09 Javascript
关于Layui Table隐藏列问题
2019/09/16 Javascript
Vue数字输入框组件示例代码详解
2020/01/15 Javascript
javascript单张多张图无缝滚动实例代码
2020/05/10 Javascript
vant中的toast层级改变操作
2020/11/04 Javascript
Python实现的堆排序算法示例
2018/04/29 Python
flask框架中勾子函数的使用详解
2018/08/01 Python
python打开windows应用程序的实例
2019/06/28 Python
python中利用numpy.array()实现俩个数值列表的对应相加方法
2019/08/26 Python
使用OpenCV circle函数图像上画圆的示例代码
2019/12/27 Python
python GUI框架pyqt5 对图片进行流式布局的方法(瀑布流flowlayout)
2020/03/12 Python
PyQt5实现仿QQ贴边隐藏功能的实例代码
2020/05/24 Python
python设置表格边框的具体方法
2020/07/17 Python
网购亚洲时装、美容产品和生活百货:YesStyle
2016/09/15 全球购物
美国最好的保健品打折网店:Swanson
2017/08/04 全球购物
土耳其新趋势女装购物网站:Addax
2020/01/07 全球购物
银行职员工作失误检讨书
2014/10/14 职场文书
金正昆讲礼仪观后感
2015/06/11 职场文书
2016大学迎新欢迎词
2015/09/29 职场文书
Python代码风格与编程习惯重要吗?
2021/06/03 Python