1,
我们在计算环境光的时候,光的强度是唯一的影响因素。然后处理漫射光的时候公式中加入了光的方向参数。
镜面反射包含了上面所有的综合因素并且添加了一个新的元素:观察者的位置。
镜面反射时光以一定角度照射到物体表面,同时会在法线的另一侧对称的角度上反射出去,如果观察者刚好在反射光线的路径上那么就会看到格外强烈的光线。
镜面反射最终的结果是物体在从某个角度看上去会十分明亮,而移动开后这个光亮又会消失。现实中好的镜面反射的例子是金属物体,这些物体有时候看上去由于太亮了导致看不到他本来的颜色而是直接照向你眼睛的白色的亮光。但这种属性在其他的一些材料上是没有的(比如:木头)。很多东西根本不会发光,不管光源从什么角度照射以及观察者在什么位置。所以,镜面反射光的存在更取决于反射物体的材料性质而不是光源本身。
'I' 是入射光
'N' 是表面法线
'R' 反射光,和入射光'I'关于法线对称,但方向相反
'V' 是从入射光和反射光交点处(入射点)到观察者眼睛的向量,表示观察者视线
'α' 反射光'R'和观察者视线'V'的夹角
我们将使用夹角'α'来对镜面反射光现象进行建模。有一点可以看出当观察者视线和反射光重合时(夹角为0),反射光的强度大。观察者慢慢从反射光'R'移开时,夹角慢慢变大,而我们希望随着角度增大反射光要慢慢衰弱。明显,这里又要使用差积运算来计算夹角'α'的余弦值了,这个值将作为计算镜面反射光公式的反射参数。当'α'为0时余弦值为1,这是我们反射参数的大值。随着夹角'α'增大余弦值慢慢减小,直到夹角达到90度时就彻底没有镜面反射的效果了。当然,夹角大于90度时余弦值为负,也没有任何反射效果,也就是观察者不在反射光的路径范围内。
我们要用到'R'和'V'来计算夹角'α'。'V'可以通过世界坐标系中观察者位置和光的入射点位置的差计算得到。camera已经在世界空间进行维护了,我们只需要将它的位置传给shader着色器。
另外上面的图是经过简化了的模型,光在物体表面只有一个入射点(事实上不是,这里只是为了好分析)。
事实上,整个三角形都被点亮了(假设它面向光源),因此我们要计算每一个像素的镜面反射效果(和漫反射光的计算一样)。
我们必须要知道每个像素在世界空间的位置,这个不难:可以将顶点变换到世界空间,让光栅器对像素的世界空间位置进行插值并将结果传给片段着色器。事实上,这个和之前教程中对法线的处理操作是一样的。
最后是要使用'I'向量(由应用传给shader)来计算反射光线'R'。如下图:
首先要强调向量没有起点的概念,所有方向相同且长度相同的向量都是同一个向量。
因此,图中将入射光向量'I'复制到表面下面位置向量本身是不变的。
目标是求向量'R',根据向量的加法,'R'等于'I'+'V'。
'I'是已知的,所以我们要求'V'。
注意法线'N'的反向向量为'-N', 计算'I'和'-N'的点积可以得到'I'在'-N'上的投影,这也是'V'的模长度的一半。
另外'V'和'N'的方向是相同的,所以只要用计算的那个投影长度乘以单位向量'N'再乘以2就是向量'V'了。
GLSL提供了一个叫做'reflect'的内部函数就是做的上面这个计算。可以看下面这个函数在shader中的用法。这里得出计算镜面反射的最终公式:
开始先是将光的颜色和物体表面的颜色相乘,这个和在计算环境光以及漫反射光时一样。得到的结果再和材料的镜面反射强度参数('M')相乘。如果材料没有反射性能,比如木头,那么镜面反射参数就为0,整个公式的结果也就为0了,而像金属这种发光材料镜面反射能力就会很强。之后再乘以光线和观察者视线夹角的余弦值,这也是最后一个调整镜面反射光强度的参数('镜面参数'或者叫做'发光参数')。
这三个参数都是独立于光线本身的,因为当同一束光照到不同的材料上(比如:木头和金属)时会有不同的反射发光效果。
目前对材料属性的使用模型还是很局限的,同一个绘制回调的所有三角形会得到这些属性的一样的值。如果同一个模型的不同部分的三角形图元是不同的材料,这样就不合理了。在后面的教程中讲关于mesh网格的加载时我们会在一个模块中产生不同的镜面参数值并作为顶点缓冲器的一部分(而不是shader的一个参数),这样我们就可以在同一个绘制回调中使用不同的镜面光照参数来处理三角形图元。
2,
上面顶点着色器多了最后一行代码,世界变换矩阵(之前用来变换法线的那个世界变换矩阵)这里用来将顶点的世界坐标传给片段着色器。