翻译文
原文标题:Android Lesson Three: Moving to Per-Fragment Lighting
原文链接:http://www.learnopengles.com/android-lesson-three-moving-to-per-fragment-lighting/
使用每片段照明
欢迎来到第三课!这节课,我们将会在第二课的基础上, 学习如何使用每像素技术来达到相同的照明。 简单的正方体即使使用标准的漫射照明我们也能看到差异。 |
前提条件
本系列的每节课都以前面的课程为基础,本节课是第二课的补充,因此请务在阅读了之前的课程后再来回顾。
下面是本系列课程的前几课:
什么是每像素照明
随着着色器的使用,每像素照明在游戏中是一种相对较新的现象。许多有名的旧游戏,例如原版的半条命,都是在着色器之前开发出来的,主要使用静态照明,通过一些技巧模拟动态照明,使用每顶点(也称为Gouraud阴影)照明或其他技术,如动态光照贴图。
光照贴图可以提供非常好的效果,有时可以比单独的着色器提供更好的效果,因为可以预先计算昂贵的光线计算。但缺点是它们占用了大量内存并使用它们进行动态照明仅限于简单的计算。
使用着色器,现在很多这些计算转给GPU,这可以完成更多实时的效果。
从每顶点照明转移到每片段照明
这本课中,我们将针对每顶点解决方案和每片段解决方案查看相同的照明代码。尽管我将这种类型称为每像素,但在OpenGL ES中我们实际上使用片段,并且几个片段可以贡献一个像素的最终值。
手机的GPU变得越来越快,但是性能仍然是一个问题。对于“软”照明例如地形,每顶点照明可能足够好。确保您在质量和速度之间取得适当的平衡。
在某些情况下可以看到两种类型的照明之间的显著差异。看看下面的屏幕截图:
每顶点照明; 在正方形四个顶点为中心 |
每片段照明; 在正方形四个顶点为中心 |
在左图的每顶点照明中正方体的 正面看起来像是平面阴影,不能 表明附近有光源。这是因为正面 的四个顶点和光源距离差不多相 等,并且四个点的低光强度被简 单的插入两个三角形构成的正面。 相对比,每片段照明很好的 显示了亮点特性 |
每顶点照明; 在正方形角落 |
每片段照明; 在正方形角落 |
左图显示了一个Gouraud阴影 立方体。当光源移动到立方体正 面角落时,可以看到类似三角形 的效果。这是因为正面实际上是 由两个三角形组成,并且在每个 三角形不同方向插值,我们能看 到构成立方体的基础几何图形。 每片段的版本显示上没有此类插 值的问题并且它在边缘附近显示 了一个漂亮的圆形高光。 |
每顶点照明概述
我们来看看第二课讲的着色器;在该课程中可以找到详细的着色器说明。
顶点着色器
uniform mat4 u_MVPMatrix; // 一个表示组合model、view、projection矩阵的常量
uniform mat4 u_MVMatrix; // 一个表示组合model、view矩阵的常量
uniform vec3 u_LightPos; // 光源在眼睛空间的位置
attribute vec4 a_Position; // 我们将要传入的每个顶点的位置信息
attribute vec4 a_Color; // 我们将要传入的每个顶点的颜色信息
attribute vec3 a_Normal; // 我们将要传入的每个顶点的法线信息
varying vec4 v_Color; // 这将被传入片段着色器
void main() // 顶点着色器入口
{
// 将顶点转换成眼睛空间
vec3 modelViewVertex = vec3(u_MVMatrix * a_Position);
// 将法线的方向转换成眼睛空间
vec3 modelViewNormal = vec3(u_MVMatrix * vec4(a_Normal, 0.0));
// 将用于哀减
float distance = length(u_LightPos - modelViewVertex);
// 获取从光源到顶点方向的光线向量
vec3 lightVector = normalize(u_LightPos - modelViewVertex);
// 计算光线矢量和顶点法线的点积,如果法线和光线矢量指向相同的方向,那么它将获得最大的照明
float diffuse = max(dot(modelViewNormal, lightVector), 0.1);
// 根据距离哀减光线
diffuse = diffuse * (1.0 / (1.0 + (0.25 * distance * distance)));
// 将颜色乘以亮度,它将被插入三角形中
v_Color = a_Color * diffuse;
// gl_Position是一个特殊的变量用来存储最终的位置
// 将顶点乘以矩阵得到标准化屏幕坐标的最终点
gl_Position = u_MVPMatrix * a_Position;
}
片段着色器
precision mediump float; // 我们将默认精度设置为中等,我们不需要片段着色器中的高精度
varying vec4 v_Color; // 这是从三角形每个片段内插的顶点着色器的颜色
void main() // 片段着色器入口
{
gl_FragColor = v_Color; // 直接将颜色传递
}
正如您所见,大部分工作都在我们的着色器中做的。转移到每片段着色照明意味着,我们的片段着色器还有更多的工作要做。
实现每片段照明
以下是移动到每片段照明后的代码的样子。
顶点着色器 new
uniform mat4 u_MVPMatrix; // 一个表示组合model、view、projection矩阵的常量
uniform mat4 u_MVMatrix; // 一个表示组合model、view矩阵的常量
attribute vec4 a_Position; // 我们将要传入的每个顶点的位置信息
attribute vec4 a_Color; // 我们将要传入的每个顶点的颜色信息
attribute vec3 a_Normal; // 我们将要传入的每个顶点的法线信息
varying vec3 v_Position;
varying vec4 v_Color;
varying vec3 v_Normal;
// 顶点着色器入口点
void main()
{
// 将顶点位置转换成眼睛空间的位置
v_Position = vec3(u_MVMatrix * a_Position);
// 传入颜色
v_Color = a_Color;
// 将法线的方向转换在眼睛空间
v_Normal = vec3(u_MVMatrix * vec4(a_Normal, 0.0));
// gl_Position是一个特殊的变量用来存储最终的位置
// 将顶点乘以矩阵得到标准化屏幕坐标的最终点
gl_Position = u_MVPMatrix * a_Position;
}
顶点着色器比之前更加的简单。我们添加了两个线性插值变量用来传入到片段着色器:顶点位置和顶点法线。它们将在片段着色器计算光亮的时候被使用。
片段着色器 new
precision mediump float; //我们将默认精度设置为中等,我们不需要片段着色器中的高精度
uniform vec3 u_LightPos; // 光源在眼睛空间的位置
varying vec3 v_Position; // 插入的位置
varying vec4 v_Color; // 插入的位置颜色
varying vec3 v_Normal; // 插入的位置法线
void main() // 片段着色器入口
{
// 将用于哀减
float distance = length(u_LightPos - v_Position);
// 获取从光源到顶点方向的光线向量
vec3 lightVector = normalize(u_LightPos - v_Position);
// 计算光线矢量和顶点法线的点积,如果法线和光线矢量指向相同的方向,那么它将获得最大的照明
float diffuse = max(dot(v_Normal, lightVector), 0.1);
// 根据距离哀减光线
diffuse = diffuse * (1.0 / (1.0 + (0.25 * distance * distance)));
// 颜色乘以亮度哀减得到最终的颜色
gl_FragColor = v_Color * diffuse;
}
使用每片段照明,我们的片段着色器还有更多的工作要做。我们基本上将朗伯计算和哀减移到了每像素级别,这为我们提供了更逼真的照明,而无需添加更多顶点。
进一步练习
我们可以在顶点着色器中计算距离,然后赋值给变量通过线性插值传入片段着色器吗?
教程目录
- OpenGL Android课程一:入门
- OpenGL Android课程二:环境光和漫射光
- OpenGL Android课程三:使用每片段照明
- OpenGL Android课程四:介绍纹理基础
- OpenGL Android课程五:介绍混合(Blending)
- OpenGL Android课程六:介绍纹理过滤
打包教材
可以在Github下载本课程源代码:下载项目
本课的编译版本也可以再Android市场下:google play 下载apk
“我”也编译了个apk,方便大家下载:github download