一、坐标系

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. 用java Graphics生成验证码

    以下下是API文档对Graphics的介绍! Graphics 类是所有图形上下文的抽象基类,允许应用程序在组件(已经在各种设备上实现)以及闭屏图像上进行绘制. Graphics 对象封装了 Java ...

  2. 爬虫cookie

    Cookie Cookie 是指某些网站服务器为了辨别用户身份和进行Session跟踪,而储存在用户浏览器上的文本文件,Cookie可以保持登录信息到用户下次与服务器的会话. Cookie原理 HTT ...

  3. Django中的app及mysql数据库篇(ORM操作)

    Django常见命令 在Django的使用过程中需要使用命令让Django进行一些操作,例如创建Django项目.启动Django程序.创建新的APP.数据库迁移等. 创建Django项目 一把我们都 ...

  4. Arthas诊断工具使用资料

    1.https://github.com/alibaba/arthas/issues/327 2.https://alibaba.github.io/arthas/jad.html 3.https:/ ...

  5. Knockout v3.4.0 中文版教程-7-计算监控-依赖跟踪如何工作

    3.依赖跟踪如何工作 初学者不需要知道这一点,但更高级的开发人员将想知道为我们怎么实现KO自动跟踪依赖性和自动更新UI的正确部分... 它其实相当简单优雅,跟踪算法如下: 当你定义一个计算监控,KO立 ...

  6. 从士兵到程序员再到 SOHO 程序员 (三) - 游击战与阻力

    从士兵到程序员再到 SOHO 程序员 (三) - 游击战与阻力 原文地址:http://blog.huhao.name/blog/2014/03/01/become-a-freelancer-3/ 作 ...

  7. sql自动审核工具-inception

    [inception使用规范及说明文档](http://mysql-inception.github.io/inception-document/usage/)[代码仓库](https://githu ...

  8. [Vijos1308]埃及分数(迭代加深搜索 + 剪枝)

    传送门 迭代加深搜索是必须的,先枚举加数个数 然后搜索分母 这里有一个强大的剪枝,就是确定分母的范围 #include <cstdio> #include <cstring> ...

  9. 【边双连通】poj 3352 Road Construction

    http://poj.org/problem?id=3352 [题意] 给定一个连通的无向图,求最少加多少条边使得这个图变成边双连通图 [AC] //#include<bits/stdc++.h ...

  10. 转载:shell中awk printf的用法

    转载:http://www.linuxawk.com/jiaocheng/83.html 6. printf函数   打印输出时,可能需要指定字段间的空格数,从而把列排整齐.在print函数中使用制表 ...