前言
上两篇文章里我们分别绘制了最简单的三角形和纹理图片
下面来讲一下纹理取样,混合,和多重纹理
纹理取样,循环
示例代码来源于下面这本书,
OpenGL ES应用开发实践指南:iOS卷
纹理取样设置函数
1 | glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER, GL_NEAREST) |
GL_TEXTURE_MAG_FILTER
用于多个纹素对应一个顶点即片元时候的处理方式,GL_NEAREST
是取最近的纹素,GL_LINEAR
则是取这多个纹素的混合结果
1 | glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); |
GL_TEXTURE_MAG_FILTER
参数用于没有足够的可用纹素来唯一性的映射一个或者多个纹素到每个片元时配置取样. GL_NEAREST是取最近的纹素,GL_LINEAR则是取附近多个纹素的混合结果
GL_LINEAR的直观显示效果就是图片模糊的渲染了。
我们知道顶点的坐标系U,V坐标和纹理的S,T坐标一一映射对用,如果U,V大于1,或者小于0,也就是超出了纹理坐标系,我们可以设置取样边缘的纹素,或者重复纹理取样
1 | glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);//取样纹理边缘的纹素 |
接下来我们用下面这个Demo来看下实际效果
这里列举下核心代码
首先是顶点数组
1 | typedef struct { |
属性
1 | @interface OpenGLES_3_2ViewController(){ |
ViewDidload中初始化context和baseEffect以及load顶点缓存和纹理
1 | self.preferredFramesPerSecond = 60; |
loadVertexBuffer和loadTexture
1 | - (void)loadVertexBuffer{ |
绘制部分代码
1 | - (void)glkView:(GLKView *)view drawInRect:(CGRect)rect{ |
核心部分,我们在系统方法update里面更新顶点坐标和纹理取样设置参数,系统update方法调用频率和系统屏幕帧数一致
1 | - (void)update{ |
需要注意的是,我们更新完了数据源顶点的坐标需要重新glBufferData刷新GPU顶点缓存。
以上呢我们就实现了一个基于OPENGL ES2.0的简单动画,并且呢直观演示了纹理取样的不同参数的实际效果。
纹理混合
之前的Demo我们都是绘制了一张图片,如果是多个图片,也就是多个纹理的绘制如何处理呢。
OpenGL支持纹理混合,开启纹理混合非常的简单。
常用的混合调用以下函数
1 | //开启混合 |
GL_ONE_MINUS_SRC_ALPHA该模式是让源片元的透明度元素和正在更新的像素的颜色元素相乘。
GL_SRC_ALPHA用于让源片元的透明度元素和其他的片元的透明度元素依次相乘。
那么帧缓存最终的像素颜色计算公式如下
好,我们就写一个Demo来做一下纹理混合
核心代码如下
1 | - (void)fillTexture{ |
这里开启GL_BLEND
绘制部分代码如下
1 | - (void)glkView:(GLKView *)view drawInRect:(CGRect)rect{ |
我们需要依次绘制纹理1和纹理2,运行结果图如下
图2倍绘制在了图1的上方,这取决于纹理的绘制顺序。
这种通过多次读写像素颜色渲染缓存叫做多通道渲染。
多通道渲染从性能上来看,每次更新界面图形都需要渲染多次,需要从帧缓存读取颜色数据和片元数据混合,再次写回帧缓存,显然多次的内存读取决定了混合在性能上是不佳的,是次优选择
多重纹理
目前CPU都支持同时从至少两个纹理缓存中取样纹素,也就是多重纹理,从而可以替代纹理混合,优化性能。
GLkit中GLKBaseEffect同时支持两个纹理。
核心代码如下
1 | - (void)fillTexture{ |
这里创建了两个纹理缓存并且设置了纹理2的envMode为GLKTextureEnvModeDecal,GLKTextureEnvModeDecal是和glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);效果一样的。计算公式一致
1 | - (void)fillVertexArray{ |
这里设置顶点纹理数据指针偏移时候开启了GLKVertexAttribTexCoord0 和GLKVertexAttribTexCoord1两个纹理
绘制部分
1 | - (void)glkView:(GLKView *)view drawInRect:(CGRect)rect{ |
这里多重纹理就避免了重复和多次绘制,性能比纹理混合要好,是绘制多个纹理时候的优先选择。
Demo代码地址:LearnOpenGLESDemo
源码来源于书籍:1. OpenGL ES应用开发实践指南:iOS卷