Hello Triangle:OpenGL ES 2.0 版的“Hello world”

Hello Triangle:OpenGL ES 2.0 版的“Hello world”

本文的文字大部分都是从《OpenGL ES 2.0 编程向导》中摘抄而来,特此说明。

该文是基于OpengGL ES 2.0的,算是本人学习OpenGL的“Hello world”吧。

一个OpengGL ES 2.0程序的实现大致如下所示:

  • 装载顶点和片段着色器。
  • 创建一个项目对象,联系顶点和片段着色器,链接项目。
  • 设置视窗。
  • 清除颜色缓冲区。
  • 最基本的渲染。

准备工作

作为一个“Hello world”类的程序,功能就是在手机上绘制一个三角形。因而代码就会比较简陋,预先准备的食材也就比较简单,就是下面这五个文件(连布局文件都省了):

AndroidManifest.xml

<!-- Tell the system this app requires OpenGL ES 2.0. -->
<uses-feature android:glEsVersion="0x00020000" android:required="true" />

MainActivity.java

public class MainActivity extends Activity {

    private TriangleGLSurfaceView mTriangleGLSurfaceView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mTriangleGLSurfaceView = new TriangleGLSurfaceView(getApplication());
        setContentView(mTriangleGLSurfaceView);
    }
}

TriangleGLSurfaceView.java

public class TriangleGLSurfaceView extends GLSurfaceView{

    public TriangleGLSurfaceView(Context context) {
        super(context);
        setEGLContextClientVersion(2);
        setRenderer(new TriangleRender());
    }

}

TriangleRender.java

public class TriangleRender implements Renderer{

    private Triangle mTriangle;

    @Override
    public void onSurfaceCreated(GL10 gl, EGLConfig config) {
        GLES20.glClearColor(0, 0, 0, 1);
        mTriangle = new Triangle();
    }

    @Override
    public void onSurfaceChanged(GL10 gl, int width, int height) {
        GLES20.glViewport(0, 0, width, height);
    }

    @Override
    public void onDrawFrame(GL10 gl) {
        // Clear the color buffer
        GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT);
        mTriangle.draw();
    }

}

Triangle.java

public class Triangle {
    ...省略大段文字,会在下面补充啦...
}

具体实现

下面这些操作大都在Triangle.java这个类里面实现的。

创建一个简单的顶点和片段着色器

OpenGL ES 2.0 中,在有效的顶点和片段着色器被装载前,什么渲染都做不了。定义定点着色器:

private static final String VERTEX_SHADER =
        "attribute vec4 vPosition;          \n" +
        "void main()                        \n" +
        "{                                  \n" +
        "   gl_Position = vPosition;        \n" +
        "}";

顶点着色器定义一个输入attribute,它是4 个成员的矢量vPosition。后面的draw函数将为每个顶点设置位置变量值。main函数声明着色器宣布着色器开始执行。着色器主体非常简单,它复制输入vPosition 属性到gl_Position 输出变量中,每个顶点着色器必须输出位置值到gl_Position变量中,这个变量传入到管线的下一个阶段中。

顶点着色器如下:

private static final String FRAGMENT_SHADER =
        "precision mediump float;           \n" +
        "void main()                        \n" +
        "{                                  \n" +
        "    gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);       \n" +
        "}";

第一行宣布着色器默认的浮点变量精度。main 函数上,它的输出值(1.0, 0.0, 0.0, 1.0)赋给变量gl_FragColor,gl_FragColor是片段着色器最终的输出值,本例中输出值是红色。

典型的一个游戏或应用将不会内联一个着色器源码串,大多数的实际应用着色器应该填充文字或数据,然后被API 装载,我们为简化起见,在程序源码上直接赋值。

编译和装载着色器

定义了着色器源码后,我们将着色器装入OpenGL ES,这由loadShader函数完成,检查没有错误后,这个函数返回着色器对象,这个对象在后面被用于连接项目对象

private int loadShader(int type, String shaderCode){
    int shader;
    int[] compiled = new int[1];
    // Create the shader object
    shader = GLES20.glCreateShader(type);
    if (shader == 0) {
        return 0;
    }
    // Load the shader source
    GLES20.glShaderSource(shader, shaderCode);
    // Compile the shader
    GLES20.glCompileShader(shader);
    // Check the compile status
    GLES20.glGetShaderiv(shader, GLES20.GL_COMPILE_STATUS, compiled, 0);
    if (compiled[0] == 0) {
        Log.e(TAG, "Could not compile shader " + type + ":");
        Log.e(TAG, GLES20.glGetShaderInfoLog(shader));
        GLES20.glDeleteShader(shader);
        shader = 0;
    }
        return shader;
}

如果着色器对象装载成功,一个新的着色器对象被返回,在后面将连接到项目对象上。

创建项目对象链接着色器

一旦应用程序已经创建了顶点、片段着色器对象,它需要去创建项目对象,项目是最终的链接对象,每个着色器在被绘制前都应该联系到项目或者项目对象。

  • 1.创建项目对象
// Create the program object.
mProgram = GLES20.glCreateProgram();
if (mProgram == 0) {
    return;
}
GLES20.glAttachShader(mProgram, vertexShader);
GLES20.glAttachShader(mProgram, fragmentShader);
  • 2.设定顶点着色器vPosition 属性:
// Bind vPosition to attribute 0
GLES20.glBindAttribLocation(mProgram, 0, "vPosition");

顶点属性、顶点矩阵、缓冲区对象, 讲述更多的细节现在我们看glBindAttribLocation函数绑定vPosition 属性到顶点着色器位置0,当我们指定顶点数据后,位置指针指向下一个位置。

  • 3.链接项目检查错误:
// Link the program
GLES20.glLinkProgram(mProgram);
// Check the link status
int[] linkStatus = new int[1];
GLES20.glGetProgramiv(mProgram, GLES20.GL_LINK_STATUS, linkStatus, 0);
if (linkStatus[0] != GLES20.GL_TRUE) {
    Log.e(TAG, "Could not link program: ");
    Log.e(TAG, GLES20.glGetProgramInfoLog(mProgram));
    GLES20.glDeleteProgram(mProgram);
    mProgram = 0;
}

所有这些步骤后,编译着色器,检查错误,创建项目对象,附加上着色器,链接项目,检查链接错误。成功后可以使用项目对象去渲染。

设定窗口和清除缓冲区

创建了渲染平面初始化并装载了着色器,准备去绘制实际的物体,我们就要用到:TriangleRender.java这个类了。而GLSurfaceView.Renderer这个接口,在Android官方给出的描述是这样的:

public interface Renderer {
    /**
     * Called when the surface is created or recreated.
     * <p>
     * Called when the rendering thread
     * starts and whenever the EGL context is lost. The EGL context will typically
     * be lost when the Android device awakes after going to sleep.
     * <p>
     * Since this method is called at the beginning of rendering, as well as
     * every time the EGL context is lost, this method is a convenient place to put
     * code to create resources that need to be created when the rendering
     * starts, and that need to be recreated when the EGL context is lost.
     * Textures are an example of a resource that you might want to create
     * here.
     * <p>
     * Note that when the EGL context is lost, all OpenGL resources associated
     * with that context will be automatically deleted. You do not need to call
     * the corresponding "glDelete" methods such as glDeleteTextures to
     * manually delete these lost resources.
     * <p>
     * @param gl the GL interface. Use <code>instanceof</code> to
     * test if the interface supports GL11 or higher interfaces.
     * @param config the EGLConfig of the created surface. Can be used
     * to create matching pbuffers.
     */
    void onSurfaceCreated(GL10 gl, EGLConfig config);
    /**
     * Called when the surface changed size.
     * <p>
     * Called after the surface is created and whenever
     * the OpenGL ES surface size changes.
     * <p>
     * Typically you will set your viewport here. If your camera
     * is fixed then you could also set your projection matrix here:
     * <pre class="prettyprint">
     * void onSurfaceChanged(GL10 gl, int width, int height) {
     *     gl.glViewport(0, 0, width, height);
     *     // for a fixed camera, set the projection too
     *     float ratio = (float) width / height;
     *     gl.glMatrixMode(GL10.GL_PROJECTION);
     *     gl.glLoadIdentity();
     *     gl.glFrustumf(-ratio, ratio, -1, 1, 1, 10);
     * }
     * </pre>
     * @param gl the GL interface. Use <code>instanceof</code> to
     * test if the interface supports GL11 or higher interfaces.
     * @param width
     * @param height
     */
    void onSurfaceChanged(GL10 gl, int width, int height);
    /**
     * Called to draw the current frame.
     * <p>
     * This method is responsible for drawing the current frame.
     * <p>
     * The implementation of this method typically looks like this:
     * <pre class="prettyprint">
     * void onDrawFrame(GL10 gl) {
     *     gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT);
     *     //... other gl calls to render the scene ...
     * }
     * </pre>
     * @param gl the GL interface. Use <code>instanceof</code> to
     * test if the interface supports GL11 or higher interfaces.
     */
    void onDrawFrame(GL10 gl);
}

我们解释的第1个函数是glViewport,它定义OpenGL ES 的将要绘制物体的2D 窗口的坐标原点和窗口宽度和高度。在OpenGL ES 中viewport 定义渲染操作将要显示的2D 长方形显示区域。

@Override
public void onSurfaceChanged(GL10 gl, int width, int height) {
    GLES20.glViewport(0, 0, width, height);
}

viewport 设定窗口的原点origin (x, y)、宽度和高度

设定窗口后,下一步是清除屏幕,在OpenGL ES 中,有多种需要绘制的缓冲区类型,颜色、深度和模板。在本例中仅仅颜色缓冲区被使用。开始每帧绘制前,我们使用glClear 清除颜色缓冲区。

@Override
public void onDrawFrame(GL10 gl) {
    // Clear the color buffer
    GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT);
    ...Something else...
}

缓冲区将被用glClearColor 函数的颜色参数值清除,本例初始化结束后,清除颜色被设定为(0.0, 0.0, 0.0, 1.0),于是屏幕变成黑色。清除颜色通过glClear 设定。

public void onSurfaceCreated(GL10 gl, EGLConfig config) {
    GLES20.glClearColor(0, 0, 0, 1);
    ...Something else...
}

装载几何图像绘制基元

现在我们已清除了颜色缓冲区,设定了视口,装载了项目对象,我们指定要绘制的几何图形是三角形,三角形顶点的坐标如下:

static float triangleCoords[] = {
    0.0f, 0.5f, 0.0f,
    -0.5f, -0.5f, 0.0f,
    0.5f, -0.5f, 0.0f
};

...

public Triangle() {
    ByteBuffer bb = ByteBuffer.allocateDirect(triangleCoords.length*4);
    bb.order(ByteOrder.nativeOrder());
    mVertexBuffer = bb.asFloatBuffer();
    mVertexBuffer.put(triangleCoords);
    mVertexBuffer.position(0);
}

...

public void draw(){
    // Use the program object
    GLES20.glUseProgram(mProgram);
    // Load the vertex data
    GLES20.glVertexAttribPointer(0, 3, GLES20.GL_FLOAT, false, 0, mVertexBuffer);
    GLES20.glEnableVertexAttribArray(0);
    GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, 3);
}

顶点位置需要被装载到GL 联系到vPosition,你是否想到先前我们绑定vPosition 变量到属性位置0,每个顶点着色器中的属性都有一个唯一的用无符号整形数标示的位置,调用glVertexAttribPointer 函数,我们把数据装载到位置0

绘制三角形最后一步是调用OpenGL ES 去绘制基元,本例中使用glDrawArrays 函数。这个函数绘制三角形、直线或带状物等基元,将在第7 章详细介绍这些几何图形的细节。

后缓冲区显示

最后在谈谈绘制在缓冲区中的三角形。最后说明缓冲区如何显示到屏幕上,讨论之前,先介绍一点双缓冲区概念。

在显示屏上看到的帧缓冲区是2 维的空间的像素数据,可能想到的方法是简单更新可见缓冲区中的数据,直接更新显示缓冲区中数据有困难,对显示屏幕缓冲区内存更新要有固定的频率。如果两次时间不同,将看到闪烁的痕迹。

为解决这个问题,系统一般使用双缓冲区:前缓冲区和后缓冲区,所有的渲染发生在后缓冲区,它是屏幕上不可见的缓冲区内存,渲染完成后,交换前后缓冲区,即在下一帧时前

缓冲区变成后缓冲区。

使用这个技术,在一帧渲染完成前,我们不显示任何图像,这种OpenGL ES 方法使用

通过EGL 实现,使用函数是eglSwapBuffers;这个EGL 函数交换前后缓冲区,eglSwapBuffers 的输入参数是EGL 显示区和窗口,这两个参数代表了实际的物理显示区和渲染区。现在我们知道我们交换缓冲区后,显示了要显示的三角形。

结果

结果就像下面图里显示的那样,不过由于没有处理横竖屏时屏幕的变化,所以样子会根据屏幕方向的切换而变化。

横屏

竖屏

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

时间: 2025-01-02 15:14:24

Hello Triangle:OpenGL ES 2.0 版的“Hello world”的相关文章

《OpenGL ES 2.0 Programming Guide》第12章“最简单的ReadPixels并保存为BMP”示例代码【C语言版】

由于<OpenGL ES 2.0 Programming Guide>原书并没有提供第12章的示例代码,书上的代码也只提到关键的步骤,而网上大多是Android/iOS版本的示例,C/C++的大都基于OpenGL或OpenGL ES 3.0,为了加深理解,遂自己实现了一份C语言版本的,希望能够帮助到同样喜欢OpenGL ES 2.0的同学. 废话不多说,直接上代码 #include "stdafx.h" #include "esUtil.h" #incl

OpenGL ES 3.0之顶点缓冲

所谓顶点缓冲就是直接将顶点数据存储在gpu的一段缓冲区,不需要从cpu拷贝到gpu.提高了程序的运行效率. 操作步骤 1.创建顶点缓冲对象 GLuint vertexBufferID; 2.分配空间 glGenBuffers(1, &vertexBufferID); 3.绑定当前顶点缓冲对象 glBindBuffer(GL_ARRAY_BUFFER, vertexBufferID); 4.初始化缓冲区数据 glBufferData(GL_ARRAY_BUFFER, sizeof(vertices

基于Cocos2d-x学习OpenGL ES 2.0系列——你的第一个三角形(1)

[本系列转自]http://cn.cocos2d-x.org/tutorial/lists?id=79 前言 在本系列教程中,我会以当下最流行的2D引擎Cocos2d-x为基础,介绍OpenGL ES 2.0的一些基本用法.本系列教程的宗旨是OpenGL扫盲,让大家在使用Cocos2d-x过程中,知其然,更知其所以然.本系列教程不会涉及非常底层的数学原理,同时也不会过多地提及OpenGL本身的一些细节知识.但是我会在每篇文章的最后给出一些参考链接,大家可以顺藤摸瓜,一举Get OpenGL这个新

基于Cocos2d-x学习OpenGL ES 2.0系列——使用VBO索引(4)

在上一篇文章中,我们介绍了uniform和模型-视图-投影变换,相信大家对于OpenGL ES 2.0应该有一点感觉了.在这篇文章中,我们不再画三角形了,改为画四边形.下篇教程,我们就可以画立方体了,到时候就是真3D了. 为什么三角形在OpenGL教程里面这么受欢迎呢?因为在OpenGL的世界里面,所有的几何体都可以用三角形组合出来.我们的四边形也一样,它可以用两个三角形组合出来. 你的第一个四边形 首先,因为OpenGL里面没有直接绘制四边形的命令的,所以我们需要画两个三角形来拼成一个四边形.

win7下搭建opengl es 2.0开发环境

http://codingnow.cn/opengles/1501.html =================================================================== 1. 下载AMD的OpenGL ES2.0的模拟器 ,下载地址: http://www.opengles-book.com/ESEmulator.2009-04-28-v1.4.APRIL_2009_RELEASE.msi 2. 下载<OpenGL ES2.0 Programming

OpenGL ES 3.0 支持的设备和模拟器

最近自己写的引擎想要做OpenGL ES 3.0,因为现在市场大部分是是 OpenGL ES 2.0,首先要找一个支持OpenGL ES 3.0的设备或者模拟器,所以顺带着对市面上支持的设备做了一些调查. 1.支持OpenGL ES 3.0设备 IOS从IOS7以上版本开始支持ES 3.0 *IOS                             GPU                                       CPU (1) iphone 5s            

OpenGL ES 3.0之Uniform详解

Uniform是变量类型的一种修饰符,是OpenGL ES  中被着色器中的常量值,使用存储各种着色器需要的数据,例如:转换矩阵.光照参数或者颜色. uniform 的空间被顶点着色器和片段着色器分享.也就是说顶点着色器和片段着色器被链接到一起进入项目,它们分享同样的uniform.因此一个在顶点着色器中声明的uniform,相当于在片段着色器中也声明过了.当应用程序装载uniform 时,它的值在顶点着色器和片段着色器都可用.在链接阶段,链接器将分配常量在项目里的实际地址,那个地址是被应用程序

OpenGL ES 3.0之VertexAttributes,Vertex Arrays,and Buffer Objects(九)

顶点数据,也称为顶点属性,指每一个顶点数据.指能被用来描述每个顶点的数据,或能被所有顶点使用的常量值.例如你想绘制一个具有颜色的立方体三角形.你指定一个恒定的值用于三角形的所有三个顶点颜色.但三角形的三个顶点位置是不同的,你需要指定一个顶点矩阵存储三个位置值. 指定顶点属性数据 顶点属性数据可以使用顶点数组或常量值指定每个顶点数据,OpenGL ES 3.0 必须至少支持16 个顶点属性.应用应该能够查询编译器支持的确切属性数.下面的程序指出如何查询. GLint maxVertexAttrib

Chapter 1 : OpenGLES 3.0 简介 (2)—— OpenGL ES 3.0

管道 如前所属,本书讲解的API版本是OpenGL ES 3.0.本书的目标是,深入讲解OpenGL ES 3.0的技术细节,给出具体的例子来说明如何使用某个特性,并且讨论了各种性能优化技术.当您读完这本书,您应该可以对OpenGL ES 3.0API有一个很好的把握.您将可以轻松的写出让人新服的OpenGL ES 3.0的应用程序,并且您不必通过阅读多种OpenGL ES的规范来搞懂某个特性是如何工作的. OpenGL ES 3.0实现了可编程着色图形管道.OpenGL ES 3.0规范包含两