基于Canvas+Exif实现的图片压缩功能

2 篇文章 0 订阅

前言:本文记录作者在vue项目中因图片上传限制及后台保存图片过大的问题,因此在图片上传到后台前做一个图片压缩的功能。如果觉得作者写的不错,希望得到您的点赞/收藏/支持,如果有不同意见,欢迎下方评论区留言。

一、初步封装的JS方法及调用

注:
1、此方法以满足大致的功能需求,但是还有一个小问题!!!下面在解释为啥,哈哈哈
2、以下代码注释已经很清晰了。

index.js

/**
 * 将图片压缩为对应尺寸
 * @param {Object} options
 * @param {String} options.img 图片的url或者base64数据
 * @param {Number} options.width 目标图片的宽度
 * @param {Number} options.height 目标图片的高度
 * @param {Number} options.quality 生成目标图片质量
 * @param {String} options.fit 图片压缩填充模式默认 scale:按比例缩放,可选 fill:按使用目标尺寸
 * @param {String} options.type 图片压缩类型默认 jpg,可选 png
 * @param {Number} options.rotate 图片旋转,由于手机拍照的角度和我们使用的头像不一致,需要旋转 默认0 仅支持 90 180 -90
 * @returns {Promise} then {width,height,img}
 */
export function pictureCompress (options) {
  return new Promise((resolve, reject) => {
    if (!options.img) {
      reject(new Error('need img'))
      return
    }
    let imgSrc = options.img,
      width = options.width || 640,
      height = options.height || 640,
      type = options.type || 'jpg',
      quality = options.quality || 0.92,
      fit = options.fit || 'scale',
      rotate = options.rotate || 0
    if (width <= 0 || height <= 0) {
      reject(new Error('dist width or height need > 0'))
      return
    }
    if (!/jpg|png|jpeg/.test(type)) {
      reject(new Error('type need jpg or png!'))
      return
    }
    if (rotate !== 90 && rotate !== -90 && rotate !== 0 && rotate !== 180) {
      reject(new Error('rotate mast be 0 90 -90 180!'))
      return
    }
    let changeWidthAndHeight = rotate === 90 || rotate === -90
    let image = new Image()
    image.src = imgSrc
    image.onload = function () {
      let distSize = getDistSize({
        width: changeWidthAndHeight ? this.naturalHeight : this.naturalWidth,
        height: changeWidthAndHeight ? this.naturalWidth : this.naturalHeight
      }, {
        width: changeWidthAndHeight ? height : width,
        height: changeWidthAndHeight ? width : height
      }, fit)
      let imgData = compress(this, distSize.width, distSize.height, type, quality, rotate)
      resolve({
        width: distSize.width,
        height: distSize.height,
        img: imgData
      })
    }
    image.onerror = function (err) {
      reject(err)
    }
  })
}

/**
 * 将图片转换为固定尺寸的
 * @param {Image} img 图片数据
 * @param {Number} width 转换之后的图片宽度
 * @param {Number} height 转换之后的图片高度
 * @param {String} type base64的图片类型 jpg png
 * @param {Number} quality 转换之后的图片质量
 */
export function compress (img, width, height, type, quality, rotate) {
  var canvas = document.createElement('canvas')
  var ctx = canvas.getContext('2d')
  let types = {
    'jpg': 'image/jpeg',
    'jpeg': 'image/jpeg',
    'png': 'image/png'
  }
  canvas.width = width
  canvas.height = height
  ctx.drawImage(img, 0, 0, width, height)
  return canvas.toDataURL(types[type], quality)
}

/**
 * 选择源尺寸与目标尺寸比例中较小的那个,保证图片可以完全显示
 * 最大值不超过1,如果图片源尺寸小于目标尺寸,则不做处理,返回图片原尺寸
 * @param {Object} source 源图片的宽高
 * @param {Object} dist 目标图片的宽高
 */
export function getDistSize (source, dist, fit) {
  if (fit === 'fill') return dist
  let scale = Math.min(dist.width / source.width, dist.height / source.height, 1)
  return {
    width: Math.round(source.width * scale),
    height: Math.round(source.height * scale)
  }
}

// 将base64转换为file文件
export function dataURLtoFile (dataurl, filename) {
  let arr = dataurl.split(',')
  let mime = arr[0].match(/:(.*?);/)[1]
  let bstr = atob(arr[1])
  let n = bstr.length
  let u8arr = new Uint8Array(n)
  while (n--) {
    u8arr[n] = bstr.charCodeAt(n)
  }
  return new File([u8arr], filename, { type: mime })
}

下面是调用该方法:index.vue

import { pictureCompress , dataURLtoFile } from "./index.js"
afterRead (file) {
	 pictureCompress({
          img: file.content,  //上传的图片的base64
        }).then(res => {
          //console.log(res) 可自行输出res返回的参数数据
          //此处为将压缩过后的base64转换成file文件流格式
          console.log(dataURLtoFile(res.img,file.file.name))
      	})
}

到这里大家以为就能使用了吗?答案是可以的。但是!注意了canvas进行移动端手机照片上传时,发现IOS手机上传照片会逆时针旋转90度的问题。那么我们来看以下的代码如何解决的吧~~

二、使用Exif实现图片旋转

解决这个问题的思路是:获取到照片拍摄的方向角,对旋转后的IOS照片进行角度旋转修正。
这里主要用到Orientation属性(最常出现的几个参数)。说明如下:

旋转角度参数
1
180°3
顺时针90°6
逆时针90°8

1、安装Exif.js

cnpm install exif-js --save

2、引入并使用

import EXIF from 'exif-js'
  let orientation
  EXIF.getData(img, function () {
    orientation = Number(EXIF.getTag(this, 'Orientation'))
  })

那么到现在,就已经拿到了图片旋转角度的参数。就可以对上面初步封装的方法进行一个修改。

三、完整封装的JS方法

注:
1、因为是针对IOS机型的修改,所以增加了一层判断条件。详情看代码~

index.js

import EXIF from 'exif-js'

/**
 * 将图片压缩为对应尺寸
 * @param {Object} options
 * @param {String} options.img 图片的url或者base64数据
 * @param {Number} options.width 目标图片的宽度
 * @param {Number} options.height 目标图片的高度
 * @param {Number} options.quality 生成目标图片质量
 * @param {String} options.fit 图片压缩填充模式默认 scale:按比例缩放,可选 fill:按使用目标尺寸
 * @param {String} options.type 图片压缩类型默认 jpg,可选 png
 * @param {Number} options.rotate 图片旋转,由于手机拍照的角度和我们使用的头像不一致,需要旋转 默认0 仅支持 90 180 -90
 * @returns {Promise} then {width,height,img}
 */
export function pictureCompress (options) {
  return new Promise((resolve, reject) => {
    if (!options.img) {
      reject(new Error('need img'))
      return
    }
    let imgSrc = options.img,
      width = options.width || 640,
      height = options.height || 640,
      type = options.type || 'jpg',
      quality = options.quality || 0.92,
      fit = options.fit || 'scale',
      rotate = options.rotate || 0
    if (width <= 0 || height <= 0) {
      reject(new Error('dist width or height need > 0'))
      return
    }
    if (!/jpg|png|jpeg/.test(type)) {
      reject(new Error('type need jpg or png!'))
      return
    }
    if (rotate !== 90 && rotate !== -90 && rotate !== 0 && rotate !== 180) {
      reject(new Error('rotate mast be 0 90 -90 180!'))
      return
    }
    let changeWidthAndHeight = rotate === 90 || rotate === -90
    let image = new Image()
    image.src = imgSrc
    image.onload = function () {
      let distSize = getDistSize({
        width: changeWidthAndHeight ? this.naturalHeight : this.naturalWidth,
        height: changeWidthAndHeight ? this.naturalWidth : this.naturalHeight
      }, {
        width: changeWidthAndHeight ? height : width,
        height: changeWidthAndHeight ? width : height
      }, fit)
      let imgData = compress(this, distSize.width, distSize.height, type, quality, rotate)
      resolve({
        width: distSize.width,
        height: distSize.height,
        img: imgData
      })
    }
    image.onerror = function (err) {
      reject(err)
    }
  })
}

/**
 * 将图片转换为固定尺寸的
 * @param {Image} img 图片数据
 * @param {Number} width 转换之后的图片宽度
 * @param {Number} height 转换之后的图片高度
 * @param {String} type base64的图片类型 jpg png
 * @param {Number} quality 转换之后的图片质量
 */
export function compress (img, width, height, type, quality, rotate) {
  var canvas = document.createElement('canvas')
  var ctx = canvas.getContext('2d')
  let types = {
    'jpg': 'image/jpeg',
    'jpeg': 'image/jpeg',
    'png': 'image/png'
  }
  canvas.width = width
  canvas.height = height
  let orientation
  EXIF.getData(img, function () {
    orientation = Number(EXIF.getTag(this, 'Orientation'))
  })
  // 针对IOS系统上传照片会旋转
  if (!!navigator.userAgent.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/)) {
    if (orientation != "" && orientation != 1) {
      switch (orientation) {
        case 3:// 需要180度旋转
          rotateImg(img, 'right2', canvas, width, height);
          break;
        case 6:// 需要顺时针(向左)90度旋转
          rotateImg(img, 'left', canvas, width, height);
          break;
        case 8:// 需要逆时针(向右)90度旋转
          rotateImg(img, 'right', canvas, width, height);
          break;
        default:
          ctx.drawImage(img, 0, 0, width, height);
          break;
      }
    } else {
      ctx.drawImage(img, 0, 0, width, height)
    }
  } else {
    ctx.drawImage(img, 0, 0, width, height)
  }
  return canvas.toDataURL(types[type], quality)
}

export function rotateImg (img, direction, canvas, width, height) {
  var min_step = 0;
  var max_step = 3;
  if (img == null) {
    return;
  }
  var step = 2;
  if (step == null) {
    step = min_step;
  }
  if (direction == 'right') {
    step++;
    step > max_step && (step = min_step);
  } else if (direction == 'right2') {
    step = 2;
  } else {
    step--;
    step < min_step && (step = max_step);
  }
  var degree = step * 90 * Math.PI / 180;
  var ctx = canvas.getContext('2d');
  switch (step) {
    case 0:
      canvas.width = width;
      canvas.height = height;
      ctx.drawImage(img, 0, 0, width, height);
      break;
    case 1:
      canvas.width = height;
      canvas.height = width;
      ctx.rotate(degree);
      ctx.drawImage(img, 0, -height, width, height);
      break;
    case 2:
      canvas.width = width;
      canvas.height = height;
      ctx.rotate(degree);
      ctx.drawImage(img, -width, -height, width, height);
      break;
    case 3:
      canvas.width = height;
      canvas.height = width;
      ctx.rotate(degree);
      ctx.drawImage(img, -width, 0, width, height);
      break;
  }
}

/**
 * 选择源尺寸与目标尺寸比例中较小的那个,保证图片可以完全显示
 * 最大值不超过1,如果图片源尺寸小于目标尺寸,则不做处理,返回图片原尺寸
 * @param {Object} source 源图片的宽高
 * @param {Object} dist 目标图片的宽高
 */
export function getDistSize (source, dist, fit) {
  if (fit === 'fill') return dist
  let scale = Math.min(dist.width / source.width, dist.height / source.height, 1)
  return {
    width: Math.round(source.width * scale),
    height: Math.round(source.height * scale)
  }
}

// 将base64转换为file文件
export function dataURLtoFile (dataurl, filename) {
  let arr = dataurl.split(',')
  let mime = arr[0].match(/:(.*?);/)[1]
  let bstr = atob(arr[1])
  let n = bstr.length
  let u8arr = new Uint8Array(n)
  while (n--) {
    u8arr[n] = bstr.charCodeAt(n)
  }
  return new File([u8arr], filename, { type: mime })
}

index.vue

import { pictureCompress , dataURLtoFile } from "./index.js"
afterRead (file) {
	 pictureCompress({
          img: file.content,  //上传的图片的base64
        }).then(res => {
          //console.log(res) 可自行输出res返回的参数数据
          //此处为将压缩过后的base64转换成file文件流格式
          console.log(dataURLtoFile(res.img,file.file.name))
          //下面就是请求接口上传压缩后的图片代码位置↓
      	})
}

至此,图片上传前的压缩功能就写好了!
最后来复习一下:如果觉得作者写的不错,希望得到您的点赞/收藏/支持,如果有不同意见,欢迎下方评论区留言。

  • 4
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
部分使用说明: 点击开始菜单-运行命令,在对话框输入: "jhead –命令参数 jpeg文件" 例如: “jhead -de D:\A.jpg” 删除D盘盘根目录下A.jpg文件的exif信息。 “jhead -de D:\*.jpg” 删除D盘根目录下所有jpg文件的exif信息。其中星号是通配符。 二、通用指令参数 -te 将其他jpeg文件的eixf导入目标jpeg。例如"jhead –te D:\B.jpg D:\A.jpg" -dc 删除jpeg信息中的备注。注意,jpeg文件有两个备注,一是和其他文件一样的备注,另一个是exif信息中的备注。jhead仅对exif信息有效。 -de 完全删除exif信息。 -du 删除非原始exif信息,例如Photoshop、Turbophoto之类编辑后修改exif留下的信息。 -purejpg 删除所有jpeg文件非必须的信息。相当于-de、-dc和-du的集合,可以将文件减小数k。 -ce 修改文件的jpeg文件头部分备注(此备注并非exif信息)。该指令会打开文本编辑器,并在编辑器关闭时将备注信息存入文件。 -cs 导出备注。例如"jhead –cs D:\988.txt D:\A.jpg" -ci 导入备注。例如"jhead –ci D:\988.txt D:\A.jpg" -cl 直接输入备注。。例如"jhead –cl 我的备注 D:\A.jpg" 三、其他指令 时间日期 -ft 将jpeg文件的“修改时间”修改为exif信息中记录的时间。 -n[] 该指令会将文件名修改为exif信息中记录的“创建时间”;如果jpeg文件没有exif或者exif中的创建时间不可用,则将文件名修改为文件的“修改时间”。 默认的格式-顺序为MMDD-HHMMSS 格式-顺序参数如下: %d-日(01-31) %H-小时(00-23) %j-一年中的第几天(001-366) %m-月(01-12) %M-分钟(00-59) %S-秒(00-59) %U-一年中的第几周(00-53) %w-星期几(0-6,周日为0) %y-两位数纪年(00-99) %Y-四位数几年 %i-添加数字序号 例如: jhead -n%Y%m%d-%H%M%S d:\*.jpg 将所有jpg文件修改为YYYYMMDD-HHMMSS.jpg的格式。 -nf 与“-n”相同功能相同,不保留原文件名。 -a 修改不同扩展名的同名文件名,相机拍摄的avi短片exif信息存储在与其同名的thm文件中,可用此指令给avi文件更名。一般与“-n”指令共同使用。 -ta 修正时差,例如时差根据时区确定,例如+1:00或者-1:00 -da- 修正日期。日期格式是yyyy:mm:dd、yyyy:nn:dd+hh:mm或者 yyyy:mm:dd+hh:mm:ss。根据前后参数时间差调整exif的时间。 -ts 直接修改exif中的拍摄时间,日期-时间格式为yyyy:mm:dd-hh:mm:ss 缩略图 -dt 删除exif中的缩略图。这个缩略图一般为240x160像素,10k大小,用于数码相机、Windows XP查看照片,删除它不会影响工作。 -st 将exif中的缩略图复制为另一个jpeg文件 -rt 用另一个jpeg文件替换exif中的缩略图 -rgt[大小] 刷新exif缩略图,其中大小为缩略图的最大边长。 旋转 -autorot 根据exif中记录的水平方向信息转动jpeg照片。 -norot 清除exif中的水平方向信息。 四、使用技巧 1)用开始菜单的"运行"指令并不直观,可以通过运行cmd命令进入DOS命令提示符界面操作。在DOS界面进入操作照片文件夹(不懂DOS操作的朋友建议稍稍学习DOS指令,今后也会受用无穷),在文件夹中运行jhead命令,用“*.jpg”表示文件夹中所有的jpeg文件,可以进行批处理。 2)-te(复制exif信息)作用在于可以恢复被其他编辑软件删除的exif信息。编辑照片之前,先在照片文件夹中建立一个名为“backup”的备份文件夹,将原是照片复制到backup文件夹中,然后再编辑照片,编辑软件可能会删除或修改exif。编辑结束后进入DOS界面照片文件夹输入: jhead –te “backup\&i” *.jpg 照片的exif信息就会从backup文件夹的原始文件中复制回来。其中“&i”表示与目标文件同名的文件,前面的“backup\”表示原始文件位置。 3)-purejpg指令可以删除所有exif信息,让照片减小若干k字节

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值