Android开发之绘制平面上的多边形功能分析
作者:pku_android 发布时间:2023-12-13 13:31:57
本文实例讲述了Android开发之绘制平面上的多边形功能。分享给大家供大家参考,具体如下:
计算机里的3D图形其实是由很多个平面组合而成的。所谓“绘制3D”图形,其实是通过多个平面图形形成的。调用GL10图形绘制2D图形的步骤如下:
i. 调用GL10的glEnableClientState(GL10.GL_VERTEX_ARRAY);
方法启用顶点坐标数组。
ii. 调用GL10的glEnableClientState(GL10.GL_COLOR_ARRAY);
方法启用顶点颜色数组。
iii. 调用GL10的glVertex (int size,int type,int stride,Buffer pointer);
方法设置顶点的位置数据。这个方法中pointer参数用于指定顶点坐标值,但这里并未使用三维数组来指定每个顶点X、Y、Z坐标的值,pointer依然是一个一维数组,其格式为(x1,y1,z1,x2,y2,z2,x3,y3,z3…xN,yN,zN);也就是说该数组里将会包含3N个数值,每三个值指定一个顶点的X、Y、Z坐标值的类型,如果顶点坐标值为float类型,则指定为GL10.GL_FLOAT;如果顶点坐标值为整数,则指定为GL10.GL_FIXED。
iv. 调用GL10的glColorPointer(int size,int type,int stride,Buffer pointer)
方法设置顶点的颜色数据。这个方法中pointer参数用于指定顶点的颜色值,pointer依然是一个一维数组,,其格式为(r1,g1,b1,a1,x2,y2,z2,a2,x3,y3,z3,a3…xN,yN,zN,aN);也就是该数组里将会包含4N个数值,每4个值指定一个顶点的红绿蓝透明度的颜色值。第一个参数size指定多少个元素指定一个顶点位置,该size参数通常总是4,;type参数指定顶点坐标值的类型,如果顶点坐标值为float类型,则指定为GL10.GL_FLOAT;如果顶点坐标值为整数,则指定为GL10.GL_FIXED。
v. 调用GL10的glDrawArrays(int mode,int first,int count)
方法绘制平面。该方法的第一个参数用于指定绘制图形类型,第二个参数指定从哪个顶点开始绘制,第三个参数指定总共绘制的定点数量。
vi. 绘制完成后,调用GL10的glFinish()
方法结束绘制;并调用glDisableClientState(int)
方法来停用顶点坐标数据,顶点颜色数据。
掌握上面的步骤之后,接下来通过示例程序来绘制几个简单的图形。
public class MyRenderer implements Renderer
{
float[] triangleData = new float []{
0.1f,0.6f,0.0f,//上顶点
-0.3f,0.0f,0.0f,//左顶点
0.3f,0.1f,0.0f//右顶点
};
int[] triangleColor = new int []{
65535,0,0,0,//上顶点红色
0,65535,0,0,//左顶点绿色
0,0,65535,0//右顶点蓝色
};
float[] rectData = new float[]{
0.4f,0.4f,0.0f,//右上顶点
0.4f,-0.4f,0.0f,//右下顶点
-0.4f,0.4f,0.0f,//左上顶点
-0.4f,-0.4f,0.0f//左下顶点
};
int[] rectColor = new int []{
0,65535,0,0,//右上顶点绿色
0,0,65535,0,//右下顶点蓝色
65535,0,0,0,//左上顶点红色
65535,65535,0,0//左下顶点黄色
};
float[] rectData2 = new float[]{
-0.4f,0.4f,0.0f,//左上顶点
0.4f,0.4f,0.0f,
0.4f,-0.4f,0.0f,
-0.4f,-0.4f,0.0f
rectColorBuffer };
float[] pentacle = new float[]{
0.4f,0.4f,0.0f,
-0.2f,0.3f,0.0f,
0.5f,0.0f,0f,
-0.4f,0.0f,0f
-0.1f,-0.3f,0f
};
FloatBuffer triangleDataBuffer;
IntBuffer triangleColorBuffer;
FloatBuffer rectDataBuffer;
FloatBuffer rectDataBuffer2;
FloatBuffer pentacleBuffer;
IntBuffer rectDataBuffer;
public MyRenderer()
{
//将顶点位置数据数组包装成FloatBuffer;
triangleDataBuffer = FloatBuffer.wrap(triangleData);
rectDataBuffer =FloatBuffer.wrap(rectData);
rectDataBuffer2 =FloatBuffer.wrap(rectData2);
pentacleBuffer = FloatBuffer.wrap(pentacle);
//将顶点颜色数据数组包装成IntBuffer;
triangleColorBuffer=IntBuffer.wrap(triangleColor);
rectColorBuffer= IntBuffer.wrap(rectColor);
};
//关闭抗抖动
gl.glDisable(GL10.GL_DITHER);
//设置系对透视进行修正
gl.glHint(GL10.GL_PERSPECTIVE_CORRECTION_HINT, GL10.GL_FASTEST);
gl.glClearColor(0, 0, 0, 0);
//设置阴影平滑模式
gl.glShadeModel(GL10.GL_SMOOTH);
//启用深度测试
gl.glEnable(GL10.GL_DEPTH_TEST);
//设置深度测试的类型
gl.glDepthFunc(GL10.GL_LEQUAL);
}
public void onSurfaceChanged(GL10 gl,int width,int height)
{
//设置3D视窗的大小和位置
gl.glViewport(0,0,width,height);
//将当前矩阵模式设为投影矩阵
gl.glMatrixMode(GL10.GL_PROJECTION);
//初始化单位矩阵
gl.glLoadIdentity();
//计算透视视窗的宽度、高度比
float ratio = (float)width/height;
//调用此方法设置透视视窗的空间大小
gl.glFrustumf(-ratio,ratio,-1,1,1,10);
}
public void onDrawFrame(GL10 gl)
{
//清除屏幕缓存和深度缓存
gl.glClear(GL10.GL_COLOR_BUFFER_BIT|GL10.GL_DEPTH_BUFFER_BIT);
//启用顶点坐标数据
gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
//启用顶点颜色数据
gl.glEnableClientState(GL10.GL_COLOR_ARRAY);
//设置当前矩阵堆栈为模型堆栈
gl.glMatrixMode(GL10.GL_MODELVIEW);
//绘制第一个图形,重置当前的模型视图矩阵
gl.glLoadIdentity();
gl.glTranslatef(-0.32f, 0.35f, -1f);
//设置顶点的位置数据
gl.glVertexPointer(3, GL10.GL_FLOAT, 0, triangleDataBuffer);
//设置顶点的颜色数据
gl.glColorPointer(4, GL10.GL_FIXED, 0, triangleColorBuffer);
//根据顶点数据绘制平面图形
gl.glDrawArrays(GL10.GL_TRIANGLES, 0, 3);
//绘制第二个图形
gl.glLoadIdentity();
gl.glTranslatef(0.6f, 0.8f, -1.5f);
gl.glVertexPointer(3, GL10.GL_FLOAT, 0, rectDataBuffer);
gl.glColorPointer(4, GL10.GL_FIXED, 0, rectColorBuffer);
gl.glDrawArrays(GL10.GL_TRIANGLE_STRIP,0,4);
//绘制第三个图形
gl.glLoadIdentity();
gl.glTranslatef(-0.4f, -0.5f, -1.5f);
gl.glVertexPointer(3,GL10.GL_FLOAT, 0, rectDataBuffer2);
gl.glDrawArrays(GL10.GL_TRIANGLE_STRIP,0,4);
//
gl.glLoadIdentity();
gl.glTranslatef(0.4f, -0.5f, -1.5f);
//设置使用纯色填充
gl.glColor4f(1.0f,0.2f,0.2f,0.0f);
gl.glDisableClientState(GL10.GL_COLOR_ARRAY);
gl.glVertexPointer(3, GL10.GL_FLOAT, 0,penTacleBuffer);
gl.glDrawArrays(GL10.GL_TRIANGLE_STRIP,0,5);
//绘制结束
gl.glFinish();
gl.glDisableClientState(GL10.GL_VERTEX_ARRAY);
}
}
上面的程序使用GL10绘制图形的关键代码:加载顶点位置数据;加载顶点颜色数据;调用GL10的glDrawArrays绘制。
在Activity中定义一个GLSurfaceView,并使用上面的Renderer进行绘制,程序如下:
public void Polygon extends Activity
{
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
//创建一个GLSurfaceView,用于显示OpenGL绘制的图形
GLSurfaceView glView = new GLSurfaceView(this);
//创建GLSurfaceView的内容绘制器
MyRenderer myRender = new MyRenderer();
//为GLSurfaceView设置绘制器
glView.setRenderer(myRender);
setContentView(glView);
}
}
可能大家会觉得奇怪,为什么第二个和第三个图形只是定义4个坐标点的顺序略有不同,为什么图形的差异这么大呢?应为glDrawArrays
方法第一个参数指定绘制的模式,GL10.GL_TRIANGLES是绘制三角形, GL10.GL_TRIANGLE_STRIP是用多个三角形来绘制多边形。
对于第2个图形,当调用glDrawArrays(int mode,int first,int count )
方法时,若指定第一个参数是GL10.GL_TRIANGLE_STRIP时,系统总会从first个顶点开始,每3个顶点绘制一个三角形。
希望本文所述对大家Android程序设计有所帮助。
来源:http://blog.csdn.net/pku_android/article/details/7574608
猜你喜欢
- 最近在做的工作要用到本地方法,需要在Java中加载不少动态链接库(以下为方便延用Windows平台下的简写dll,但并不局限于Windows
- 下面一段简单的代码给大家分享java 获取对象中为null的字段,具体代码如下所述:private static String[] getN
- Handler是什么?Handler 是一个可以实现多线程间切换的类,通过 Handler 可以轻松地将一个任务切换到 Handler 所在
- 指纹识别作为最新兴起的用户身份验证机制,已经被越来越多的应用程序所采用,相比传统的密码九宫格等验证方法,指纹识别更加安全,如今越来越多的安卓
- 详解Java中HashSet和TreeSet的区别1. HashSetHashSet有以下特点:不能保证元素的排列顺序,顺序有可能发生变化不
- 这篇文章主要介绍了Spring案例打印机的实现过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋
- 前端网络访问,主流方案就是 Ajax,Vue 也不例外,在 Vue2.0 之前,网络访问较多的采用 vue-resources,Vue2.0
- 红黑树红黑树是一种数据结构与算法课堂上常常提到但又不会细讲的树,也是技术面试中经常被问到的树,然而无论是书上还是网上的资料,通常都比较刻板难
- 一、ToolBar1、在build.gradle中添加依赖,例如:compile 'com.android.support:appc
- 一、前言在日常工作中,如果涉及到与第三方进行接口对接,有的会使用WebService的方式,这篇文章主要讲解在.NET Framework中
- springboot静态资源加载顺序优先级看springboot源码里面springboot静态资源加载规则我们经常会使用springboo
- 一、序言在日常一线开发过程中,总有列表转树的需求,几乎是项目的标配,比方说做多级菜单、多级目录、多级分类等,有没有一种通用且跨项目的解决方式
- 我们知道多线程因为同时处理子线程的能力,对于程序运行来说,能够达到很高的效率。不过很多人对于多线程的执行方法还没有尝试过,本篇我们将为大家介
- 汉诺塔的规则是:一共三根柱子,一根柱子从上到
- 目录一、简介二、入门案例三、自定义认证逻辑四、自定义授权逻辑五、注销登录六、记住我功能七、会话管理一、简介Spring Security是一
- 前言《接月饼小游戏》是一个基于java的自制游戏,不要被月亮砸到,尽可能地多接月饼。此小项目可用来巩固JAVA基础语法,swing的技巧用法
- 本文实例讲述了c#分页读取GB文本文件的方法。分享给大家供大家参考。具体如下:一、应用场景:① .我在做BI开发测试的时候,有可能面对sou
- GlobalLock的作用对于某条数据进行更新操作,如果全局事务正在进行,当某个本地事务需要更新该数据时,需要使用@GlobalLock确保
- C#查看堆栈通常是在异常处理中,出现异常之后通过异常的堆栈可以很方便的得到出现这个错误的代码调用路径。这个很有用,是否可以在没有异常出现时使
- 当前的Winform分页控件中,当前导出的数据一般使用Excel来处理,Excel的文档可以用于后期的数据展示或者批量导入做准备,因此是比较