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实现垂直下拉动画菜单示例
Apr 22 HTML / CSS
任意一块网页内容实现“活”的背景(目前火狐浏览器专有)
May 07 HTML / CSS
目前不被任何主流浏览器支持的CSS3属性汇总
Jul 21 HTML / CSS
CSS3制作酷炫的三维相册效果
Jul 01 HTML / CSS
巧用CSS3的calc()宽度计算做响应模式布局的方法
Mar 22 HTML / CSS
html5-websocket基于远程方法调用的数据交互实现
Dec 04 HTML / CSS
HTML5到底会有什么发展?HTML5的前景展望
Jul 07 HTML / CSS
HTML5中的音频和视频媒体播放元素小结
Jan 29 HTML / CSS
实例讲解利用HTML5 Canvas API操作图形旋转的方法
Mar 22 HTML / CSS
详解CSS故障艺术
May 25 HTML / CSS
分享几个实用的CSS代码块
Jun 10 HTML / CSS
CSS 鼠标点击拖拽效果的实现代码
Dec 24 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将回调函数作用到给定数组单元的方法
2014/08/19 PHP
叫你如何修改Nginx与PHP的文件上传大小限制
2014/09/10 PHP
php学习笔记之基础知识
2014/11/08 PHP
PHP如何通过传引用的思想实现无限分类(代码简单)
2015/10/13 PHP
全面解析PHP操作Memcache基本函数
2016/07/14 PHP
javascript实现轮显新闻标题链接
2007/08/13 Javascript
asp批量修改记录的代码
2008/06/25 Javascript
javascript中的绑定与解绑函数应用示例
2013/06/24 Javascript
js动态修改input输入框的type属性(实现方法解析)
2013/11/13 Javascript
Jquery 改变radio/checkbox选中状态,获取选中的值(示例代码)
2013/12/12 Javascript
js showModalDialog弹出窗口实例详解
2014/01/07 Javascript
js操作模态窗口及父子窗口间相互传值示例
2014/06/09 Javascript
JavaScript中数组成员的添加、删除介绍
2014/12/30 Javascript
javascript如何定义对象数组
2016/06/07 Javascript
使用jquery.qrcode.js生成二维码插件
2016/10/17 Javascript
AngularJs表单验证实例代码解析
2016/11/29 Javascript
js实现简单的获取验证码按钮效果
2017/03/03 Javascript
angular ng-click防止重复提交实例
2017/06/16 Javascript
js判断用户是输入的地址请求的路径(实例讲解)
2017/07/18 Javascript
QRCode.js二维码生成并能长按识别
2018/10/16 Javascript
Vue路由 重定向和别名的区别说明
2020/09/09 Javascript
[52:02]DOTA2-DPC中国联赛 正赛 Phoenix vs Dragon BO3 第二场 2月26日
2021/03/11 DOTA
Python Deque 模块使用详解
2014/07/04 Python
Python实现的递归神经网络简单示例
2017/08/11 Python
Python实现PS滤镜特效之扇形变换效果示例
2018/01/26 Python
Python实现网站表单提交和模板
2019/01/15 Python
django配置app中的静态文件步骤
2020/03/27 Python
python 追踪except信息方式
2020/04/25 Python
canvas绘制树形结构可视图形的实现
2020/04/03 HTML / CSS
英国的知名精品百货公司:House of Fraser(福来德)
2016/08/14 全球购物
ManoMano英国:欧洲第一家专注于DIY和园艺市场的电商平台
2020/03/12 全球购物
竞聘书格式及范文
2014/03/31 职场文书
项目合作意向书模板
2014/07/29 职场文书
学校个人对照检查材料
2014/08/26 职场文书
社区四风存在问题及整改措施
2014/10/26 职场文书
2016年圣诞节活动总结范文
2016/04/01 职场文书