Android OpenGL ES(九)----构建几何物体

1.三角形扇

一个三角形扇以一个中心顶点作为起始,使用相邻的两个顶点创建第一个三角形,接下来的每个顶点都会创建一个三角形,围绕起始的中心点按扇形展开。为了使这个扇形闭合,我们只需要在最后重复第二个点。(以长方形为例)

构建三角形扇的步骤,如下图所示:

要使用OpenGL绘制这个三角形扇,需要在渲染类的onDrawFrame()中,使用如下方法:

GLES20.glDrawArray(GLES20.GL_TRIANGLE_FAN,0,6);

第一个参数是告诉OpenGL要绘制一个三角形扇,第二参数是告诉OpenGL从本地顶点数据的第几个位置开始取顶点坐标,第三个参数是告诉OpenGL要取多少个顶点坐标。

根据上面的6个点,就可以绘制一个长方形了。

三角形扇在OpenGL的应用:长方形,正方形,圆等。

2.三角形带

一个三角形带的前三个顶点定义了第一个三角形。这之后的每个额外的顶点都定义了另外的一个三角形。

构建三角形带的步骤,如下图所示:

要使用OpenGL绘制这个三角形带,需要在渲染类的onDrawFrame()中,使用如下方法:

GLES20.glDrawArray(GLES20.GL_TRIANGLE_STRIP,0,6);

第一个参数是告诉OpenGL要绘制一个三角形带,第二参数是告诉OpenGL从本地顶点数据的第几个位置开始取顶点坐标,第三个参数是告诉OpenGL要取多少个顶点坐标。

三角形带在OpenGL的应用:长方形,圆柱的侧面等。

3.圆柱体

想象一下在自己的手机上构建的圆柱体,并且以一个角度观察它,假如我们把圆柱放在桌面上,从侧面观察它,你会发现,圆柱体是一个由一个顶部的圆加上侧面卷起来的长方形构成。结合本文前2节的讲解,就可以知道其实就是一个三角形带和三角形扇构建一个圆柱体。

要构建三角形扇,我们首先定义一个圆心顶点,接着,我们围绕圆心的点按扇形展开,并把第一个点绕圆周重复两次使其圆闭合。我们接下来使用三角函数和单位圆的概念生成那个点。

为了生成沿一个圆周边的点,我们首先需要一个循环,它的覆盖范围从0到360度的整个圆,或者0到2π弧度。要找到圆周上的一个点的X的位置,我们需要调用cos(angle),如果你是放在Z-X平面那么,Z的位置我们就需要调用sin(angle);我们用圆的半径缩放这两个位置。这是圆柱上的圆的绘画过程。

如果是圆柱的侧面,我们就需要看图了解一下,我们假设圆柱垂直方向以Y为中心,圆柱高height,得到如下图:

我们该怎么用程序绘画出来这个圆柱体呢?其实在OpenGL如果想绘制的图像越清晰,那么它绘制的点就会越多越密集,所以由我们自己决定绘制这个圆柱体需要多少个顶点。

我们要计算圆柱体顶部顶点数量的方法作为开始,我们定义一个求圆柱体上面圆的顶点数的方法,如下:

private static int sizeOfCricleInVertices(int numPoint){

return 1+(numPoint+1);

}

一个圆柱体的顶部是一个用三角形扇构造的圆;它有一个顶点在圆心,围着圆的每个顶点点都有一个顶点,并且围着圆的顶点要重复两次,才能使圆闭合。

下面是计算圆柱体侧面顶点的数量:

private static int sizeOfOpenCylinderInVertices(int numPoint){

return (numPoints+1)*2;

}

一个圆柱体侧面是一个卷起来的长方形,由一个三角形带构造,围着顶部圆的每个点都需要两个顶点,并且前两个顶点要重复两次才能使这个管闭合。看三角形带,我们直指定了上面的点的数量,自然下面的点也要计算进去,所以都需要两个顶点。

添加几何图形的类

我们要构建几何物体,其实可以分解成几个类,这样便于管理和重用。创建一个新的类,为Geometry,在该类的内部我们定义一个坐标类,也就是点类:

public static class Point(){

public final float x,y,z;

public Point(float x,float y,float z){

this.x=x;

this.y=y;

this.z=z;

}

public Point translateY(float distance){

return new Point(x,y+distance,z);

}

}

其中有一个辅助函数用于把这个点沿着Y轴平移。我们也需要给,下面我们也给圆一个定义,如下,也为Geometry的内部类:

public static class Circle{

private final Point center;

private final float radius;

public Circle(Point center,float radius){

this.center=center;

this.radius=radius;

}

public Circle scale(float scale){

return new Circle(center,radius*scale);

}

}

我们同样在圆的类里面定义了一个辅助函数,用于缩放圆的半径,最后是给圆柱一个定义,如下:

public static class Cylinder{

public final Point center;

public final float radius;

public finla float height;

public Cylinder(Point center,float radius,float height){

this.center=center;

this.radius=radius;

this.height=height;

}

}

一个圆柱体就像一个扩展的圆,它有一个中心,一个半径和一个高度。

你可能注意到了我们已经把这几个几何物体定义的类定义为不可变的;无论什么时候改动它,都会返回一个新的对象。这有助于使代码更容易使用和理解。但是当你需要提高性能时,你也许想一直用简单的浮点数组,并用静态函数改变它们。

添加物体构建器

我们现在可以开始写物体构建器了,在你的objects包中创建一个名为“ObjectBuilder”的类,在类的内部,以下面的代码作为开始:

private static final int FLOATS_PER_VERTEX=3;

private final float[] vertexData;

private int offset=0;

private ObjectBuilder(int sizeInVertices){

this.vertexData=new float[sizeInVertices*FLOATS_PER_VERTEX];

}

我们定义了一个常量用来保存椒一个顶点需要多少浮点数,一个数组用于保存这些顶点,以及一个变量用于记录数组中下一个顶点的位置。这个构造函数基于需要的顶点数量初始化数组。

用三角形扇构建圆

在ObjectBuilder类中创建一个名为appendCircle的新方法,并加入如下代码:

private void appendCircle(Circle circle,int numPoint){

final int startVertex=offset/FLOATS_PER_VERTEX;

final int numVertices=sizeOfCircleInVertices(numPoint);

this.vertexData[offset++]=circle.center.x;

this.vertexData[offset++]=circle.center.y;

this.vertexData{offset++}=circle.center.z;

for(i=0;i<numPoint;i++){

float angleInRadians=(float)i/(float numPoints)*((float)Math.PI*2f);

this.vertexData[offset++]=circle.center.x+circle.radius*FloatMath.cos(angleInRadians);

this.vertexData[offset++]=circle.center.y;

this.vertexData[offset++]=circle.center.z+circle.radius*FloatMath.sin(angleInRadians);

}

}

我们知道我们想要使用本地存储的顶点,必须设置偏移量,也就是多少个顶点才是正确的坐标,比如我们现在在绘画一个圆柱体,把圆的顶点和侧面的顶点都存储在本地,如果我们开始绘画圆,那么自然偏移是0,但是圆的顶点都绘完后,我们总不能还是从开始的顶点开始取值把,所以跳过取顶点,跳过的就是偏移值startVertex。而numVertices就是要取顶点的长度。

接着我们在ObjectBuilder类的开始处定义一个绘画的接口,顺便也加入一个变量,如下:

static interface DrawCommand{

void draw();

}

private final List<DrawCommand> drawList=new ArrayList<DrawCommand>();

这个常量用于保存绘画命令,如下:

this.drawList.add(new Command(){

public void draw(){

GLES20.glDrawArrays(GLES20.GL_TRIANGLE_FAN,startVertex,numVertices);

}

});

用三角形带构造圆柱体的侧面,为了代码的重用,我们额外定义了一个绘制侧面的方法appendOpenCylinder(),它也需要偏移量和长度,而且,看第二节的图片,你发现没有,上面一排和下面一排的顶点Y值是一样的,所以首先我们在appendOpenCylinder()加入下面四个常量:

private void appendOpenCylinder(Cylinder cylinder,int numPoints){

final int startVertex=offset/FLOATS_PER_VERTEX;

final int numVertices=sizeOfOpenCylinderInVertices(numPoints);

final float yStart=cylinder.center.y-(cylinder.height/2f);

final float yEnd=cylinder.center.y+(cylinder.height/2f);

}

然后加入如下代码生成实际的三角形带:

for(int i=0;i<=numPoints;i++){

float angleInRadians=(float)i/(float numPoints)*((float)Math.PI*2F);

float xPosition=cylinder.center.x+cylinder.radius*FloatMath.cos(angleInRadians);

float zPosition=cylinder.center.z+cylinder.radius*FloatMath.sin(angleInRadians);

this.vertexData[offset++]=xPosition;

this.vertexData[offset++]=yStart;

this.vertexData[offset++]=zPosition;

this.vertexData[offset++]=xPosition;

this.vertexData[offset++]=yEnd;

this.vertexData[offset++]=zPosition;

}

最后在该方法中加入如下代码完成:

this.drawList.add(new Command(){

public void draw(){

GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP,startVertex,numVertices);

}

});

我们在ObjectBuilder类中实现了构造圆和圆柱侧面的方法。是不是还少了什么,没错,你获得的顶点和数据怎么传递给其他的类呢?所以我们在ObjectBuilder里面定义了一个包装类,将存储的绘制命令和顶点数据都传递给它,如下:

static class GenerateData{

final float[] vertexData;

final List<Command> drawList;

GenerateData(float[] vertexData,List<DrawCommand> drawList){

this.drawList=drawList;

this.vertexData=vertexData;

}

}

最后就是将圆和侧面叠加成一个圆柱体,我们ObjectBuilder加入如下的代码:

static GeneratedData createCylindrical(Cylinder cylinder,int numPoints){

int size=sizeOfCricleInVertices()+sizeOfOpenCylinderInVertices(numPoints);//计算总的顶点数

ObjectBuilder builder=new ObjectBuilder(size);//根据顶点数实例化vertexData;

Circle circle=new Circle(cylinder.center.translateY(cylinder.height/2f),cylinder.radius);

builder.appendCircle(circle,numPoints);

builder.appendOpenCylinder(cylinder,numPoints);

return builder.build();

}

private GenerateData build(){

return new GenerateData(this.vertexData,this.drawList);

}

这样一个圆柱体绘制类就完成了。

源代码如下:http://download.csdn.net/detail/liyuanjinglyj/8859411

附上效果图:

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

时间: 2024-11-03 22:18:16

Android OpenGL ES(九)----构建几何物体的相关文章

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

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

Android OpenGL ES零基础系列(三):OpenGL ES的渲染管道及VertexShader与FragmentShader

前言 在前2篇文章中,我们都说到着色器,且在第二篇中正式说到,这着色器只能用在OpenGL ES2.x等可编程管道里面,而在OpenGL ES1.x是不能用的.但我们一直没有说这是为什么,两者有什么区别.那这篇我们就一起来学习下OpenGL ES中的渲染管道. 正文 管道,英文名叫Pipeline,相信用过FaceBook图片加载库的同学对这个管道并不陌生,因为SimpleImageDrawee里面也是用的管道来对图片进行的一个处理.由于其底层也是C,因此我可以大胆的猜想,FaceBook图片加

Android OpenGL ES

1.Android OpenGL ES 简明开发教程3D 坐标变换: http://www.linuxidc.com/Linux/2011-10/45756p4.htm

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: non-square texture - continue

previous: [工作记录] Android OpenGL ES 2.0: square texture not supported on some device recently I found that it is the mipmap of a non-square texture that cause the black texture problem: http://stackoverflow.com/questions/5052762/using-mipmaps-results-

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 2.0: square texture not supported on some device

npot texture: non-power-of-two texture.rectangle texture: non-square (height != wdith) 在测试Samsumg Galaxy S4的时候, 发现rectangle texture支持不好, 虽然创建成功, 但是绘制有问题. 不同的模块出现类似的情况: 纹理采样出的颜色是(0,0,0,1). 在排除了所有可能的runtime问题以后, 尝试性改了纹理格式无果, 又尝试性的改了下尺寸, 竟然好了. 我擦... 而Ga

Android OpenGL ES 画球体

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

[转]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 简明开发教程五