Android OpenGL ES(六)----进入三维在代码中创建投影矩阵和旋转矩阵

我们现在准备好在代码中添加透视投影了。Android的Matrix类为它准备了两个方法------frustumM()和perspectiveM()。不幸的是,frustumM()的个缺陷,它会影响某些类型的投影,而perspectiveM()只是从Android的ICS版本开始才被引入,在早期的Android版本里并没有这个方法。我们可以简单地支持ICS及其以上的版本,但是这样会丢掉很大一部分市场,一些用户依然运行早期的Android版本。

作为替代,我们可以创建我们自己的方法来实现投影矩阵。

1.创建自己的perspectiveM

在工具包中创建新的类MatrixHelper,在开始处加入如下方法签名:

public static void perspectiveM(float[] m,float yFovInDegress,float aspect,float n,float f){

计算焦距

我们要做的第一件事就是计算焦距,这将基于在Y轴上的视野。就在方法签名之后加入如下代码:

final float angleInRadians=(float)(yFovInDegress*Math.PI/180.0);

final float a=(float)(1.0/Math.tan(angleInRadians/2.0));

我们使用Java的Math类计算那个正切函数,因为它需要弧度角,所以我们把视野从度转换为弧度。接着计算焦距。

输出矩阵

我们现在可以更具上一节的数学证明直接写出矩阵,增加如下代码:

m[0]=a/aspect;

m[1]=0f;

m[2]=0f;

m[3]=0f;

m[4]=0f;

m[5]=a;

m[6]=0f;

m[7]=0f;

m[8]=0f;

m[9]=0f;

m[10]=-((f+n)/(f-n));

m[11]=-1f;

m[12]=0f;

m[13]=0f;

m[14]=-((2f*f*n)/(f-n));

m[15]=0f;

这就是把矩阵数据存到了参数m定义的浮点数组中,这个数组需要至少16个元素。OpenGL把矩阵数据按照以列为主的顺序存储,这就意味着我们一次写一列数据,而不是一次写一行。前四个值是第一列,下一组四个数是第二列,以此类推。

2.开始使用投影矩阵

我们现在将转而使用那个透视投影矩阵了。打开你的渲染类,并从onSurfaceChanged()去掉所有的代码,只保留glViewPort()调用,加入如下代码:

MatrixHelper.perspectiveM(projectionMatrix, 45, (float) width / (float) height, 1f, 10f);

这会用45度的视野创建一个透视投影。 这个视椎体从Z值为-1的位置开始,在Z值为-10的位置结束。

加入MatrixHelper的导入后,继续前面的程序运行,你可以会发现桌面不见了,因为我们没有给桌子指定Z的位置,默认情况下Z在为0的位置。因为这个视椎体是从Z值为-1的位置开始的,除非把它移动到那个距离内,否则我们无法看到桌子。

不要硬编码Z的值,在使用投影矩阵进行投影之前,让我们使用一个平移矩阵把桌子移出来。依照惯例,我们把这个矩阵称为模型矩阵。

利用模型矩阵移动物体

在类的顶部加入如下矩阵的定义:

private final float[] modelMatrix=new float[16];

我们要使用这个矩阵把桌面移动到那个距离内。在onSurfaceChanged()结尾处,加入如下代码:

Matrix.setIdentityM(modelMatrix, 0);

Matrix.translateM(modelMatrix,0,0f,0f,-2f);

这就是把模型矩阵设为单位矩阵,在沿着Z轴平移-2。当我们把桌面的坐标与这个矩阵相乘的时候,那些坐标最终会沿着Z轴负方向移动2个单位。

相乘一次还是相乘两次

我们现在要做一个选择:我们依然需要把这个矩阵应用于每个顶点,因此第一个选项是给顶点着色器新增一个额外的矩阵。我们把每个顶点都与这个模型矩阵相乘,让它们沿着Z轴负方向移动2个单位,接下来把每个顶点与投影矩阵相乘。这样,OpenGL就可以做透视除法,并把这些顶点变换到归一化设备坐标了。

如果不想这么麻烦,还有一个更好的方式:我们可以把模型矩阵与投影矩阵相乘,得到一个矩阵,然后把这个矩阵传递给顶点着色器。通过这种方式我们就可以在着色器中仅保留一个矩阵。

选择适当的顺序

为了弄清楚我们应该使用那种顺序,让我们看一下只使用投影矩阵的数学运算:

VerteXeye代表场景中的顶点与投影矩阵相乘之前的位置。我们一旦加入模型矩阵来移动那个桌子,这个数学运算看起来就像这样。

VerteXmodel代表顶点在模型矩阵放进场景中之前的位置。把这两个表达式合并在一起,最后得到的公式如下:

为了使用一个矩阵替换这两个矩阵,我们不得不把投影矩阵乘以模型矩阵,就是把投影矩阵放在左边,把模型矩阵放在右边。

更新代码使用一个矩阵

让我们把这个新的矩阵代码封装一下,把下面的代码加到onSurfaceChanged()中的translate()调用后面:

final float[] temp=new float[16];

Matrix.multiplyMM(temp,0,projectionMatrix,0,modelMatrix,0);

System.arraycopy(temp, 0, projectionMatrix, 0, temp.length);

不论什么时候把两个矩阵相乘,都需要一个临时变量来存储其结果。如果尝试直接写入这个结果,这个结果将是未定义的。

我们首先创建了一个临时的浮点数组用来存储其临时结果;然后调用multiplyMM()把投影矩阵和模型矩阵相乘,其结果存进这个临时数组。下一步,我们调用System.arraycopy()把结果存回projectMatrix,它现在包含模型矩阵与投影矩阵的组合效应。

如果我们现在运行这个程序,会发现这依然是个平面的桌面。

3.增加旋转

既然我们已经有一个配置好的投影矩阵和一个可以移动的桌子的模型矩阵,那么我们需要做的就是旋转这张桌子,以便可以从某个角度观察它。如果使用旋转矩阵,我们只需要一行代码就可以做到。我们还从来没有用过旋转,现在花一些时间了解一下这些旋转是怎么工作的。

需要弄清楚一件时是我们需要围绕哪个轴旋转以及旋转多少度。让我们再看下图:

要搞清楚一个物体是怎么围绕一个给定的轴旋转,我们将使用右手坐标规则:伸出你的右手,握拳,让大拇指指向正轴方向。倘若是一个正角度的旋转,卷曲的手指会告诉你一个物体是怎么围绕那个轴旋转的。观察上图,当你把大拇指指向X轴正方向时,看看旋转的方向是怎么跟随手指的绕轴线卷曲的。

分别用X轴,Y轴和Z轴试一下这个旋转。如果绕Y轴旋转,桌子会绕着它的顶端和低端水平旋转。如果绕着Z轴旋转,桌子会在一个圆圈内旋转。我们想要做的是让桌子绕着X轴向后旋转,因为这会让桌子看起来更有层次。

旋转矩阵

我们将使用一个旋转矩阵去做实际的旋转。旋转矩阵使用正弦和余弦三角函数把旋转转换成缩放因子。下面就是绕X轴旋转所用矩阵定义:

然后是绕Y轴旋转所用的矩阵:

最后,还有一个绕Z轴旋转所用的矩阵:

把所有矩阵合并为一个通用的旋转矩阵,使其可以基于任意一个角度和向量旋转,这也是可能的。

作为一个测试,让我们试试绕着X轴旋转。我们从一个点开始,它在原点上面一个单位,也就是Y值是1。把它绕X轴旋转90度。首先,让我们准备这个旋转矩阵:

让把这个矩阵与这个点的位置向量相乘,看看得到了什么:

这个点从(0,1,0)被移动到了(0,0,1)。如果我们回过头来看一下上面那个旋转坐标轴,并对X轴使用右手规则,我们可以看到正向旋转是如何把一个点沿着一个绕X轴的圈移动的。

4.在代码中加入旋转

我们现在准备好把这个旋转加入代码了。回到onSurfaceChanged(),调整那个平移矩阵,并加入一个旋转矩阵,如下:

Matrix.translateM(modelMatrix,0,0f,0f,-2.5f);

Matrix.rotateM(modelMatrix,0,-60f,1f,0f,0f);

我们把这张桌子放得更远了一点,因为我们一旦把它旋转了,它的底部会距离我们更近。我们接着把它绕X轴旋转-60度,这会让桌子处于一个很好的角度,就像我们站在它前面一样。

这张桌子现在看起来应该如下图所示:

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-12-26 21:11:41

Android OpenGL ES(六)----进入三维在代码中创建投影矩阵和旋转矩阵的相关文章

Android OpenGL ES(十二):三维坐标系及坐标变换初步 .

OpenGL ES图形库最终的结果是在二维平面上显示3D物体(常称作模型Model)这是因为目前的打部分显示器还只能显示二维图形.但我们在构造3D模型时必须要有空间现象能力,所有对模型的描述还是使用三维坐标.也就是使用3D建模,而有OpenGL ES库来完成从3D模型到二维屏幕上的显示. 这个过程可以分成三个部分: 坐标变换,坐标变换通过使用变换矩阵来描述,因此学习3D绘图需要了解一些空间几何,矩阵运算的知识.三维坐标通常使用齐次坐标来定义.变换矩阵操作可以分为视角(Viewing),模型(Mo

Android OpenGL ES 开发教程 从入门到精通

From:http://blog.csdn.net/mapdigit/article/details/7526556 Android OpenGL ES 简明开发教程 Android OpenGL ES 简明开发教程一:概述 Android OpenGL ES 简明开发教程二:构造OpenGL ES View Android OpenGL ES 简明开发教程三:3D绘图基本概念 Android OpenGL ES 简明开发教程四:3D 坐标变换 Android OpenGL ES 简明开发教程五

[转]Android OpenGL ES 开发教程 从入门到精通

本文转自:http://blog.csdn.net/mapdigit/article/details/7526556 Android OpenGL ES 简明开发教程 Android OpenGL ES 简明开发教程一:概述 Android OpenGL ES 简明开发教程二:构造OpenGL ES View Android OpenGL ES 简明开发教程三:3D绘图基本概念 Android OpenGL ES 简明开发教程四:3D 坐标变换 Android OpenGL ES 简明开发教程五

Android OpenGL ES(八)绘制点Point ..

上一篇介绍了OpenGL ES能够绘制的几种基本几何图形:点,线,三角形.将分别介绍这几种基本几何图形的例子.为方便起见,暂时在同一平面上绘制这些几何图形,在后面介绍完OpenGL ES的坐标系统和坐标变换后,再介绍真正的3D图形绘制方法. 在Android OpenGL ES(六):创建实例应用OpenGLDemos程序框架 创建了示例应用的程序框架,并提供了一个“Hello World”示例. 为避免一些重复代码,这里定义一个所有示例代码的基类OpenGLESActivity,其定义如下:

Android OpenGL ES(七)基本几何图形定义 .

在前面Android OpenGL ES(六):创建实例应用OpenGLDemos程序框架 我们创建了示例程序的基本框架,并提供了一个“Hello World”示例,将屏幕显示为红色. 本例介绍OpenGL ES 3D图形库支持的几种基本几何图形,通常二维图形库可以绘制点,线,多边形,圆弧,路径等等.OpenGL ES 支持绘制的基本几何图形分为三类:点,线段,三角形.也就是说OpenGL ES 只能绘制这三种基本几何图形.任何复杂的2D或是3D图形都是通过这三种几何图形构造而成的. 比如下图复

Android OpenGL ES零基础系列(一):理解GLSurfaceView,GLSurfaceView.Render的基本用法

转载请注明出处 前言 OpenGL ES是OpenGL的一个子集,是针对手机.PDA和游戏主机等嵌入式设备而设计的.该API由Khronos集团定义推广,Khronos是一个图形软硬件行业协会,该协会主要关注图形和多媒体方面的开放标准. 因此OpenGL ES作为第三方库被应用在android中. 到目前为止,OpenGL ES已经发展有了3个版本,OpenGL ES 1.0 , OpenGL ES 2.0 , OpenGL ES 3.0.其中OpenGL ES 1.0 是以OpenGL 1.3

Android OpenGL ES 画球体

最近因为兴趣所向,开始学习OpenGL绘图.本文以"画球体"为点,小结一下最近所学. > 初识OpenGL ES 接触OpenGL是从Android开始的.众所周知,Android View 是线程不安全的,于是只允许在主线程中对View进行操作.然而假如我们需要实现复杂的界面,特别是开发游戏,在主线程中画大量图像,会耗费比较长的时间,使得主线程没能及时响应用户输入,甚至出现ANR.于是Android提供了一个 SurfaceView类,通过双缓冲机制(两块画布?三块画布?),允

Android OpenGL ES(十三)通用的矩阵变换指令 .

Android OpenGL ES 对于不同坐标系下坐标变换,大都使用矩阵运算的方法来定义和实现的.这里介绍对应指定的坐标系(比如viewmodel, projection或是viewport) Android OpenGL ES支持的一些矩阵运算及操作. OpenGL ES 中使用四个分量(x,y,z,w)来定义空间一个点,使用4个分量来描述3D坐标称为齐次坐标 :所谓齐次坐标就是将一个原本是n维的向量用一个n+1维向量来表示. 它有什么优点呢? 许多图形应用涉及到几何变换,主要包括平移.旋转

Android OpenGL ES向导学习笔记(扫盲专用)

Android 目前支持下面几个版本的OpenGL ES API : OpenGL ES 1.0 和 1.1 :Android 1.0和更高的版本支持这个API规范. OpenGL ES 2.0 : Android 2.2(API 8)和更高的版本支持这个API规范. OpenGL ES 3.0 : Android 4.3(API 18)和更高的版本支持这个API规范. OpenGL ES 3.1 :  Android 5.0(API 21)和更高的版本支持这个API规范. 支持OpenGL E