android 图片旋转

作者:王娇       

       在Android中进行图像旋转需要使用Matrix,它包含了一个3*3的矩阵,专门用于进行图像变换匹配。Matrix ,中文里叫矩阵,高等数学里有介绍,在图像处理方面,主要是用于平面的缩放、平移、旋转等操作。Matrix没有机构体,它必须初始化,然后通过reset方法和set方法来实现。

        首先介绍一下矩阵运算。加法和减法就不用说了,太简单了,对应位相加就好。图像处理,主要用到的是乘法 。下面是一个乘法的公式:


在 Android 里面, Matrix 由 9 个 float 值构成,是一个 3*3 的矩阵。如下图。



没专业工具,画的挺难看。解释一下,上面的 sinX 和 cosX ,表示旋转角度的 cos 值和 sin 值,注意,旋转角度是按顺时针方向计算的。 translateX 和 translateY 表示 x 和 y 的平移量。 scale 是缩放的比例, 1 是不变, 2 是表示缩放 1/2,这样子。

Matrix的操作,总共分为translate(平移),rotate(旋转),scale(缩放)和skew(倾斜)四种,每一种变换在Android的API里都提供了set,post和pre三种操作方式,除了translate,其他三种操作都可以指定中心点。set是直接设置Matrix的值,每次set一次,整个Matrix的数组都会变掉。

我们现在通过setRotate设置旋转角度,用creatBitmap创建一个经过旋转等处理的Bitmap对象,然后将Bitmap绘制到屏幕之上,于是就实现了旋转操作。

下面使用一个示例来说明Matix的使用以及旋转的方式及运行效果。


package cn.edu.pku;

import android.app.Activity;
import android.os.Bundle;
import android.view.KeyEvent;

public class PictureRotateActivity extends Activity {
    /** Called when the activity is first created. */
	
	private GameRotateView1 gameview = null;
	
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        gameview = new GameRotateView1(this);  
        setContentView(gameview); 
    }
	@Override
	public boolean onKeyDown(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		if ( gameview == null )
		{
			return false;
		}
		if ( keyCode ==  KeyEvent.KEYCODE_BACK)
		{
			this.finish();
			return true;
		}
		return gameview.onKeyDown(keyCode,event);
	}
	@Override
	public boolean onKeyUp(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		super.onKeyUp(keyCode, event);
		return true;
	} 
}


具体图像旋转处理代码如下:


package cn.edu.pku;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.drawable.BitmapDrawable;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.View;

public class GameRotateView1 extends View implements Runnable {

	Bitmap bitmap = null;
	int bitmapWidth = 0;
	int bitmapHeight = 0;
	
	float angle = 0.0f;
	
	Matrix matrix = new Matrix();
	
	public GameRotateView1(Context context) {
		super(context);
		// TODO Auto-generated constructor stub
		
		setFocusableInTouchMode(true); //设置可以捕捉键盘事件
		//获取图像资源
		bitmap = ((BitmapDrawable) getResources().getDrawable(R.drawable.cute)).getBitmap();
		bitmapWidth = bitmap.getWidth();
		bitmapHeight = bitmap.getHeight();
		new Thread(this).start();
	}

	public void run() {
		// TODO Auto-generated method stub
		while(!Thread.currentThread().isInterrupted()){
			try{
				Thread.sleep(100);
			}catch (InterruptedException e) {
				// TODO: handle exception
				Thread.currentThread().interrupt();
			}
			postInvalidate();  //可以直接在线程中更新界面
		}
	}

	@Override
	protected void onDraw(Canvas canvas) {
		// TODO Auto-generated method stub
		super.onDraw(canvas);
		
		matrix.reset();
		matrix.setRotate(angle); //设置旋转
		
		//按照matrix的旋转构建新的Bitmap
		Bitmap bitmapcute = Bitmap.createBitmap(bitmap, 0, 0, bitmapWidth, bitmapHeight, matrix, true);
		
		//绘制旋转之后的图像
		GameRotateView1.DrawImage(canvas, bitmapcute, (320 - bitmapWidth)/2, 10);
		bitmapcute = null;
	}

	@Override
	public boolean onKeyDown(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		if(keyCode == KeyEvent.KEYCODE_DPAD_LEFT){
			angle--;
		}else if(keyCode == KeyEvent.KEYCODE_DPAD_RIGHT){
			angle++;
		}
		return true;
	}
	
	@Override
	public boolean onTouchEvent(MotionEvent event) {
		// TODO Auto-generated method stub
		return true;
	}

	
	@Override
	public boolean onKeyUp(int keyCode, KeyEvent event) {
		// TODO Auto-generated method stub
		return false;
	}

	
	
	@Override
	public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) {
		// TODO Auto-generated method stub
		return true;
	}

	/**
	 * 绘制一个Bitmap
	 * canvas	画布
	 * bitmap	图片
	 * x			屏幕上的x坐标
	 * y			屏幕上的y坐标
	 */
	public static void DrawImage(Canvas canvas, Bitmap _bitmap, int x, int y)
	{
		/* 绘制图像 */
		canvas.drawBitmap(_bitmap, x, y, null);
	}


最后我们通过键盘的左右键可以实现图像的选装,在这里实现的图像的右旋转:





评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值