热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

AndroidOpenGLES2.0绘制三角形(二)

这篇文章主要为大家详细介绍了AndroidOpenGLES2.0绘制三角形的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

选择绘制三角形作为OpenGL ES 2.0的第一个实例,是因为前文中提到的,点、线、三角形是OpenGL ES世界的图形基础。无论多么复杂的几何物体,在OpenGL ES的世界里都可以用三角形拼成。关于Android OpenGL ES 三角形的绘制,在Android官方文档中有详细的说明和步骤,本文实例也是依照官方文档步骤绘制的三角形。

步骤

依照官方文档中的说明,Android中利用OpenGL ES 2.0绘制三角形的步骤为:

1. 在AndroidManifest.xml文件中设置使用的OpenGL ES的版本:


3.0的版本为0x00030000,3.1的版本为0x00030001。
需要注意的是前一篇博客中提到的Android各个版本对于OpenGL ES版本的支持,设置Android应用的minSDK不应该小于使用的支持OpenGL ES版本的最低Android SDK版本。

2. 毫无疑问的,显示三角形,需要一个载体。创建显示三角形的Activity,利用GLSurfaceView作为显示三角形的View,图形的具体渲染工作都是在Render中完成的。

3. 实现GLSurfaceView的Render,在Render中完成三角形的绘制,具体行为有:

  • 加载顶点和片元着色器
  • 确定需要绘制图形的坐标和颜色数据
  • 创建program对象,连接顶点和片元着色器,链接program对象。
  • 设置视图窗口(viewport)。
  • 将坐标数据颜色数据传入OpenGL ES程序中
  • 使颜色缓冲区的内容显示到屏幕上。

具体实现

我们设置好OpenGL ES版本、创建入口Activity并设置好GLSurfaceView做为显示载体后,就进入了我们最主要的工作了。

第一步

首先,我们需要编写一个简单的顶点着色器和一个简单的片元着色器:
顶点着色器:

 attribute vec4 vPosition;
 void main() {
  gl_Position = vPosition;
 }

片元着色器:

 precision mediump float;
 uniform vec4 vColor;
 void main() {
  gl_FragColor = vColor;
 }

gl_Position和gl_FragColor都是Shader的内置变量,分别为定点位置和片元颜色。

第二步

然后,我们确定我们要绘制的图形的顶点坐标和颜色:
我们现在需要绘制的是在一个三维空间中绘制一个三角形,三角形当然是三个顶点了。因为我们三角形只是一个平面图形,为了方便,我们现在不设置相机(相机在后面的博客中使用时在讲解)的情况下,三角形正对我们来呈现。所以我们把三个顶点的Z坐标都设定为0。
上篇博客中也有提到OpenGL ES坐标映射到屏幕上,从屏幕中心垂直到上下左右边缘距离都为1.0,所以(-1.0,0,0)和(0,1.0,0)到原点的距离在屏幕上呈现出来的结果是不一样的,图解如下(左边是理想状态,右边是实际状态):

所以,为了不超出屏幕,我们的坐标数据设置为:

float triangleCoords[] = {
   0.5f, 0.5f, 0.0f, // top
   -0.5f, -0.5f, 0.0f, // bottom left
   0.5f, -0.5f, 0.0f // bottom right
 };

颜色数据,我们设置为单一颜色:

float color[] = { 1.0f, 1.0f, 1.0f, 1.0f }; //白色

第三步

接着我们开始在Render中实现我们的三角形绘制了。Render接口有三个方法,分别为onSurfaceCreated、onSurfaceChanged和onDrawFrame。
在onSurfaceCreated方法中,我们来创建program对象,连接顶点和片元着色器,链接program对象。

 //将背景设置为灰色
 GLES20.glClearColor(0.5f,0.5f,0.5f,1.0f); 
 //申请底层空间
 ByteBuffer bb = ByteBuffer.allocateDirect(
    triangleCoords.length * 4);
 bb.order(ByteOrder.nativeOrder());
 //将坐标数据转换为FloatBuffer,用以传入给OpenGL ES程序
 vertexBuffer = bb.asFloatBuffer();
 vertexBuffer.put(triangleCoords);
 vertexBuffer.position(0); 
 int vertexShader = loadShader(GLES20.GL_VERTEX_SHADER,
    vertexShaderCode);
 int fragmentShader = loadShader(GLES20.GL_FRAGMENT_SHADER,
    fragmentShaderCode);

 //创建一个空的OpenGLES程序
 mProgram = GLES20.glCreateProgram();
 //将顶点着色器加入到程序
 GLES20.glAttachShader(mProgram, vertexShader);
 //将片元着色器加入到程序中
 GLES20.glAttachShader(mProgram, fragmentShader);
 //连接到着色器程序
 GLES20.glLinkProgram(mProgram);

第四步

在onSurfaceChanged中设置设置视图窗口:

GLES20.glViewport(0,0,width,height);

第五步

最后在onDrawFrame中绘制:

 //将程序加入到OpenGLES2.0环境
 GLES20.glUseProgram(mProgram);

 //获取顶点着色器的vPosition成员句柄
 mPositiOnHandle= GLES20.glGetAttribLocation(mProgram, "vPosition");
 //启用三角形顶点的句柄
 GLES20.glEnableVertexAttribArray(mPositionHandle);
 //准备三角形的坐标数据
 GLES20.glVertexAttribPointer(mPositionHandle, COORDS_PER_VERTEX,
   GLES20.GL_FLOAT, false,
   vertexStride, vertexBuffer);
 //获取片元着色器的vColor成员的句柄
 mColorHandle = GLES20.glGetUniformLocation(mProgram, "vColor");
 //设置绘制三角形的颜色
 GLES20.glUniform4fv(mColorHandle, 1, color, 0);
 //绘制三角形
 GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, vertexCount);
 //禁止顶点数组的句柄
 GLES20.glDisableVertexAttribArray(mPositionHandle);

最终效果

源码地址

所有的代码全部在一个项目中,托管在Github上——Android OpenGLES 2.0系列博客的Demo

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 解决微信电脑版无法刷朋友圈问题:使用安卓远程投屏方案
    在工作期间想要浏览微信和朋友圈却不太方便?虽然微信电脑版目前不支持直接刷朋友圈,但通过远程投屏技术,可以轻松实现在电脑上操作安卓设备的功能。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 从零开始构建完整手机站:Vue CLI 3 实战指南(第一部分)
    本系列教程将引导您使用 Vue CLI 3 构建一个功能齐全的移动应用。我们将深入探讨项目中涉及的每一个知识点,并确保这些内容与实际工作中的需求紧密结合。 ... [详细]
  • 本报告涵盖了个人博客账号和码云账号的注册过程,以及对网络工程专业学习的反思与展望。通过回顾初入大学时的专业选择,分析当前的专业知识和技能水平,并对未来的职业规划进行了详细讨论。 ... [详细]
  • 本文详细介绍了 com.facebook.drawee.view.SimpleDraweeView 中的 setScaleType 方法,提供了多个实际代码示例,并解释了其在不同场景下的应用。 ... [详细]
author-avatar
真实的小莹_808
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有