图片操作系列 —(2)手势旋转图片

前言

在上次的文章:图片操作系列 —(1)手势缩放图片功能中,我们已经学会了如何用手势来对图片进行缩放。这次我们继续来看第二个操作,那就是如何用手势来旋转图片。

所以我们本文我们一共要实现二个功能:

  1. 根据二个手指头的旋转来使图片跟着旋转
  2. 当二个手指头放开后,图片会自动回归到合适的位置。

我说明下第二个功能点的意思:什么叫回归到合适的位置,比如如图一,我们只转动了一点点,没有超过45度,然后放在手指,然后就会回到图二的样子。但是如果超过了45度,然后放开手指,就回变成图三的样子。

图一

图二

图三

前面基本的东西说明我都不说了。比如Matrix等知识。大家可以直接参考图片操作系列 —(1)手势缩放图片功能

ps:我这边可以再贴出相关基础的链接:
android matrix 最全方法详解与进阶(完整篇)
Android Matrix


根据二个手指头的旋转来使图片跟着旋转:

我们知道使图片进行旋转特定的角度很简单:

使用Matrix.postRotate(float degrees, float px, float py)方法即可。绕着(px,py)点进行旋转degrees角度。

所以我们的问题就变成了如果获取二个手指头在做旋转手势的时候,相应的角度的变化,从而通过Matrix.postRotate方法来让图片也跟着变化。

1.获取二个手指头的手势监听

图片操作系列 —(1)手势缩放图片功能文中我们知道,控制图片的缩放是专门有个ScaleGestureDetector;在OnTouch事件中把相应的事件传递给ScaleGestureDetector。然后监听处理。我们也可以模仿着写一个RotateGestureDetector来进行图片旋转的监听和处理。

public interface IRotateDetector {

    /**
     * handle rotation in onTouchEvent
     *
     * @param event The motion event.
     * @return True if the event was handled, false otherwise.
     */
    boolean onTouchEvent(MotionEvent event);
    
        /**
     * is the Gesture Rotate
     *
     * @return true:rotating;false,otherwise
     */
    boolean isRotating();
}

public class RotateGestureDetector implements IRotateDetector{
    
    private int mLastAngle = 0;//最后一次的角度值
    private IRotateListener mListener;//用来旋转的回调Listener
    private boolean mIsRotate;//是否处于旋转
    
    //用来设置回调Listener的方法
    public void setRotateListener(IRotateListener listener) {
        this.mListener = listener;
    }
    
    //用来接收触摸事件
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        return doRotate(event);
    }
    
    //真正的计算手势操作所得到的角度值的方法,及回调调用。
    private boolean doRotate(MotionEvent ev) {
        if (ev.getPointerCount() != 2) {
            return false;
        }
        //Calculate the angle between the two fingers
        int pivotX = (int) (ev.getX(0) + ev.getX(1)) / 2;
        int pivotY = (int) (ev.getY(0) + ev.getY(1)) / 2;
        float deltaX = ev.getX(0) - ev.getX(1);
        float deltaY = ev.getY(0) - ev.getY(1);

        double radians = Math.atan(deltaY / deltaX);
      
        int degrees = (int) Math.round(Math.toDegrees(Math.atan2(deltaY,deltaX)));
       
        switch (ev.getActionMasked()) {
            case MotionEvent.ACTION_DOWN:
                mLastAngle = degrees;
                mIsRotate = false;
                break;
            case MotionEvent.ACTION_UP:
                mIsRotate = false;
                break;
            case MotionEvent.ACTION_POINTER_DOWN:
                mLastAngle = degrees;
                mIsRotate = false;
                break;
            case MotionEvent.ACTION_CANCEL:
            case MotionEvent.ACTION_POINTER_UP:
                mIsRotate = false;
                upRotate(pivotX, pivotY);
                mLastAngle = degrees;
                break;
            case MotionEvent.ACTION_MOVE:
                mIsRotate = true;
                int degreesValue = degrees  - mLastAngle;
                if (degreesValue > 45) {
                    //Going CCW across the boundary
                    rotate(-5, pivotX, pivotY);
                } else if (degreesValue < -45) {
                    //Going CW across the boundary
                    rotate(5, pivotX, pivotY);
                } else {
                    //Normal rotation, rotate the difference
                    rotate(degreesValue, pivotX, pivotY);
                }
                //Save the current angle
                mLastAngle = degrees;
                break;
        }
        return true;
    }
    
    //回调的方法之一:控制图片根据手势的变化实时进行旋转
    private void rotate(int degree, int pivotX, int pivotY) {
        if (mListener != null) {
            mListener.rotate(degree, pivotX, pivotY);
        }
    }

    //回调的方法之一:最后某个手指放开后,控制图片自动回归到合适的位置。
    private void upRotate(int pivotX, int pivotY) {
        if (mListener != null) {
            mListener.upRotate(pivotX, pivotY);
        }
    }

}

2.获取二个手指头的角度变化

所以我们只需要来分析一下具体OnTouch事件中的doRotate方法即可:

//真正的计算手势操作所得到的角度值的方法,及回调调用。
private boolean doRotate(MotionEvent ev) {
    //如果触摸的手指头不是2个,直接返回。
    if (ev.getPointerCount() != 2) {
        return false;
    }
    
    //获取二个手指头的中心点的X与Y值,等会选择二个手指头的中心点作为旋转的中心
    int pivotX = (int) (ev.getX(0) + ev.getX(1)) / 2;
    int pivotY = (int) (ev.getY(0) + ev.getY(1)) / 2;
    //获取二个手指头之间的X和Y的差值
    float deltaX = ev.getX(0) - ev.getX(1);
    float deltaY = ev.getY(0) - ev.getY(1);
    //获取角度
    int degrees = (int) Math.round(Math.toDegrees(Math.atan2(deltaY,deltaX)));
   
    switch (ev.getActionMasked()) {
        case MotionEvent.ACTION_DOWN:
            mLastAngle = degrees;
            mIsRotate = false;
            break;
        case MotionEvent.ACTION_UP:
            mIsRotate = false;
            break;
        case MotionEvent.ACTION_POINTER_DOWN:
            mLastAngle = degrees;
            mIsRotate = false;
            break;
        case MotionEvent.ACTION_CANCEL:
        case MotionEvent.ACTION_POINTER_UP:
            mIsRotate = false;
            upRotate(pivotX, pivotY);
            mLastAngle = degrees;
            break;
        case MotionEvent.ACTION_MOVE:
            mIsRotate = true;
            /*
            每次把上一次的角度赋值给mLastAngle,然后获取当前新获取的角度degrees,
            二者相减获取到二个手指头在移动的时候相应的角度变化。
            */
            int degreesValue = degrees  - mLastAngle;
            
            /*
            这里主要出现这么个情况,二个手指头如果相隔有一段距离,那么在移动的过程中,角度不会一下子变化很大.
            但是比如我们这里故意二个手指头是碰在一起的,然后二个手指头稍微动一下,你就会发现角度变化会很大。
            这样图片就会瞬间也旋转了很大的角度,让人体验感觉很怪,所以我们这里瞬间顺时针或者逆时针超过45度,都只移动5度值。
            */
            if (degreesValue > 45) {
                rotate(-5, pivotX, pivotY);
            } else if (degreesValue < -45) {
                rotate(5, pivotX, pivotY);
            } else {
                rotate(degreesValue, pivotX, pivotY);
            }
            //Save the current angle
            mLastAngle = degrees;
            break;
    }
    return true;
}

doRotate方法中最主要的就是根据二个手指头触摸获取到的X,Y的差值,根据Math.atan2来获取到角度。我们具体来看下为什么这样可以来获取角度:

先附上一个基础概念:Math.atan与Math.atan2

image

假设我们先点击了(50,50),再点击(10,10),这时候我们的deltaX = 40,deltaY = 40;也就是说
我们的弧度就是Math.atan2(40,40),而角度就是再用Math.toDegrees对弧度进行转换即可。最终获得额角度是45度。

我们可以通过图形来查看为什么Math.atan2(40,40)对应的角度是45度。


image

如果我们的第二个手指头从(10,10)移动到了(50,10),也就是说最后变成了Math.atan2(40,0),根据图形来看我们就知道是:

image

所以一共旋转了45度,所以我们的图片也跟着顺时针旋转45度即可。

那假如我们的二个手指头的放入顺序反过来,变成:

image

那这时候就变成了Math.atan2(-40,-40),我们根据图形就知道了角度:

image

这时候还是跟刚才一样的操作,把(10,10)这个点移动到了(50,10),那这时候就是Math.atan2(-40,0);

image

所以最终得到的旋转的角度是(-135)-(-90) = 45度,所以最终也是顺时针旋转45度。所以我们不管是哪个手指头先放下都不影响结果。

也许有人就会问了,你这边按照二个手指的中点作为旋转中心去旋转,岂不是会旋转超出原来的图片的边界。如果你还记得我们上一篇文章:图片操作系列 —(1)手势缩放图片功能,这篇文章最后的内容讲的就是当图片超过边界,如果能随着手势慢慢回到边界里面:checkMatrixBounds()

3.在Activity中设置Listener来进行图片的旋转

然后我们只需要在相应的Activity处对回调回来的(degreesValue, pivotX, pivotY)三个值做相应的旋转即可。

rotateGestureDetector.setRotateListener(new IRotateListener() {
    @Override
    public void rotate(int degree, int pivotX, int pivotY) {
        //图片跟着手势进行旋转
        mSuppMatrix.postRotate(degree, pivotX, pivotY);
        //Post the rotation to the image
        checkAndDisplayMatrix();
    }

    @Override
    public void upRotate(int pivotX, int pivotY) {
        
        //当手指头松开的时候,让图片自动更新到合适的位置。
        float[] v = new float[9];
        mSuppMatrix.getValues(v);
        // calculate the degree of rotation
        int angle = (int) Math.round(Math.toDegrees(Math.atan2(v[Matrix.MSKEW_Y], v[Matrix.MSCALE_X])));

        mRightAngleRunnable = new RightAngleRunnable(angle, pivotX, pivotY);
        photoView.post(mRightAngleRunnable);
    }
});

手指头松开手图片自动旋转到合适位置:

我们知道,前面图片跟着旋转,是获取到了(int degree, int pivotX, int pivotY)这三个值,然后让mSuppMatrix.postRotate(degree, pivotX, pivotY);那我们就当手指头松开的时候,获取到最终这个图片比原来变化了多少角度即可。然后根据这个当前最终图片的变化角度来进行适当的旋转,让其旋转到合适位置。

我们来具体看怎么实现的:

@Override
public void upRotate(int pivotX, int pivotY) {
    
    //当手指头松开的时候,让图片自动更新到合适的位置。
    float[] v = new float[9];
    mSuppMatrix.getValues(v);
    // calculate the degree of rotation
    int angle = (int) Math.round(Math.toDegrees(Math.atan2(v[Matrix.MSKEW_Y], v[Matrix.MSCALE_X])));

    mRightAngleRunnable = new RightAngleRunnable(angle, pivotX, pivotY);
    photoView.post(mRightAngleRunnable);
}

Matrix,中文里叫矩阵,高等数学里有介绍,在图像处理方面,主要是用于平面的缩放、平移、旋转等操作。在Android里面,Matrix由9个float值构成,是一个3*3的矩阵。最好记住。如下图:


image

image
image

我们发现mSuppMatrix.getValues(v)方法返回的9个float值中,第一个为cosX,第四个为sinX,所以我们就取下标为0和3的值,也就是MSCALE_X和MSKEW_Y。我们用Math.atan2(v[Matrix.MSKEW_Y], v[Matrix.MSCALE_X])来获取弧度。再用Math.toDegrees来获取相应的最终图片的旋转的度数。

public class Matrix {

    public static final int MSCALE_X = 0;   //!< use with getValues/setValues
    public static final int MSKEW_X  = 1;   //!< use with getValues/setValues
    public static final int MTRANS_X = 2;   //!< use with getValues/setValues
    public static final int MSKEW_Y  = 3;   //!< use with getValues/setValues
    public static final int MSCALE_Y = 4;   //!< use with getValues/setValues
    public static final int MTRANS_Y = 5;   //!< use with getValues/setValues
    public static final int MPERSP_0 = 6;   //!< use with getValues/setValues
    public static final int MPERSP_1 = 7;   //!< use with getValues/setValues
    public static final int MPERSP_2 = 8;   //!< use with getValues/setValues
    
    ......
    ......
    ......
}
    

然后我们再把获取到的角度和中心点,通过一个Runnable来进行图片最后的矫正:

mRightAngleRunnable = new RightAngleRunnable(angle, pivotX, pivotY);
photoView.post(mRightAngleRunnable);

我们知道最后是RightAngleRunnable来进行图片的矫正,所以我们具体来分析下这个Runnable:

class RightAngleRunnable implements Runnable {
        private static final int RECOVER_SPEED = 4;
        private int mOldDegree;
        private int mNeedToRotate;
        private int mRoPivotX;
        private int mRoPivotY;

        RightAngleRunnable(int degree, int pivotX, int pivotY) {
            Log.v("dyp4", "oldDegree:" + degree + "," + "calDegree:" + calDegree(degree));
            this.mOldDegree = degree;
            this.mNeedToRotate = calDegree(degree);
            this.mRoPivotX = pivotX;
            this.mRoPivotY = pivotY;
        }

        //最终计算需要矫正的角度值
        /*
            例如:
            
            比如最终是60度,这时候其实是超过了45度,应该矫正成90度,
            所以最终要多给它30度。顺时针多选择30度。这里计算会得到30。
            
            比如如果是-60度,这时候应该是变成-90读,所以我们逆时针多旋转30度。
            这时候计算会得到-30。
            
            如果是20度,这时候没有超过45度,所以应该矫正成0度,
            所以最终要逆时针转回20度,所以这里计算会得到-20。
            
            如果是-120度,这时候要变成-90度,所以要顺时针转回30度,
            所以计算会得到30。
        */
        private int calDegree(int oldDegree) {
            int N = Math.abs(oldDegree) / 45;
            if ((0 <= N && N < 1) || 2 <= N && N < 3) {
                return -oldDegree % 45;
            } else {
                if (oldDegree < 0) {
                    return -(45 + oldDegree % 45);
                } else {
                    return (45 - oldDegree % 45);
                }
            }
        }
        
        
        /*
        我们上面的calDegree方法可以获得我们需要矫正的角度,但是我们不是一下子就让图片选择N度,而是慢慢的转过来。
        比如我们用RECOVER_SPEED = 4,4度的慢慢来旋转过来,不会给用户很突兀的感觉。
        */
        @Override
        public void run() {
            if (mNeedToRotate == 0) {
                return;
            }
            if (photoView == null) {
                return;
            }
            if (mNeedToRotate > 0) {
                //Clockwise rotation
                if (mNeedToRotate >= RECOVER_SPEED) {
                    mSuppMatrix.postRotate(RECOVER_SPEED, mRoPivotX, mRoPivotY);
                    mNeedToRotate -= RECOVER_SPEED;
                } else {
                    mSuppMatrix.postRotate(mNeedToRotate, mRoPivotX, mRoPivotY);
                    mNeedToRotate = 0;
                }
            } else if (mNeedToRotate < 0) {
                //Counterclockwise rotation
                if (mNeedToRotate <= -RECOVER_SPEED) {
                    mSuppMatrix.postRotate(-RECOVER_SPEED, mRoPivotX, mRoPivotY);
                    mNeedToRotate += RECOVER_SPEED;
                } else {
                    mSuppMatrix.postRotate(mNeedToRotate, mRoPivotX, mRoPivotY);
                    mNeedToRotate = 0;
                }
            }


            checkAndDisplayMatrix();
            Compat.postOnAnimation(photoView, this);
        }
    }

结尾

还是老样子,希望大家不要吐槽。有问题留言哈哈。。O(∩_∩)O哈哈~
PS:有好的画图软件介绍吗。。求介绍o(╥﹏╥)o

附上Demo地址:ScaleImageVewDemo(已经把图片旋转的Activity demo 加入里面)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • 效果图: Github链接:https://github.com/boycy815/PinchImageView ...
    CQ_TYL阅读 2,212评论 0 0
  • 概述 项目开发中,大家APP开发一般都会用到上传图片,比如是上传了自己的生活照,然后在某个界面处查看上传的图片,这...
    青蛙要fly阅读 6,193评论 2 20
  • 手势图片控件 PinchImageView 点击图片框架 photoView packagecom.example...
    Ztufu阅读 720评论 0 1
  • 源码地址 实现原理概览 我们要实现手指控制图片的平移、旋转、缩放,首先得知道手指做了什么动作,比如用户两指间距离是...
    篱开罗阅读 5,842评论 15 12
  • 没有选择权,就意味着没有自由。如果父母什么都替孩子拿主意、做选择,孩子就不能根据自己的内心和感觉去做事。孩...
    佛山大树阅读 226评论 0 0