整整忙了一个月了,总算清闲下来了,从上次写完环境光后又过了这么长时间,继续学习......加油!!今天整理下漫反射光并记录下来,那就直接进入主题吧,开始漫反射光的学习。
漫反射光是在环境光的基础上添加一个定向光让物体看起来更真实,定向光照射到物体表面,会根据物体表面不同的凹凸程度呈现出强度差异的反射光,让物体看起来更有立体感;当光线方向固定后,决定反射光强度的就是物体表面的法线方向。如果光线方向平行于物体表面(即和物体表面的法线垂直时)这时候的反射光是最弱的,如果光线方向垂直于物体表面(即和物体表面的法线平行时)这时候的反射光是最强的。
漫反射光的公式:I = Aintensity*Acolor + Dintensity*Dcolor *N.L
公式说明:前面的Aintensity*Acolor仍然是环境光的公式,后面一部分表示添加的反射光,其中Dintensity表示光照强度,Dcolor 表示颜色,N表示物体表面的法线,L表示光线方向,N.L则是N和L两向量的点乘,等价于:N.L=|N|*|L|*cos(a)(a为两法线夹角).注意:L的方向是自顶点指向光源的。
下面看看相对于环境光代码相关变动的地方,除了Draw中的变化,其他代码均无变化,所以这里主要看看Draw里面的代码
effect.CurrentTechnique = effect.Techniques["DiffuseLight"]; //读取fx foreach (EffectPass pass in effect.CurrentTechnique.Passes) { pass.Apply(); foreach (ModelMesh mesh in model.Meshes) { foreach (ModelMeshPart part in mesh.MeshParts) { effect.Parameters["World"].SetValue(modelTransform[mesh.ParentBone.Index] * Matrix.CreateScale(30.0f) * Matrix.CreateRotationY((float)angle) * Matrix.CreateRotationX((float)angle)); effect.Parameters["View"].SetValue(viewMatrix); effect.Parameters["Projection"].SetValue(projectionMatrix); effect.Parameters["AmbientColor"].SetValue(new Vector4(0.0f, 0.5f, 0f, 1.0f)); effect.Parameters["DiffuseColor"].SetValue(new Vector4(1.0f, 0.4f, 0.24f, 0.1f)); effect.Parameters["AmbientIntensity"].SetValue(0.5f); effect.Parameters["LightDirection"].SetValue(Vector4.Normalize(new Vector4(1, 0, 0.0f, 1.0f))); graphics.GraphicsDevice.SetVertexBuffer(part.VertexBuffer); graphics.GraphicsDevice.Indices = part.IndexBuffer; graphics.GraphicsDevice.DrawIndexedPrimitives(PrimitiveType.TriangleList, part.VertexOffset, 0, part.NumVertices, part.StartIndex, part.PrimitiveCount); } } }
代码中可以看出环境光AmbientColor:Vector4(0,0.5,0,1)这样做的目地是为了可以在下面的截图中看到意图,同时光线方向是平行于X轴指向X轴正半轴的。下面看看运行效果
其中橘黄色是反射光的颜色也即是DiffuseColor:Vector4(1.0f, 0.4f, 0.24f, 0.1f),而绿色是环境光的颜色AmbientColor:Vector4(0,0.5,0,1)光线方向是沿着X轴。可以看到未被照射到的地方色差明显。此时我们修改下LightDirection将其改成Vector4(0, 0, 1.0f, 1.0f) 沿着Z轴正方向(对比两张效果图即可明白光线方向的不同对最终效果的影响)
接下来我们看看fx里面的代码:
float4x4 World; float4x4 View; float4x4 Projection; float4 AmbientColor; float4 DiffuseColor; float AmbientIntensity; float4 LightDirection; // TODO: add effect parameters here. struct VertexShaderInput { float4 Position : POSITION0; }; struct VertexShaderOutput { float4 Position : POSITION0; float3 N:TEXCOORD0; }; VertexShaderOutput VertexShaderDiffuse(float4 Position:POSITION0,float3 N:NORMAL) { VertexShaderOutput output; float4 worldPosition = mul(Position, World); float4 viewPosition = mul(worldPosition, View); output.Position = mul(viewPosition, Projection); output.N=normalize(mul(N,World)); return output; } float4 PixelShaderDiffuse(VertexShaderOutput input) : COLOR0 { return AmbientColor*AmbientIntensity+1.0f*DiffuseColor*saturate(dot(LightDirection,input.N)); } technique DiffuseLight { pass Pass0 { // TODO: set renderstates here. VertexShader = compile vs_2_0 VertexShaderDiffuse(); PixelShader = compile ps_2_0 PixelShaderDiffuse(); } }
除了公式不同以外,在这里我们看到一个TEXCOORD0,TEXCOORDN是可以存放任何数据的寄存器,在相应位置没有存放纹理坐标的时候可以用来存储任何数据,并在以后读取出来。
漫反射光的知识也就这么多了,期待下节镜面反射光照的学习.......