一、坐标系

Android应用层坐标系原点在左上角,坐标范围(0,0)——(width,height)。

Android底层坐标系原点在屏幕中央,坐标范围(-1000,,1000)——(1000,1000)。

二、Scale缩放

Matrix3*3的矩阵结构如下

 {MSCALE_X,MSKEW_X,MTRANS_X,
MSKEW_Y,MSCALE_Y,MTRANS_Y,
MPERSP_0,MPERSP_1,MPERSP_2}

scale就是缩放,我们调用Matrix的setScale、preScale、postScale,实际在内部,就是通过修改MSCALE_X和MSCALE_Y来实现的。

rotate,数值为正,顺时针旋转;数值为负,逆时针旋转

 Matrix matrix = new Matrix();

             // 缩放图片动作
matrix.postScale(scaleWidth, scaleHeight);
// 创建新的图片
Bitmap resizedBitmap = Bitmap.createBitmap(bitmapOrg, 0, 0,
width, height, matrix, true); i1.setImageBitmap(resizedBitmap); // 缩放图片动作
matrix.postScale(scaleWidth2, scaleHeight2);
// 创建新的图片
resizedBitmap = Bitmap.createBitmap(bitmapOrg2, 0, 0,
width2, height2, matrix, true); i2.setImageBitmap(resizedBitmap);

而后对原矩阵进行矩阵相乘完成缩放。

三、Translate

matrix.setScale(interpolatedTime, interpolatedTime);
 matrix.preTranslate(-centerX, -centerY);

matrix.postTranslate(centerX, centerY);

preTranslate是指在setScale前,平移,postTranslate是指在setScale后平移

注意他们参数是平移的距离,而不是平移目的地的坐标!

由于缩放是以(0,0)为中心的,所以为了把界面的中心与(0,0)对齐,就要preTranslate(-centerX, -centerY),

四、set....、pre....、post...的区别

  1. setScale(sx,sy),首先会将该Matrix设置为对角矩阵,即相当于调用reset()方法,然后在设置该Matrix的MSCALE_X和MSCALE_Y直接设置为sx,sy的值
  2. preScale(sx,sy),不会重置Matrix,而是直接与Matrix之前的MSCALE_X和MSCALE_Y值结合起来(相乘),M' = M * S(sx, sy)。
  3. postScale(sx,sy),不会重置Matrix,而是直接与Matrix之前的MSCALE_X和MSCALE_Y值结合起来(相乘),M' = S(sx, sy) * M。

(1)pre的执行顺序

 Matrix matrix=new Matrix();  

 float[] points=new float[]{10.0f,10.0f};  

 matrix.preScale(2.0f, 3.0f);//
matrix.preTranslate(8.0f,7.0f);// matrix.mapPoints(points); Log.i("test", points[0]+""); Log.i("test", points[1]+"");

结果为点坐标为(36.0,51.0)

可以得出结论,进行变换的顺序是先执行preTranslate(8.0f,7.0f),在执行的preScale(2.0f,3.0f)。这也是为什么有的人比喻为pre...是向后生长的,即对于一个Matrix的设置中,

所有pre....是倒着向后执行的。

(2)post的执行顺序

 Matrix matrix=new Matrix();  

 float[] points=new float[]{10.0f,10.0f};  

 matrix.postScale(2.0f, 3.0f);//  

 matrix.postTranslate(8.0f,7.0f);//  

 matrix.mapPoints(points);  

 Log.i("test", points[0]+"");  

 Log.i("test", points[1]+"");  

结果为点坐标为(28.0,37.0)

可以得出结论,进行变换的顺序是先执行postScale(2.0f,3.0f),在执行的postTranslate(8.0f,7.0f)。这 也是为什么有的人比喻为post...是向前生长的,即对于一个Matrix的设置中,所有post....是顺着向前执行的。

(3)pre与post交替

 Matrix matrix=new Matrix();  

 float[] points=new float[]{10.0f,10.0f};  

 matrix.postScale(2.0f, 3.0f);  

 matrix.preRotate(90);  

 matrix.mapPoints(points);  

 Log.i("test", points[0]+"");  

 Log.i("test", points[1]+"");  

结果为点坐标为(-20.0,30.0)。先旋转方向再缩放。

 Matrix matrix=new Matrix();  

 float[] points=new float[]{10.0f,10.0f};  

 matrix.preRotate(90);  

 matrix.postScale(2.0f, 3.0f);  

 matrix.mapPoints(points);  

 Log.i("test", points[0]+"");  

 Log.i("test", points[1]+"");  

结果为点坐标依旧为为(-20.0,30.0)

(4)

 Matrix matrix = new Matrix();  

 float[] points = new float[] { 10.0f, 10.0f };  

 matrix.postScale(2.0f, 3.0f);// 第1步  

 matrix.preRotate(90);// 第2步  

 matrix.postTranslate(8.0f, 7.0f);// 第3步  

 matrix.preScale(1.5f, 2.5f);// 第4步  

 matrix.mapPoints(points);  

 Log.i("test", points[0] + "");  

 Log.i("test", points[1] + "");  

结果为点坐标依旧为为(-42.0,52.0)
经过前面的结论和推算,可以发现执行的顺序是   4----2----1---3

(5)

 Matrix matrix = new Matrix();  

 float[] points = new float[] { 10.0f, 10.0f };  

 matrix.postScale(2.0f, 3.0f);// 第1步  

 matrix.preRotate(90);// 第2步  

 matrix.setScale(1.4f, 2.6f);// 第3步  

 matrix.postTranslate(8.0f, 7.0f);// 第4步  

 matrix.preScale(1.5f, 2.5f);// 第5步  

 matrix.mapPoints(points);  

 Log.i("test", points[0] + "");  

 Log.i("test", points[1] + "");  

结果为点坐标依旧为为(29.0,72.0)
经过计算,可以发现,在第3步setScale之前的第1、2步根本就没有用了,直接被第3步setScale覆盖,在从第3开始执行的。

顺序为2---1----3----5----4,因为2、1被覆盖了,所以没有效果,相当于直接执行3-----5----4

总结:最后可以得出结论,在对matrix该次变换之前的所有设置中,先检测有没有setScale,如果有,直接跳到setScale那一步开始 执行变换,然后在倒着执行下面所有的pre...变换,在顺着执行所有post....的变换。所以在对Matrix变换设置的时候,一定要注意顺序,不 同的顺序,会有不同的结果。

参考:http://blog.csdn.net/u010838555/article/details/44307615

http://blog.csdn.net/yhjhl520/article/details/8063191

http://www.cnblogs.com/anee/archive/2012/10/18/2729145.html

Android——坐标系及转化的更多相关文章

  1. Android 坐标系和 MotionEvent 分析、滑动

    1.Android坐标系 在Android中,屏幕最左上角的顶点作为Android坐标系的原点,这个点向左是X轴正方向,这个点向下是Y轴正方向. 系统提供了getLocationOnScreen(in ...

  2. Android 时间戳简单转化

    时间戳就是如1377216000000 这种格式我们在mysql数据库中会经常用到把时间转换成时间戳或把时间戳转换成日期格式了,下面我来介绍安卓中时间戳操作转换方法. 一.原理 时间戳的原理是把时间格 ...

  3. Android dp px转化公式

    // DisplayMetrics metrics = getResources().getDisplayMetrics(); // int statusBarHeight = (int) Math. ...

  4. android中的坐标系以及获取坐标的方法

    android中有两种坐标系,分别称之为Android坐标系和视图坐标系.而对应的也有一些相关的方法可以获取坐标系中的 坐标值.只有搞清楚这些区别,才能在实现的时候不至于出错或者得不到你想要的效果. ...

  5. Android View体系(一)视图坐标系

    前言 Android View体系是界面编程的核心,他的重要性不亚于Android四大组件,在这个系列中我会陆续讲到View坐标系.View的滑动.View的事件分发等文章来逐步介绍Android V ...

  6. Android Scroll分析——滑动效果产生

    相对于在Android2.x版本上出现的长按.点击事件的效果,不得不说,滑动操作具有更好的用户体验.因此,从Android 4.X版本开始,出现了更多滑动操作的效果.越来越多第三方应用模仿这样的效果, ...

  7. Android坐标系统

     1 背景 去年有很多人私信告诉我让说说自定义控件,其实通观网络上的很多博客都在讲各种自定义控件,但是大多数都是授之以鱼,却很少有较为系统性授之于渔的文章,同时由于自己也迟迟没有时间规划这一系列文章, ...

  8. view坐标_ _ Android应用坐标系统全面详解

    转:http://blog.csdn.net/yanbober/article/details/50419117 1 背景 去年有很多人私信告诉我让说说自定义控件,其实通观网络上的很多博客都在讲各种自 ...

  9. android相关技能

    深读: 如:View.ViewGroup.AdapterView.ListView.GridView.Window.ViewDragHelper.ItemTouchHelper.SurfaceView ...

随机推荐

  1. css 标题

    纯CSS制作的复古风格的大标题 .vintage{ background: #EEE url( ...

  2. 【转】OPC远程访问相关配置信息

    原文:http://blog.gkong.com/kking_25653.ashx 对于远程访问OPC服务器,需要在客户和服务器计算机上都进行DCOM设置,本文提供一些具体配置方法.(by Kevin ...

  3. C#中类的实例是不能 获取到类中的静态方法和静态变量(Static)的,及原因

    类中的静态方法和变量是共享的.只能用类名去调用.

  4. SPOJ - ADAFIELD ,Set+map,STL不会超时!

    ADAFIELD - Ada and Field 这个题,如果用一个字来形容的话:-----------------------------------------------嗯! 题意:n*m的空白 ...

  5. vue 自定义日历组件

    <template> <div class=""> <div class="calendarTraffic" name=" ...

  6. [BZOJ1604] [Usaco2008 Open]Cow Neighborhoods 奶牛的邻居(好题)

    传送门 良心题解 #include <set> #include <cstdio> #include <iostream> #include <algorit ...

  7. 算法复习——单调队列(sliding windows,ssoi)

    题目: 题目描述 给你一个长度为 N 的数组,一个长为 K 的滑动的窗体从最左移至最右端,你只能见到窗口的 K 个整数,每次窗体向右移动一位,如下表:

  8. 【(最小权点基)tarjan强连通分量缩点+tarjan模板】HDU 5934 Bomb

    [AC] #include<bits/stdc++.h> using namespace std; typedef long long ll; int n; ; ; const int i ...

  9. 【NOI2012】骑行川藏

    获得成就:第一次在信竞做神仙数学题 先放个前言,$OI$ 出大型数学题还是比较麻烦的,因为主要是考你数学推导 / 手算式子,你算出来之后把公式套个板子,就得到结论——$OI$ 的大型数学题的代码都是板 ...

  10. poj1717 Dominoes (背包)

    A domino is a flat, thumbsized tile, the face of which is divided into two squares, each left blank ...