http://blog.csdn.net/racehorse/article/details/6664717

简单的纹理贴图(Simple Texture)
为了在GLSL中应用纹理,我们需要访问每个顶点的纹理坐标。GLSL中提供了一些属性变量,每个纹理单元一个:

  1. attribute vec4 gl_MultiTexCoord0;
  2. attribute vec4 gl_MultiTexCoord1;
  3. attribute vec4 gl_MultiTexCoord2;
  4. attribute vec4 gl_MultiTexCoord3;
  5. attribute vec4 gl_MultiTexCoord4;
  6. attribute vec4 gl_MultiTexCoord5;
  7. attribute vec4 gl_MultiTexCoord6;
  8. attribute vec4 gl_MultiTexCoord7;

GLSL还为访问每个纹理的纹理矩阵提供了一个一致变量数组:

  1. uniform mat4 gl_TextureMatrix[gl_MaxTextureCoords];

顶点shader可以通过上面所示的内容访问OpenGL程序中指定的纹理坐标。然后必须为每个顶点计算纹理坐标,并保存在预先定义的易变变量gl_TexCoord[i]中,i表示纹理单元号。
下面这条语句直接复制OpenGL程序中指定的纹理坐标,作为纹理单元0的顶点纹理坐标。

  1. gl_TexCoord[0]  = gl_MultiTexCoord0;

下面是个简单的例子,在顶点shader中设置纹理单元0的纹理坐标。

  1. void main()
  2. {
  3. gl_TexCoord[0] = gl_MultiTexCoord0;
  4. gl_Position = ftransform();
  5. }

如果你想使用纹理矩阵,可以这样操作:

  1. void main()
  2. {
  3. gl_TexCoord[0] = gl_TextureMatrix[0] * gl_MultiTexCoord0;
  4. gl_Position = ftransform();
  5. }

前面说过,gl_TexCoord是一个易变变量,所以在片断shder中可以访问经过插值的纹理坐标。
为了访问纹理的数值,在片断shader中有必要声明一个特殊的变量,对一个2D纹理可以可以这样写:

  1. uniform sampler2D tex;

如果是1D或者3D的纹理,可以改成sampler1D和sampler3D。
这个用户定义的变量tex包含我们将会使用的纹理单元,通过texture2D函数我们可以得到一个纹素(texel),这是一个纹理图片中的像素。函数参数分别为simpler2D以及纹理坐标:

  1. vec4 texture2D(sampler2D, vec2);

函数的返回值已经考虑了所有在OpenGL程序中定义的纹理设置,比如过滤、mipmap、clamp等。
我们的片断shader可以写成如下形式:

  1. uniform sampler2D tex;
  2. void main()
  3. {
  4. vec4 color = texture2D(tex,gl_TexCoord[0].st);
  5. gl_FragColor = color;
  6. }

注意访问gl_TexCoord时选择子st的使用。在本教程前面关于数据类型和变量的讨论中说过,访问纹理坐标时可以使用如下选择子:s、t、p、q。(r因为和rgb选择子冲突而没有使用)

本节内容Shader Designer的工程下载地址:
http://www.lighthouse3d.com/wp-content/uploads/2011/03/textureSimple.zip
组合纹理与片断

OpenGL允许我们通过多种方式将纹理颜色和片断颜色联合到一起。下表显示了RGBA模式时可用的联合方式:

GL_REPLACE C = Ct A = At
GL_MODULATE C = Ct*Cf A = At*Af
GL_DECAL C = Cf * (1 – At) + Ct * At A = Af

表中Ct和At表示纹理的颜色和alpha值,Cf和Af表示片断(fragment)的颜色和alpha值,C和A表示最终的颜色和alpha值。
上一节的例子就相当于使用了GL_REPLACE模式。下面我们我们准备在一个立方体上实现与GL_MODULATE等同的效果。两个shader只计算使用一个白色方向光的散射以及环境光成分,关于材质的完整定义请参照光照有关的章节。
因为使用了光照,所以顶点shader中必须处理法线信息。必须将法线变换到视图空间然后归一化,光线方向向量也必须归一化(光线方向向量已经由OpenGL变换到了视图空间)。现在新的顶点shader如下:

  1. varying vec3 lightDir,normal;
  2. void main()
  3. {
  4. normal = normalize(gl_NormalMatrix * gl_Normal);
  5. lightDir = normalize(vec3(gl_LightSource[0].position));
  6. gl_TexCoord[0] = gl_MultiTexCoord0;
  7. gl_Position = ftransform();
  8. }

在片断shader中,光照得到的片断的颜色和alpha值在cf和af中分别计算。shader中剩余代码按照GL_MODULATE的公式计算:

  1. varying vec3 lightDir,normal;
  2. uniform sampler2D tex;
  3. void main()
  4. {
  5. vec3 ct,cf;
  6. vec4 texel;
  7. float intensity,at,af;
  8. intensity = max(dot(lightDir,normalize(normal)),0.0);
  9. cf = intensity * (gl_FrontMaterial.diffuse).rgb +
  10. gl_FrontMaterial.ambient.rgb;
  11. af = gl_FrontMaterial.diffuse.a;
  12. texel = texture2D(tex,gl_TexCoord[0].st);
  13. ct = texel.rgb;
  14. at = texel.a;
  15. gl_FragColor = vec4(ct * cf, at * af);
  16. }

Shader Designer的工程下载地址:
http://www.lighthouse3d.com/wp-content/uploads/2011/03/textureComb.zip

多重纹理
在GLSL中实现多重纹理十分容易,我们只需要访问所有纹理即可。因为我们打算给每个纹理使用相同的纹理坐标,所以顶点shader不需要改动。片断shader中只需要进行些许改动,加上多个纹理的颜色值。

  1. varying vec3 lightDir,normal;
  2. uniform sampler2D tex;
  3. void main()
  4. {
  5. vec3 ct,cf;
  6. vec4 texel;
  7. float intensity,at,af;
  8. intensity = max(dot(lightDir,normalize(normal)),0.0);
  9. cf = intensity * (gl_FrontMaterial.diffuse).rgb +
  10. gl_FrontMaterial.ambient.rgb;
  11. af = gl_FrontMaterial.diffuse.a;
  12. texel = texture2D(tex,gl_TexCoord[0].st) +
  13. texture2D(l3d,gl_TexCoord[0].st);
  14. ct = texel.rgb;
  15. at = texel.a;
  16. gl_FragColor = vec4(ct * cf, at * af);
  17. }

效果如下:

下面添加点不同的效果:在黑暗中发光。我们希望第二个纹理能在黑暗中发光,在没有光照时达到最亮,在有光照时变暗。

我们通过两步计算最终的颜色:首先将第一个纹理与片断颜色进行modulate计算,然后根据光照强度(indensity)加上第二个纹理单元。
如果indensity是0,第二个纹理单元取最大值,如果indensity为1,只取第二个纹理单元颜色的10%,当indensity在0和1之间时按这两个大小进行插值。可以使用smoothstep函数实现这个要求:

  1. genType smoothStep(genType edge0, genType edge1, genType x);

如果x <= edge0结果是0,如果x >= edge1结果为1,如果edge0 < x < edge1结果在0和1之间进行Hermite插值。在本例中我们按如下方式调用:
coef = smoothStep(1.0, 0.2, intensity);
下面的片断shader实现了需要的效果:

  1. varying vec3 lightDir,normal;
  2. uniform sampler2D tex,l3d;
  3. void main()
  4. {
  5. vec3 ct,cf,c;
  6. vec4 texel;
  7. float intensity,at,af,a;
  8. intensity = max(dot(lightDir,normalize(normal)),0.0);
  9. cf = intensity * (gl_FrontMaterial.diffuse).rgb +
  10. gl_FrontMaterial.ambient.rgb;
  11. af = gl_FrontMaterial.diffuse.a;
  12. texel = texture2D(tex,gl_TexCoord[0].st);
  13. ct = texel.rgb;
  14. at = texel.a;
  15. c = cf * ct;
  16. a = af * at;
  17. float coef = smoothstep(1.0,0.2,intensity);
  18. c += coef *  vec3(texture2D(l3d,gl_TexCoord[0].st));
  19. gl_FragColor = vec4(c, a);
  20. }

Shader Designer的工程下载地址:
http://www.lighthouse3d.com/wp-content/uploads/2011/03/textureGlow.zip

【GLSL教程】(八)纹理贴图 【转】的更多相关文章

  1. GLSL纹理贴图 【转】

    转载:http://blog.csdn.net/hgl868/article/details/7872466 简单的纹理贴图(Simple Texture) 为了在GLSL中应用纹理,我们需要访问每个 ...

  2. (转载)Cocos2dx-OpenGL ES2.0教程:纹理贴图(6)

    在上一篇文章中,我们介绍了如何绘制一个立方体,里面涉及的知识点有VBO(Vertex Buffer Object).IBO(Index Buffer Object)和MVP(Modile-View-P ...

  3. IOS 中openGL使用教程3(openGL ES 入门篇 | 纹理贴图(texture)使用)

    在这篇文章中,我们将学习如何在openGL中使用纹理贴图. penGL中纹理可以分为1D,2D和3D纹理,我们在绑定纹理对象的时候需要指定纹理的种类.由于本文将以一张图片为例,因此我们为我们的纹理对象 ...

  4. Three.js基础探寻八——法向材质与材质的纹理贴图

    4.法向材质 法向材质可以将材质的颜色设置为其法向量的方向,有时候对于调试很有帮助. 法向材质的设定很简单,甚至不用设置任何参数: new THREE.MeshNormalMaterial() 材质的 ...

  5. 翻译:GLSL的顶点位移贴图

    翻译:GLSL的顶点位移贴图 翻译自: Vertex Displacement Mapping using GLSL 译者: FreeBlues 说明: 之所以选择这篇文档, 是因为现在但凡提到位移贴 ...

  6. 【GLSL教程】(一)图形流水线 【转】

    http://blog.csdn.net/racehorse/article/details/6593719 这是一些列来自lighthouse3d的GLSL教程,非常适合入门.我将边学习边翻译该教程 ...

  7. Esfog_UnityShader教程_NormalMap法线贴图

    咳咳,好久没有更新了,一来是这段时间很忙很忙,再来就是自己有些懒了,这个要不得啊,赶紧补上.在前面我们已经介绍过了漫反射和镜面反射,这两个是基本的光照类型,仅仅依靠它们就想制作出精美的效果是远远不够的 ...

  8. Android OpenGL ES 开发(九): OpenGL ES 纹理贴图

    一.概念 一般说来,纹理是表示物体表面的一幅或几幅二维图形,也称纹理贴图(texture).当把纹理按照特定的方式映射到物体表面上的时候,能使物体看上去更加真实.当前流行的图形系统中,纹理绘制已经成为 ...

  9. OpenGL ES课程VI之纹理贴图(原文对照)

    http://www.educity.cn/wenda/92368.html OpenGL ES教程VI之纹理贴图(原文对照) OpenGL ES Tutorial for Android – Par ...

随机推荐

  1. BZOJ 4326:NOIP2015 运输计划(二分+差分+lca)

    NOIP2015 运输计划Description公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航道建立在两个星球之间,这 n−1 条航道连通了 L 国的所 ...

  2. PIC单片机之时钟设置

    PIC单片机之时钟设置 http://blog.csdn.net/superanters/article/details/8541650 内部时钟和外部时钟? PIC单片机有许多型号可以设置成 用外部 ...

  3. [luogu1707] 刷题比赛 [矩阵快速幂]

    题面: 传送门 思路: 一眼看上去是三个递推......好像还挺麻烦的 仔细观察一下,发现也就是一个线性递推,但是其中后面的常数项比较麻烦 观察一下,这里面有以下三个递推是比较麻烦的 第一个是$k^2 ...

  4. 转JSON提示No serializer found for class

    在调用Json串生成方法时,提示: No serializer found for class com.jeremxy.domain.EpgDetail and no propertiesdiscov ...

  5. 【06】Vue 之 组件化开发

    组件其实就是一个拥有样式.动画.js逻辑.HTML结构的综合块.前端组件化确实让大的前端团队更高效的开发前端项目.而作为前端比较流行的框架之一,Vue的组件和也做的非常彻底,而且有自己的特色.尤其是她 ...

  6. Page_Load与Page_PreRender的执行顺序

    原文发布时间为:2009-10-25 -- 来源于本人的百度文章 [由搬家工具导入] Page_PreRender 服务器控件将要呈现给其包含的 控件时发生。简单的理解为page中的控件渲染调用此事件 ...

  7. OpenGL入门学习(二)

    http://developer.178.com/201103/94954661733.html 一.点.直线和多边形我们知道数学(具体的说,是几何学)中有点.直线和多边形的概念,但这些概念在计算机中 ...

  8. C语言字符串操作总结大全(超详细)【转】

    转自:http://www.jb51.net/article/37410.htm )字符串操作 strcpy(p, p1) 复制字符串 strncpy(p, p1, n) 复制指定长度字符串 strc ...

  9. netbeans8.2下struts2的Java Web开发Demo1

    struts2框架主要是封装了servlet,简化了jsp跳转的复杂操作,并且提供了易于编写的标签,可以快速开发view层的代码. 过去,我们用jsp和servlet搭配,实现展现时,大体的过程是: ...

  10. JSON Web Token的使用

    定义 JSON Web Token(JWT)是一个非常轻巧的规范.这个规范允许我们使用JWT在用户和服务器之间传递安全可靠的信息. 适用场景 1.用于向Web应用传递一些非敏感信息.例如完成加好友.下 ...