OpenGL 从入门到成魔-第7章-纹理和纹理坐标

注:参考自bilibili系列视频,OpenGL 从入门到成魔-第7章-纹理和纹理坐标,更详细的内容可以从视频获取https://www.bilibili.com/video/BV1bZ4y1W7tX

纹理介绍

我们可以为每个顶点添加颜色来增加图形的细节,从而创建出有趣的图像。但是,如果想让图形看起来更真实,我们就必须有足够多的顶点,从而指定足够多的颜色。这将会产生很多额外开销,因为每个模型都会需求更多的顶点,每个顶点又需求一个颜色属性。程序员们更喜欢用纹理,纹理是一个2D图片,他可以用来添加物体的细节。我们可以在一张图片上插入非常多的细节,这样就可以让物体非常精细而不用指定额外的顶点。

为了能够把纹理映射(Map)到三角形上,我们需要指定三角形的每个顶点各自对应纹理的哪个部分。这样每个顶点就会关联着一个纹理坐标(Texture Coordinate),用来标明该从纹理图像的哪个部分采样(使用纹理坐标获取纹理颜色叫做采样)。之后在图形的其它片段上进行片段插值(Fragment Interpolation)。

纹理在shader中的表示

GLSL有一个供纹理对象使用的内建数据类型,叫做采样器(Sampler)。我们可以声明一个uniform sampler2D texture把一个纹理添加到片段着色器中,通过在CPU调用 glUniform1f()函数,向texture传值,从而实现载入纹理(图片)到Shader。

Texture 的函数调用

image.png

image.png
  1. glGenTextures(1, &textureId) 创建texture
  2. glBindTexture(GL_TEXTURE_2D, textureId) 绑定texture
  3. glTexParameteri() 必须设置的四个选项
  4. glPixelStorei(GL_UNPACK_ALIANMENT, 1); 默认是4字节对齐,如果图像宽度不是4的倍数,会补齐到4字节,图像会有偏移。这里我们设置成1字节对齐,这样会影响性能。如果到了性能瓶颈,这里可以选择为4字节对齐。
  5. glTexImage2D(...)
  • 第一个参数指定了纹理目标(Target)。设置为GL_TEXTURE_2D意味着会生成与当前绑定的纹理对象在同一个目标上的纹理(任何绑定到GL_TEXTURE_1D和GL_TEXTURE_3D的纹理不会受到影响)。
  • 第二个参数为纹理指定多级渐远纹理的级别,如果你希望单独手动设置每个多级渐远纹理的级别的话。这里我们填0,也就是基本级别。
  • 第三个参数告诉OpenGL我们希望把纹理储存为何种格式。即在GPU中图片存储形式,主要指定几个通道。
  • 第四个和第五个参数设置最终的纹理的宽度和高度。
  • 下个参数应该总是被设为0(历史遗留的问题)。
  • 第七个参数定义了源图的格式,即源图几个通道。
  • 第八个参数定义了源图每个通道的数据类型BYTE。
  • 最后一个参数是真正的图像数据。

当调用glTexImage2D时,当前绑定的纹理对象就会被附加上纹理图像。该函数最终会把图像数据从CPU端,搬到GPU端,即第一步创建的texture中。

创造一张RGB图片

int width = 2;
int height = 2;
unsigned char imgData[] = {
255,0,0,         0,255,0,
0,0,255,         127,127,127
}

由此,看出图片的本真。
传到上面函数的最后一个参数中。

纹理传入shader

  1. 激活纹理单元
glActiveTexture(GL_TEXTURE0);
  1. 绑定纹理
glBindTexture(GL_TEXTURE_2D, texture);
  1. 向shader中的uniform sampler2D texture变量传值, 值为0,0是纹理单元编号GL_TEXTURE0
glUniform1i(glGetUniformLocation(program, "texture"), 0);

首先,0号纹理单元和纹理绑定,然后0号纹理单元和shader里的sampler2D texture变量绑定,从而做好对应。
一个shader中,最多处理32个纹理单元,openGL es,最多处理16个。

纹理坐标

image.png

GLSL内建的texture函数来采样纹理的颜色,它第一个参数是纹理采样器,第二个参数是对应的纹理坐标。texture函数的返回值就是,在该纹理坐标上的rgba值(vec4)。

  1. 根据坐标轴,转换出纹理坐标。定义在顶点着色器中,原因是纹理坐标涉及到插值,所以要在插值前传入。注:纹理坐标本身是2维的,但是为了使用vao addVertex3D(),这里定义成了3维
 float vertexsUV[] =
    {
          1.0f,  1.0f, 0.0f,
          0.0f,  1.0f, 0.0f,
          0.0f,  0.0f, 0.0f,
          1.0f,   0.0f, 0.0f,
    };
  1. 把纹理坐标传到GPU。像顶点位置坐标一样,向VAO中添加一个VBO,调用addVertex3D,注意第三个参数是layout,我们这里是第二个顶点属性(顶点着色器的 每个输入变量叫顶点属性),故填1。
    VAO->addVertex3D(vetexsUV, 4, 1);
  1. 着色器接收纹理坐标。顶点着色器中,aTexCoord接收vertexUV, 并传递到片段着色器中去使用。我们使用GLSL内建的texture函数来采样纹理的颜色,它第一个参数是纹理采样器,第二个参数是对应的纹理坐标。texture函数会使用之前设置的纹理参数对相应的颜色值进行采样。这个片段着色器的输出就是纹理的(插值)纹理坐标上的(过滤后的)颜色。
       //顶点着色器 
       #version 330\n
        layout(location = 0) in vec3 pos;
        layout(location = 1) in vec3 aTexCoord;
        out vec3 outPos;
        out vec3 TexCoord;
        void main()
        {
            outPos = pos;
            gl_Position = vec4(pos, 1.0);
            TexCoord = aTexCoord;
        }
     
      //片段着色器
       #version 330\n
        out vec4 rgbaColor;
        in vec3 outPos;
        in vec3 TexCoord;

        uniform sampler2D t;
        void main()
        {
            vec2 uv = vec2(TexCoord.x, TexCoord.y);
            rgbaColor = texture(t, uv);
        }

图片反了

此时,我们的纹理已经绘制出来了,但是上下颠倒了。这是因为OpenGL纹理坐标系中左下角为原点(0,0),而图片的原点在左上角,所以在y轴上颠倒了。stb_image.h(图像加载库,需引入)能够在图像加载时帮助我们翻转y轴,只需要在加载任何图像前加入以下语句即可:

stbi_set_flip_vertically_on_load(true);

若没有用图像加载库,就把纹理坐标上下颠倒一下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349