详解利用exif.js解决ios手机上传竖拍照片旋转90度问题


Posted in Javascript onNovember 04, 2016

HTML5+canvas进行移动端手机照片上传时,发现iOS手机上传竖拍照片会逆时针旋转90度,横拍照片无此问题;Android手机没这个问题。

因此解决这个问题的思路是:获取到照片拍摄的方向角,对非横拍的ios照片进行角度旋转修正。

利用exif.js读取照片的拍摄信息,这里主要用到Orientation属性。

Orientation属性说明如下:详解利用exif.js解决ios手机上传竖拍照片旋转90度问题

下面就直接上代码了。

主要有html5页面和一个js,示例功能包含了图片压缩和旋转。

自己写的是uploadImage.js。

html5测试页面如下:

<!DOCTYPE html> 
<html> 
<head> 
  <meta charset="utf-8"> 
  <meta name="viewport" content="initial-scale=1.0, maximum-scale=1.0, user-scalable=no" /> 
  <title>图片上传</title> 
  <script type="text/javascript" src="js/jquery-1.8.3.js"></script> 
  <script type="text/javascript" src="js/uploadPicture/uploadImage.js" ></script> 
    <script type="text/javascript" src="js/exif.js" ></script> 
  <script> 
 
  </script> 
</head> 
<body> 
  <div style="height: 50px; line-height: 50px;text-align: center;border-bottom: 1px solid #171E28;"> 
      上传图片: 
      <input type="file" accept="image/*" id="uploadImage" capture="camera" onchange="selectFileImage(this);" /> 
    </div> 
    <div style="margin-top: 10px;"> 
      <img alt="preview" src="" id="myImage"/> 
    </div> 
</body> 
</html>

uploadImage.js如下:

function selectFileImage(fileObj) { 
  var file = fileObj.files['0']; 
  //图片方向角 added by lzk 
  var Orientation = null; 
   
  if (file) { 
    console.log("正在上传,请稍后..."); 
    var rFilter = /^(image\/jpeg|image\/png)$/i; // 检查图片格式 
    if (!rFilter.test(file.type)) { 
      //showMyTips("请选择jpeg、png格式的图片", false); 
      return; 
    } 
    // var URL = URL || webkitURL; 
    //获取照片方向角属性,用户旋转控制 
    EXIF.getData(file, function() { 
      // alert(EXIF.pretty(this)); 
      EXIF.getAllTags(this);  
      //alert(EXIF.getTag(this, 'Orientation'));  
      Orientation = EXIF.getTag(this, 'Orientation'); 
      //return; 
    }); 
     
    var oReader = new FileReader(); 
    oReader.onload = function(e) { 
      //var blob = URL.createObjectURL(file); 
      //_compress(blob, file, basePath); 
      var image = new Image(); 
      image.src = e.target.result; 
      image.onload = function() { 
        var expectWidth = this.naturalWidth; 
        var expectHeight = this.naturalHeight; 
         
        if (this.naturalWidth > this.naturalHeight && this.naturalWidth > 800) { 
          expectWidth = 800; 
          expectHeight = expectWidth * this.naturalHeight / this.naturalWidth; 
        } else if (this.naturalHeight > this.naturalWidth && this.naturalHeight > 1200) { 
          expectHeight = 1200; 
          expectWidth = expectHeight * this.naturalWidth / this.naturalHeight; 
        } 
        var canvas = document.createElement("canvas"); 
        var ctx = canvas.getContext("2d"); 
        canvas.width = expectWidth; 
        canvas.height = expectHeight; 
        ctx.drawImage(this, 0, 0, expectWidth, expectHeight); 
        var base64 = null; 
        //修复ios 
        if (navigator.userAgent.match(/iphone/i)) { 
          console.log('iphone'); 
          //alert(expectWidth + ',' + expectHeight); 
          //如果方向角不为1,都需要进行旋转 added by lzk 
          if(Orientation != "" && Orientation != 1){ 
            alert('旋转处理'); 
            switch(Orientation){ 
              case 6://需要顺时针(向左)90度旋转 
                alert('需要顺时针(向左)90度旋转'); 
                rotateImg(this,'left',canvas); 
                break; 
              case 8://需要逆时针(向右)90度旋转 
                alert('需要顺时针(向右)90度旋转'); 
                rotateImg(this,'right',canvas); 
                break; 
              case 3://需要180度旋转 
                alert('需要180度旋转'); 
                rotateImg(this,'right',canvas);//转两次 
                rotateImg(this,'right',canvas); 
                break; 
            }     
          } 
           
          /*var mpImg = new MegaPixImage(image); 
          mpImg.render(canvas, { 
            maxWidth: 800, 
            maxHeight: 1200, 
            quality: 0.8, 
            orientation: 8 
          });*/ 
          base64 = canvas.toDataURL("image/jpeg", 0.8); 
        }else if (navigator.userAgent.match(/Android/i)) {// 修复android 
          var encoder = new JPEGEncoder(); 
          base64 = encoder.encode(ctx.getImageData(0, 0, expectWidth, expectHeight), 80); 
        }else{ 
          //alert(Orientation); 
          if(Orientation != "" && Orientation != 1){ 
            //alert('旋转处理'); 
            switch(Orientation){ 
              case 6://需要顺时针(向左)90度旋转 
                alert('需要顺时针(向左)90度旋转'); 
                rotateImg(this,'left',canvas); 
                break; 
              case 8://需要逆时针(向右)90度旋转 
                alert('需要顺时针(向右)90度旋转'); 
                rotateImg(this,'right',canvas); 
                break; 
              case 3://需要180度旋转 
                alert('需要180度旋转'); 
                rotateImg(this,'right',canvas);//转两次 
                rotateImg(this,'right',canvas); 
                break; 
            }     
          } 
           
          base64 = canvas.toDataURL("image/jpeg", 0.8); 
        } 
        //uploadImage(base64); 
        $("#myImage").attr("src", base64); 
      }; 
    }; 
    oReader.readAsDataURL(file); 
  } 
} 
 
//对图片旋转处理 added by lzk 
function rotateImg(img, direction,canvas) {  
    //alert(img); 
    //最小与最大旋转方向,图片旋转4次后回到原方向  
    var min_step = 0;  
    var max_step = 3;  
    //var img = document.getElementById(pid);  
    if (img == null)return;  
    //img的高度和宽度不能在img元素隐藏后获取,否则会出错  
    var height = img.height;  
    var width = img.width;  
    //var step = img.getAttribute('step');  
    var 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);  
    }  
    //img.setAttribute('step', step);  
    /*var canvas = document.getElementById('pic_' + pid);  
    if (canvas == null) {  
      img.style.display = 'none';  
      canvas = document.createElement('canvas');  
      canvas.setAttribute('id', 'pic_' + pid);  
      img.parentNode.appendChild(canvas);  
    } */ 
    //旋转角度以弧度值为参数  
    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);  
        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;  
    }  
  }

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

Javascript 相关文章推荐
input按钮的事件处理大全
Dec 10 Javascript
浅说js变量
May 25 Javascript
JS的replace方法介绍
Oct 20 Javascript
java与javascript之间json格式数据互转介绍
Oct 29 Javascript
两个数组去重的JS代码
Dec 04 Javascript
分享10个原生JavaScript技巧
Apr 20 Javascript
javascript实现动态导入js与css等静态资源文件的方法
Jul 25 Javascript
详解JavaScript中的4种类型识别方法
Sep 14 Javascript
js实现简单计算器
Nov 22 Javascript
javaScript 事件绑定、事件冒泡、事件捕获和事件执行顺序整理总结
Oct 10 Javascript
JavaScript中如何使用cookie实现记住密码功能及cookie相关函数介绍
Nov 10 Javascript
js制作可以延时消失的菜单
Jan 13 Javascript
jQuery中 $ 符号的冲突问题及解决方案
Nov 04 #Javascript
JAVA Web实时消息后台服务器推送技术---GoEasy
Nov 04 #Javascript
详解js运算符单竖杠“|”与“||”的用法和作用介绍
Nov 04 #Javascript
使用Ajax与服务器(JSON)通信实例
Nov 04 #Javascript
AngularJS过滤器filter用法实例分析
Nov 04 #Javascript
URL的参数中有加号传值变为空格的问题(URL特殊字符)
Nov 04 #Javascript
Angular页面间切换及传值的4种方法
Nov 04 #Javascript
You might like
php的一个简单加密解密代码
2014/01/14 PHP
CodeIgniter中使用cookie的三种方式详解
2014/07/18 PHP
64位windows系统下安装Memcache缓存
2015/12/06 PHP
php die()与exit()的区别实例详解
2016/12/03 PHP
php抽象类和接口知识点整理总结
2019/08/02 PHP
Laravel框架数据库迁移操作实例详解
2020/04/06 PHP
jQuery学习总结之jQuery事件
2014/06/30 Javascript
Node.js抓取中文网页乱码问题和解决方法
2015/02/10 Javascript
JavaScript判断用户是否对表单进行了修改的方法
2015/03/18 Javascript
简介JavaScript中search()方法的使用
2015/06/06 Javascript
jQuery ajaxSubmit 实现ajax提交表单局部刷新
2016/07/04 Javascript
JavaScript仿微博输入框效果(案例分析)
2016/12/06 Javascript
js实现短信发送倒计时功能(正则验证)
2017/02/10 Javascript
JS 60秒后重新发送验证码的实例讲解
2017/07/26 Javascript
初学者AngularJS的环境搭建过程
2017/10/27 Javascript
[07:08]2014DOTA2西雅图国际邀请赛 小组赛7月11日TOPPLAY
2014/07/11 DOTA
python PIL模块与随机生成中文验证码
2016/02/27 Python
python计算两个矩形框重合百分比的实例
2018/11/07 Python
python自动化测试之如何解析excel文件
2019/06/27 Python
Ubuntu+python将nii图像保存成png格式
2019/07/18 Python
python 给图像添加透明度(alpha通道)
2020/04/09 Python
让IE支持CSS3的不完全兼容方案
2014/09/19 HTML / CSS
JD Sports瑞典:英国领先的运动时尚商店
2018/01/28 全球购物
美国唇部护理专家:Sara Happ
2019/06/19 全球购物
如何在Cookie里面保存Unicode和国际化字符
2013/05/25 面试题
个人简历自我评价
2014/02/02 职场文书
客服部班长工作责任制
2014/02/25 职场文书
二手房买卖协议书
2014/04/10 职场文书
大学优秀班集体申报材料
2014/05/23 职场文书
5s标语大全
2014/06/23 职场文书
学雷锋志愿者活动方案
2014/08/21 职场文书
12.4全国法制宣传日活动总结
2014/11/01 职场文书
组织委员竞选稿
2015/11/21 职场文书
幼师自荐信范文(2016推荐篇)
2016/01/28 职场文书
css3中transform属性实现的4种功能
2021/08/07 HTML / CSS
漫画《尖帽子的魔法工坊》宣布动画化
2022/04/06 日漫