但是,当我们把上传的图片转换成base64格式,发送给后台时,会发现偶尔会出现问题,有一些图片本来是这样的:
柴犬
处理之后却变成了这样:
柴犬2
经过测试发现,只有iOS手机竖着拍的照片才会出现这样的问题,而iOS手机横着拍的照片、Android手机拍的照片以及通过屏幕截图、网络下载等途径获得的图片都不会产生这个问题。
那么,这到底是为什么呢?
在开发过程中,由于时间紧迫,未求甚解,使用了github上的一个开源项目 lrz.js 来解决此问题,这个工具的主要用途是在尽量保证图片质量的前提下压缩图片的大小,但同时也附带了图片旋转角度纠正的功能。
通过阅读 lrz.js 的源代码,我发现它引入了一个叫做 exif.js 的库来实现旋转角度的纠正,它提供了js读取图像的原始数据的功能扩展,例如:拍照方向、相机设备型号、拍摄时间、ISO 感光度、GPS 地理位置等数据。而拍照方向就是关键所在!
exif.js 获取图像的拍照方向的代码如下:
EXIF.getData(IMG_FILE, function () { // IMG_FILE为图像数据
var orientation = EXIF.getTag(this, "Orientation")
console.log("Orientation:" + orientation)// 拍照方向
})
获取拍照方向的结果为1-8的数字:
拍照方向信息
注意:对于上面的八种方向中,加了*的并不常见,因为它们代表的是镜像方向,如果不做任何的处理,不管相机以任何角度拍摄,都无法出现镜像的情况。
这个表格代表什么意义?我们来看第一行,值为1时,右边两列的值分别为:Row #0 is Top,Column #0 is Left side,其实很好理解,它表示照片的第一行位于顶端,而第一列位于左侧,那么这张照片自然就是以正常角度拍摄的。
而这8种结果,就是第一行与第一列所在的位置的8种组合。
那么,我们来测试一下iOS手机横着拍的照片,来看看它的拍照方向是什么呢?
测试1
结果是1,即以正常角度拍摄的,其实也就是原图啦~
那么,我们再测试一下iOS手机竖着拍的照片,来看看它的拍照方向是什么呢?
测试2
原来是6!即第一行位于右侧,第一列位于顶端,其实相当于将照片顺时针旋转了90度!
所以,实际上iOS手机竖着拍出的照片与横着拍出的照片其本质上是一样的,只不过竖着拍出的照片被添加了一个顺时针旋转90°的拍照方向,所以显示的时候,就变成了上下边窄左右边宽的状态,其实也就是横着拍的照片顺时针旋转90°而成的~
那么明白了这些,文章开头所说的照片旋转bug的原因,也就很简单啦~
其实就是当我们在前端对图片进行像素处理或者drawInRect等 *** 作之后,照片的Orientaion信息,即为拍照方向信息被删除了,所以iOS手机竖着拍的照片又回到了横着的状态,看起来也就是逆时针旋转了90°!
那么如何纠正这个旋转角度呢?
其实思路也很简单:在处理图片之前,先读取并保存图片的拍照方向信息,然后在处理图片之后,再根据拍照方向,对图片进行相应的调整,lrz.js 中的代码如下:
switch (orientation) {
case 3:
ctx.rotate(180 * Math.PI / 180)
ctx.drawImage(img, -resize.width, -resize.height, resize.width, resize.height)
break
case 6:
ctx.rotate(90 * Math.PI / 180)
ctx.drawImage(img, 0, -resize.width, resize.height, resize.width)
break
case 8:
ctx.rotate(270 * Math.PI / 180)
ctx.drawImage(img, -resize.height, 0, resize.height, resize.width)
break
case 2:
ctx.translate(resize.width, 0)
ctx.scale(-1, 1)
ctx.drawImage(img, 0, 0, resize.width, resize.height)
break
case 4:
ctx.translate(resize.width, 0)
ctx.scale(-1, 1)
ctx.rotate(180 * Math.PI / 180)
ctx.drawImage(img, -resize.width, -resize.height, resize.width, resize.height)
break
case 5:
ctx.translate(resize.width, 0)
ctx.scale(-1, 1)
ctx.rotate(90 * Math.PI / 180)
ctx.drawImage(img, 0, -resize.width, resize.height, resize.width)
break
case 7:
ctx.translate(resize.width, 0)
ctx.scale(-1, 1)
ctx.rotate(270 * Math.PI / 180)
ctx.drawImage(img, -resize.height, 0, resize.height, resize.width)
break
default:
ctx.drawImage(img, 0, 0, resize.width,resize.height)
}
其中,translate是平移变换,scale(-1,1)是向左翻转,rotate是顺时针旋转。
举例说明 case 2,当图片的拍照方向为2时,即第一行位于顶端,而第一列位于右侧,其实相当于把照片进行了左右的翻转。所以,这里对图片的 *** 作是,先向右平移等于图片宽度的距离,再向左翻转,这相当于以图片水平方向的对称轴为轴进行了左右翻转,然后再以(0,0)为起始点绘制原宽高的图片,即完成了对拍照方向的纠正。
最后
经过一系列的测试,发现确实只有iOS手机的竖拍照片与横拍照片是通过拍照方向来区别的,Android手机无论竖拍还是横拍的照片,拍照方向都为1,也就是说即使丢失了拍照方向这一信息,也不会影响到图片的旋转角度。而手机或电脑的屏幕截图、网络上的图片、通过PS制作的图片等也是如此。
作者:任无名F
链接:http://www.jianshu.com/p/ad4501db178e
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
我这里认为大家都稍微了解甚至熟悉canvas的一些API,就不具体说,每一个参数代表什么意思了。<!DOCTYPE html>
<html>
<head>
<meta charset='utf-8'>
<title>图片加载平移放大缩小示例</title>
<style>
html,body{
margin:0px
padding:0px
}
canvas{
border: 1px solid #000
}
</style>
</head>
<body>
<canvas id="canvas" width="800" height="800"></canvas>
<script type="text/javascript" src="main.js"></script>
</body>
</html>
var canvas,context
function int(){
canvas=document.getElementById('canvas')
context=canvas.getContext('2d')
}
图片加载
创建一个图片对象之后,图片不能马上绘制到canvas上面,因为图片还没有加载完成。所以我们需要监听图片对象加载完事件,然后再去绘制。
var img,//图片对象
imgIsLoaded//图片是否加载完成
function loadImg(){
img=new Image()
img.onload=function(){
imgIsLoaded=true
//draw image
}
img.src="map.jpg"
}
图片绘制
绘制图像一个函数就可以搞定,但是需要记录这个图像的左上角坐标以及缩放比例。
var imgX,imgY,imgScale
function drawImage(){
context.clearRect(0,0,canvas.width,canvas.height)
context.drawImage(img,0,0,img.width,img.height,imgX,imgY,img.width*imgScale,img.height*imgScale)
}
图片平移
html5事件最小细度在DOM上,所以我们无法对canvas上的图像做监听,只能对canvas监听。
首先监听鼠标mousedown事件,等事件发生之后,再监听鼠标mousemove事件和mouseup事件
mousemove事件发生之后,获得鼠标移动的位移,相应的图片的位置改变多少
mouseup事件发生之后,取消对mousemove以及mouseup事件监听
canvas.onmousedown=function(event){
var pos=windowToCanvas(canvas,event.clientX,event.clientY)
canvas.onmousemove=function(event){
canvas.style.cursor="move"
var pos1=windowToCanvas(canvas,event.clientX,event.clientY)
var x=pos1.x-pos.x
var y=pos1.y-pos.y
pos=pos1
imgX+=x
imgY+=y
drawImage()
}
canvas.onmouseup=function(){
canvas.onmousemove=null
canvas.onmouseup=null
canvas.style.cursor="default"
}
}
function windowToCanvas(canvas,x,y){
var bbox = canvas.getBoundingClientRect()
return {
x:x - bbox.left - (bbox.width - canvas.width) / 2,
y:y - bbox.top - (bbox.height - canvas.height) / 2
}
}
图片缩放
其实缩放很简单,稍微复杂的是,如何让鼠标成为放大或者缩小的中心。如果数学几何不好,计算公式就可能看不明白了。
canvas.onmousewheel=canvas.onwheel=function(event){//chrome firefox浏览器兼容
var pos=windowToCanvas(canvas,event.clientX,event.clientY)
event.wheelDelta=event.wheelDelta?event.wheelDelta:(event.deltaY*(-40))
if(event.wheelDelta>0){
imgScale*=2
imgX=imgX*2-pos.x
imgY=imgY*2-pos.y
}else{
imgScale/=2
imgX=imgX*0.5+pos.x*0.5
imgY=imgY*0.5+pos.y*0.5
}
drawImage()
}
这个时候,基本功能就实现了,加载一张图片和加载多张图片都差不多,维护每一张图片的位置和大小,下面来整理一下代码吧。
var canvas,context
var img,//图片对象
imgIsLoaded,//图片是否加载完成
imgX=0,
imgY=0,
imgScale=1
(function int(){
canvas=document.getElementById('canvas')
context=canvas.getContext('2d')
loadImg()
})()
function loadImg(){
img=new Image()
img.onload=function(){
imgIsLoaded=true
drawImage()
}
img.src="map.jpg"
}
function drawImage(){
context.clearRect(0,0,canvas.width,canvas.height)
context.drawImage(img,0,0,img.width,img.height,imgX,imgY,img.width*imgScale,img.height*imgScale)
}
canvas.onmousedown=function(event){
var pos=windowToCanvas(canvas,event.clientX,event.clientY)
canvas.onmousemove=function(event){
canvas.style.cursor="move"
var pos1=windowToCanvas(canvas,event.clientX,event.clientY)
var x=pos1.x-pos.x
var y=pos1.y-pos.y
pos=pos1
imgX+=x
imgY+=y
drawImage()
}
canvas.onmouseup=function(){
canvas.onmousemove=null
canvas.onmouseup=null
canvas.style.cursor="default"
}
}
canvas.onmousewheel=canvas.onwheel=function(event){
var pos=windowToCanvas(canvas,event.clientX,event.clientY)
event.wheelDelta=event.wheelDelta?event.wheelDelta:(event.deltaY*(-40))
if(event.wheelDelta>0){
imgScale*=2
imgX=imgX*2-pos.x
imgY=imgY*2-pos.y
}else{
imgScale/=2
imgX=imgX*0.5+pos.x*0.5
imgY=imgY*0.5+pos.y*0.5
}
drawImage()
}
function windowToCanvas(canvas,x,y){
var bbox = canvas.getBoundingClientRect()
return {
x:x - bbox.left - (bbox.width - canvas.width) / 2,
y:y - bbox.top - (bbox.height - canvas.height) / 2
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)