OpenGL ES 环境搭建

2020-09-11 15:49:52 浏览数 (2)

在Android上运行OpenGL ES程序需要用到GLSurfaceView控件,GLSurfaceView继承自SurfaceView并实现了GLThread,通过OpenGL ES进行绘制。

OpenGL ES 与Android版本对应关系

  • OpenGL ES1.0是基于OpenGL 1.3的,OpenGL ES1.1是基于OpenGL 1.5的。Android1.0和更高的版本支持这个API规范。OpenGL ES 1.x是针对固定硬件管线的。
  • OpenGL ES2.0是基于OpenGL 2.0的,不兼容OpenGL ES 1.x。Android 2.2(API 8)和更高的版本支持这个API规范。OpenGL ES 2.x是针对可编程硬件管线的。
  • OpenGL ES3.0的技术特性几乎完全来自OpenGL 3.x的,向下兼容OpenGL ES 2.x。Android 4.3(API 18)及更高的版本支持这个API规范。
  • OpenGL ES3.1基本上可以属于OpenGL 4.x的子集,向下兼容OpenGL ES3.0/2.0。Android 5.0(API 21)和更高的版本支持这个API规范。

环境搭建

01

设置OpenGL ES的版本

Android工程中OpenGL ES的版本在AndroidManifest.xml中指定:

代码语言:javascript复制
<uses-feature android:glEsVersion="0x00020000" android:required="true" />

0x00020000表示支持OpenGL ES 2.0。

02

初始化GLSurfaceView

在布局文件中添加GLSurfaceView,

代码语言:javascript复制
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <android.opengl.GLSurfaceView
        android:id="@ id/glSurfaceView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</androidx.constraintlayout.widget.ConstraintLayout>

在Activity中初始化GLSurfaceView,

代码语言:javascript复制
class TriangleActivity : AppCompatActivity() {
    
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        //设置opengl es版
        glSurfaceView.setEGLContextClientVersion(2)
        //设置renderer
        glSurfaceView.setRenderer(MyRenderer(context = baseContext))
        //设置渲染模式
        glSurfaceView.renderMode = GLSurfaceView.RENDERMODE_CONTINUOUSLY
    }

    override fun onResume() {
        super.onResume()
        glSurfaceView.onResume()
    }

    override fun onPause() {
        super.onPause()
        glSurfaceView.onPause()
    }
}

OpenGL ES版本号和AndroidManifest.xml中版本号保持一致,当然我们也可以在设置版本之前判断当前设备是否支持设置的版本,下面的代码判断是支持ES 2.0版本。

代码语言:javascript复制
 fun supportsEs2(context: Context): Boolean {
        val configurationInfo =
            (context.getSystemService(Context.ACTIVITY_SERVICE) as ActivityManager).deviceConfigurationInfo
        return configurationInfo.reqGlEsVersion >= 0x20000
    }

setRenderMode方法是设置GLSurfaceView渲染模式,渲染模式有RENDERMODE_WHEN_DIRTY和RENDERMODE_CONTINUOUSLY两种,RENDERMODE_WHEN_DIRTY表示当需要的时候才渲染,只有在调用requestRender或者onResume等方法时才渲染,RENDERMODE_CONTINUOUSLY表示一直渲染。

setRenderMode一定要在setRenderer方法之后调用,另外一般需要在Activity或者Fragment的onPause和onResume生命周期中调用GLSurfaceView的onPause和onResume方法,节省系统资源。

03

Renderer

Renderer必须实现GLSurfaceView.Renderer接口,并实现onSurfaceCreated ,onDrawFrame,onSurfaceChanged方法,OpenGL ES的渲染工作由此Renderer实现,MyRenderer的实现如下:

代码语言:javascript复制
class MyRenderer(val context: Context) : GLSurfaceView.Renderer {
        override fun onDrawFrame(p0: GL10?) {
        }
        override fun onSurfaceChanged(p0: GL10?, width: Int, height: Int){
        }
        override fun onSurfaceCreated(p0: GL10?, p1: EGLConfig?) {
        }
}

onSurfaceCreated ,onDrawFrame,onSurfaceChanged方法说明如下:

  • onSurfaceCreated:GLSurfaceView创建完成,也代表OpenGL ES环境创建完成,通常情况下在此方法中创建Program及初始化参数。
  • onSurfaceChanged:当Surface发生变化的时候回调,比如竖屏转横屏导致GLSurfaceView大小发生变化,通常情况下在此方法中设置绘制窗口及和GLSurfaceView大小有关系的参数。
  • onDrawFrame:执行OpenGL ES渲染工作,由系统以一定的频率来调用重绘View,当设置GLSurfaceView的渲染模式为GLSurfaceView.RENDERMODE_CONTINUOUSLY或不设置时,系统就会主动回调onDrawFrame()方法, 如果设置为 RENDERMODE_WHEN_DIRTY ,手动调用requestRender(),才会渲染。

注意:所有OpenGL相关的操作必须在GLThread线程中执行,无法在主线程(UI线程)中执行,这是OpenGL ES开发中经常遇到的问题。Renderer的onSurfaceCreated ,onDrawFrame,onSurfaceChanged方法都是运行在GLThread线程。

渲染

在OpenGL ES中Shader和Program是两个非常重要的概念,Program需要Vertex Shader(顶点Shader和Fragment Shader(片段Shader),Renderer的渲染就是在执行Program。

  • Vertex Shader(顶点Shader)处理顶点数据,对于发送给GPU的每一个顶点都要执行一次Vertex Shader,它的作用就是把顶点在虚拟空间中的三维坐标变换为屏幕上的二维坐标,并带有深度信息。
  • Fragment Shader计算每个像素的颜色和其他属性。它通过应用光照值、凹凸贴图,阴影,镜面高光,半透明等处理来计算像素的颜色并输出。

01

Program与Shader

Shader可以以字符串形式存在也可以单独存放在文件中,建议写在assets目录下并以.glsl结尾,因为Android Studio安装GLSL插件可以高亮其代码,便于查找错误。

在assets下创建glsl文件夹,用于存放glsl文件,创建triangle_vertex.glsl文件,保存Vertex Shader代码:

代码语言:javascript复制
attribute vec4 vPosition;
void main() { 
  gl_Position = vPosition;
}

获取你还看不懂这段代码,没关系,你只需要知道这段代码是处理顶点数据即可,GLSL 语言相关知识后面会介绍。

创建triangle_fragment.glsl文件,保存Fragment Shader代码:

代码语言:javascript复制
precision mediump float;
void main()
{
    gl_FragColor = vec4(1,0,0,1);
}

上面代码表示顶点区域内绘制为红色,vec4内的值表示r,g,b,a。

将上面2个shader文件编译为Shader,

代码语言:javascript复制
private fun compileShader(shaderType: Int, shaderSource: String): Int {
        //创建一个空shader
        var shaderHandle: Int = GLES20.glCreateShader(shaderType)
        if (shaderHandle != 0) {
            //加载shader源码
            GLES20.glShaderSource(shaderHandle, shaderSource)
            //编译shader
            GLES20.glCompileShader(shaderHandle)
            val compileStatus = IntArray(1)
            //检查shader状态
            GLES20.glGetShaderiv(shaderHandle, GLES20.GL_COMPILE_STATUS, compileStatus, 0)
            if (compileStatus[0] == 0) {
                //输入shader异常日志
                Log.e(TAG, "Error compile shader:${GLES20.glGetShaderInfoLog(shaderHandle)}")
                //删除shader
                GLES20.glDeleteShader(shaderHandle)
                shaderHandle = 0
            }
        }
        if (shaderHandle == 0) {
            Log.e(TAG, "Error create shader")
        }
        return shaderHandle
}

ShaderType分为GLES20.GL_VERTEX_SHADER和GLES20.GL_FRAGMENT_SHADER,GLES20.GL_VERTEX_SHADER编译Vertex Shader的,GLES20.GL_ FRAGMENT _SHADER编译Fragment Shader。

将Shader链接到program,

代码语言:javascript复制
fun createAndLinkProgram(vertexCode: String, fragmentCode: String): Int {
        //创建一个空的program
        var programHandle = GLES20.glCreateProgram()
        if (programHandle != 0) {
            //编译shader
            val vertexShaderHandle = compileShader(GLES20.GL_VERTEX_SHADER, vertexCode)
            val fragmentShaderHandle = compileShader(GLES20.GL_FRAGMENT_SHADER, fragmentCode)
            //绑定shader和program
            GLES20.glAttachShader(programHandle, vertexShaderHandle)
            GLES20.glAttachShader(programHandle, fragmentShaderHandle)
            //链接program
            GLES20.glLinkProgram(programHandle)
            
            val linkStatus = IntArray(1)
            //检测program状态
            GLES20.glGetProgramiv(programHandle, GLES20.GL_LINK_STATUS, linkStatus, 0)
            if (linkStatus[0] == 0) {
                Log.e(TAG, "Error link program:${GLES20.glGetProgramInfoLog(programHandle)}")
                //删除program
                GLES20.glDeleteProgram(programHandle)
                programHandle = 0
            }
        }
        if (programHandle == 0) {
            Log.e(TAG, "Error create program")
        }
        return programHandle
}

最终返回program的句柄,这2步是固定的,因为我们将其封装为工具类GLTools,供以后使用。

使用工具类GLTools创建program:

代码语言:javascript复制
private fun createProgram() {
            var vertexCode =
                AssetsUtils.readAssetsTxt(
                    context = context,
                    filePath = "glsl/triangle_vertex.glsl"
                )
            var fragmentCode =
                AssetsUtils.readAssetsTxt(
                    context = context,
                    filePath = "glsl/triangle_fragment.glsl"
                )
            mProgramHandle = GLTools.createAndLinkProgram(vertexCode, fragmentCode)
        }

program的创建放在Renderer的onSurfaceCreated方法中,创建成功后,获取Shader中参数句柄及设置顶点数据。获取Vertex Shader中vPosition句柄:

代码语言:javascript复制
val loc = GLES20.glGetAttribLocation(mProgramHandle, "vPosition")

顶点坐标系统如下图:

顶点坐标轴以屏幕中心为原点(0,0),z轴的正方向为穿透屏幕指向外面。三角形的顶点坐标设置如下:

代码语言:javascript复制
var vertexBuffer = GLTools.array2Buffer(
            floatArrayOf(
                0.0f, 0.5f, 0.0f, // top
                -0.5f, -0.5f, 0.0f, // bottom left
                0.5f, -0.5f, 0.0f  // bottom right
            )
        )

工具类GLTools中array2Buffer是将顶点数据转换为FloatBuffer,array2Buffer方法定义如下:

代码语言:javascript复制
fun array2Buffer(array: FloatArray): FloatBuffer {
        val bb = ByteBuffer.allocateDirect(array.size * 4)
        bb.order(ByteOrder.nativeOrder())
        var buffer = bb.asFloatBuffer()
        buffer.put(array)
        buffer.position(0)
        return buffer
    }

02

创建OpenGL ES绘制窗口

创建OpenGL ES绘制窗口通常是在onSurfaceChanged中设置,

代码语言:javascript复制
GLES20.glViewport(0, 0, width, height)
  • 第一个参数(x):表示窗口x坐标,屏幕左上角为原点。
  • 第二个参数(y):表示窗口y坐标,(0,0)表示屏幕左上角。
  • 第三个参数(width):表示窗口的宽。
  • 第四个参数(height):表示窗口的高

03

onDrawFrame

绘制在onDrawFrame中执行,

代码语言:javascript复制
 override fun onDrawFrame(p0: GL10?) {
        GLES20.glUseProgram(mProgramHandle)
        GLTools.setAttributePointer(vPositionLoc, vertexBuffer, 3)
        GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, 3)
    }

GLES20.glUseProgram(mProgramHandle)表示启动当前program,mProgramHandle是上面创建Program返回的句柄。

GLTools.setAttributePointer(vPositionLoc, vertexBuffer, 3)表示将顶点数据设置给program,参数说明情况如下:

  • Location:vPosition的句柄。
  • buffers :生成的顶点数据。
  • pointSize :每个顶点个数,上面顶点的个数是3个。

setAttributePointer为封装的工具类方法:

代码语言:javascript复制
fun setAttributePointer(location: Int, buffers: FloatBuffer, pointSize: Int) {
        buffers.position(0)
        GLES20.glEnableVertexAttribArray(location)
        GLES20.glVertexAttribPointer(location, pointSize, GLES20.GL_FLOAT, false, 0, buffers)
    }

glDrawArrays方法是绘制,参数说明情况如下:

  • 第一个参数mode,表示绘制的方式,可选择的值有:GL_POINTS、GL_LINES、GL_LINE_LOOP、GL_LINE_STRIP、GL_TRIANGLES、GL_TRIANGLE_STRIP、GL_TRIANGLE_FAN。
  • 第二个参数表示从数组缓存中的哪一位开始绘制,一般为0。
  • 第三个参数表示绘制顶点的数量。

0 人点赞