vue2实现移动端上传、预览、压缩图片解决拍照旋转问题

因为最近遇到个移动端上传头像的需求,上传到后台的数据是base64位,其中为了提高用户体验,把比较大的图片用canvas进行压缩之后再进行上传。在移动端调用拍照功能时,会发生图片旋转,为了解决这个问题引入了exif去判断拍照时的信息再去处理图片,这是个很好的插件。关于exif.js可以去他的GitHub上了解,这边直接 npm install exif-js --save   安装,然后import一下就可以使用了。以下就是源码,可以直接使用。

<template>
 <div>
 <div style="padding:20px;">
 <div class="show">
 <div class="picture" :style="'backgroundImage:url('+headerImage+')'"></div>
 </div>
 <div style="margin-top:20px;">
 <input type="file" id="upload" accept="image" @change="upload">
 <label for="upload"></label>
 </div>
 </div>
 </div>
</template>
<script>
import Exif from 'exif-js'
export default {
 data () {
 return {
 headerImage:'',
 }
 },
 mounted () {
 },
 methods: {
 upload (e) {
 let files = e.target.files || e.dataTransfer.files;
 if (!files.length) return;
 this.picValue = files[0];
 this.imgPreview(this.picValue);
 },
 imgPreview (file) {
 let self = this;
 let Orientation;
 //去获取拍照时的信息,解决拍出来的照片旋转问题
 Exif.getData(file, function(){
  Orientation = Exif.getTag(this, 'Orientation');
 });
 // 看支持不支持FileReader
 if (!file || !window.FileReader) return;
 if (/^image/.test(file.type)) {
  // 创建一个reader
  let reader = new FileReader();
  // 将图片2将转成 base64 格式
  reader.readAsDataURL(file);
  // 读取成功后的回调
  reader.onloadend = async function () {
  let result = this.result;
  let img = new Image();
  img.src = result;
  //判断图片是否大于100K,是就直接上传,反之压缩图片
  if (this.result.length <= (100 * 1024)) {
  self.headerImage = this.result;
  self.postImg();
  }else {
  img.onload = function () {
  let data = self.compress(img,Orientation);
  self.headerImage = data;
  self.postImg();
  }
  }
  }
 }
 },
 postImg () {
 //这里写接口
 },
 rotateImg (img, direction,canvas) {
 //最小与最大旋转方向,图片旋转4次后回到原方向
 const min_step = 0;
 const max_step = 3;
 if (img == null)return;
 //img的高度和宽度不能在img元素隐藏后获取,否则会出错
 let height = img.height;
 let width = img.width;
 let step = 2;
 if (step == null) {
  step = min_step;
 }
 if (direction == 'right') {
  step++;
  //旋转到原位置,即超过最大值
  step > max_step && (step = min_step);
 } else {
  step--;
  step < min_step && (step = max_step);
 }
 //旋转角度以弧度值为参数
 let degree = step * 90 * Math.PI / 180;
 let ctx = canvas.getContext('2d');
 switch (step) {
  case 0:
  canvas.width = width;
  canvas.height = height;
  ctx.drawImage(img, 0, 0);
  break;
  case 1:
  canvas.width = height;
  canvas.height = width;
  ctx.rotate(degree);
  ctx.drawImage(img, 0, -height);
  break;
  case 2:
  canvas.width = width;
  canvas.height = height;
  ctx.rotate(degree);
  ctx.drawImage(img, -width, -height);
  break;
  case 3:
  canvas.width = height;
  canvas.height = width;
  ctx.rotate(degree);
  ctx.drawImage(img, -width, 0);
  break;
 }
 },
 compress(img,Orientation) {
 let canvas = document.createElement("canvas");
 let ctx = canvas.getContext('2d');
 //瓦片canvas
 let tCanvas = document.createElement("canvas");
 let tctx = tCanvas.getContext("2d");
 let initSize = img.src.length;
 let width = img.width;
 let height = img.height;
 //如果图片大于四百万像素,计算压缩比并将大小压至400万以下
 let ratio;
 if ((ratio = width * height / 4000000) > 1) {
 console.log("大于400万像素")
 ratio = Math.sqrt(ratio);
 width /= ratio;
 height /= ratio;
 } else {
 ratio = 1;
 }
 canvas.width = width;
 canvas.height = height;
 // 铺底色
 ctx.fillStyle = "#fff";
 ctx.fillRect(0, 0, canvas.width, canvas.height);
 //如果图片像素大于100万则使用瓦片绘制
 let count;
 if ((count = width * height / 1000000) > 1) {
 console.log("超过100W像素");
 count = ~~(Math.sqrt(count) + 1); //计算要分成多少块瓦片
 //  计算每块瓦片的宽和高
 let nw = ~~(width / count);
 let nh = ~~(height / count);
 tCanvas.width = nw;
 tCanvas.height = nh;
 for (let i = 0; i < count; i++) {
  for (let j = 0; j < count; j++) {
  tctx.drawImage(img, i * nw * ratio, j * nh * ratio, nw * ratio, nh * ratio, 0, 0, nw, nh);
  ctx.drawImage(tCanvas, i * nw, j * nh, nw, nh);
  }
 }
 } else {
 ctx.drawImage(img, 0, 0, width, height);
 }
 //修复ios上传图片的时候 被旋转的问题
 if(Orientation != "" && Orientation != 1){
 switch(Orientation){
  case 6://需要顺时针(向左)90度旋转
  this.rotateImg(img,'left',canvas);
  break;
  case 8://需要逆时针(向右)90度旋转
  this.rotateImg(img,'right',canvas);
  break;
  case 3://需要180度旋转
  this.rotateImg(img,'right',canvas);//转两次
  this.rotateImg(img,'right',canvas);
  break;
 }
 }
 //进行最小压缩
 let ndata = canvas.toDataURL('image/jpeg', 0.1);
 console.log('压缩前:' + initSize);
 console.log('压缩后:' + ndata.length);
 console.log('压缩率:' + ~~(100 * (initSize - ndata.length) / initSize) + "%");
 tCanvas.width = tCanvas.height = canvas.width = canvas.height = 0;
 return ndata;
 },
 }
}
</script>
<style>
*{
 margin: 0;
 padding: 0;
}
.show {
 width: 100px;
 height: 100px;
 overflow: hidden;
 position: relative;
 border-radius: 50%;
 border: 1px solid #d5d5d5;
}
.picture {
 width: 100%;
 height: 100%;
 overflow: hidden;
 background-position: center center;
 background-repeat: no-repeat;
 background-size: cover;
}
</style>

以上所述是小编给大家介绍的vue2实现移动端上传、预览、压缩图片解决拍照旋转问题,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • require.js+vue开发微信上传图片组件

    由于项目是thinkPHP做后端框架,一直以来都是多页面的后端路由,想使用火热的webpack有点无从下手(原谅我太菜,而且推广vue只有我一个人--),没办法,想把vue用起来,唯有在原来的基础上改进.使用webpack的巨大好处就是可以使用 .vue 这样的单文件来写vue组件,这样每一个组件就是一个 .vue 文件,哪里用上这个组件就引入进来,维护起来确实很爽.然而一直以来项目用的都是require.js,那又想以这样的形式来组织vue组件,还要加上vue-router和vue-resou

  • 基于VUE选择上传图片并页面显示(图片可删除)

    基于VUE选择上传图片并在页面显示,图片可删除,具体内容如下 demo例子: 依赖文件:jqueryform HTML文本内容: <template> <div id="accident"> <div class="wrapper"> <i class="icon-pic"></i>相关照片 <button type="button" @click="

  • Vue.js 2.0 移动端拍照压缩图片上传预览功能

    在学习和使用Vue.js 2.0 的过程中遇到不少不一样的地方,本来移动端开发H5应用,准备将mui框架和Vue.js+vue-router+vuex 全家桶结合起来使用,但是在拍照上传的实现过程中遇到了无法调用plus的H5+接口的问题,所以最后拍照上传功能还是使用input file方式里解决的.但是内心还是不甘心的,由于项目进度推进,迭代版本,所以不得不放弃,后续可能我将此功能使用调用H5+接口实现. 首先我来讲我实现这个拍照预览压缩上传的思路,准确的说应该是拍照或选择图片压缩之后预览及上

  • 使用vue构建一个上传图片表单

    这篇博客也不知道起个什么名字比较好,毕竟刚开始学习vue,很多还不是很熟悉,一直在慢慢摸索中:之前也习惯了用jQuery写js代码,思维逻辑也要有个转换的过程. 1. 转变思维 使用vue编写代码,首先要做的就是转换思维,vue是一个数据驱动的框架,我们只需要操作数据,数据变化后,vue会自动改变DOM结构,而jQuery是直接操作DOM的.比如刚开始写的代码中犯的错误,有一个页面中的input标签是并列多个的,而且name属性的值是自增的,那么我就用v-for循环下,把index填充进去就行了

  • vue-quill-editor实现图片上传功能

    问题描述   项目使用的vue2.0开发,项目中需要一个富文本编辑器,楼主经过一番心理挣扎选择了vue-quill-editor.具体如何引用作者在项目中已经写得很明白了,我在这里就不再赘述.   vue-quill-editor插入图片的方式是将图片转为base64再放入内容中,这样就会产生一个问题,如果图片比较大的话,富文本的内容就会很大,楼主是将内容存在数据库中的,这样一来,一方面会占用大量的数据库存储空间,另一方面当图片太大的时候富文本的内容,会超过数据库的存储上限,从而会产生内容被截断

  • 基于vue+ bootstrap实现图片上传图片展示功能

    效果图如下所示: html ..... ....... <-- key=idPicUrl --> <div class="col-sm-7" > <img :src="queryFirmInfo[key]" alt="" style="max-height:200px;max-width:250px" class="myimage" :name="key"

  • Vue.js 2.0 移动端拍照压缩图片预览及上传实例

    在学习和使用Vue.js 2.0 的过程中遇到不少不一样的地方,本来移动端开发H5应用,准备将mui框架和Vue.js+vue-router+vuex 全家桶结合起来使用,但是在拍照上传的实现过程中遇到了无法调用plus的H5+接口的问题,所以最后拍照上传功能还是使用input file方式里解决的.但是内心还是不甘心的,由于项目进度推进,迭代版本,所以不得不放弃,后续可能我将此功能使用调用H5+接口实现. 首先我来讲我实现这个拍照预览压缩上传的思路,准确的说应该是拍照或选择图片压缩之后预览及上

  • vue上传图片组件编写代码

    本文实例教大家如何编写一个vue上传图片组件,具体如下 1.首先得有一个[type=file]文件标签并且隐藏,changge事件来获取图片: <input @change="fileChange($event)" type="file" id="upload_file" multiple style="display: none"/> 2.触发隐藏的文件标签:(通过原生的click来触发) document.ge

  • vue.js 上传图片实例代码

    最近爱上了用vue.js做前端,昨天用vue上传图片时遇到了问题,最后半天时间终于摸索出来,我将相关部分的代码贴出来. 前端部分 <div class="form-group"> <label>背景图</label> <input type="file" class="form-control" @change="onFileChange"> </div> <d

  • vuejs开发组件分享之H5图片上传、压缩及拍照旋转的问题处理

    一.前言 三年.net开发转前端已经四个月了,前端主要用webpack+vue,由于后端转过来的,前端不够系统,希望分享下开发心得与园友一起学习. 图片的上传之前都是用的插件(ajaxupload),或者传统上传图片的方式,各有利弊:插件的问题是依赖jq并且会使系统比较臃肿,还有传统的web开发模式 前后端偶尔在一起及对用户体验要求低,现在公司采用webpack+vue+restfullApi开发模式 前后端完全分离,遵从高内聚,低偶尔的原则,开发人员各司其职,一则提升开发效率(从长期来看,短期

随机推荐