Directx教程(28) 简单的光照模型(7)

       现实生活中的点光源都是随着距离衰减的,比如一个电灯泡在近处会照的很亮,远处光线就很弱。本节中我们在前面光公式的基础上,再给漫反射和高光加上一个衰减因子。

      光源随着距离衰减并不是纯线性的,常用的公式是:

  • d 是光源到着色点的距离。
  • kC, kL, 和 kQ 分别是常量、线性以及二次衰减系数。

现在在light.ps中,计算光照的代码变成了:

  for ( i = 0; i < NUM_LIGHTS; i++)
  {
    //自发射颜色
    emissive = Ke[i];
   
   //计算环境光
    ambient = Ka[i] * globalAmbient[i];
   
    //计算漫反射光
    //用LightDirection就是纯平行光
    //光源位置减顶点位置
     L = normalize(lightPosition[i].xyz - P);
     d = distance(lightPosition[i].xyz, P);
     //衰减系数
     atte = 1 / (attenuation[i].x + attenuation[i].y * d +attenuation[i].z * d * d);
     diffuseLight = max(dot(N, L), 0);
     diffuse =   Kd[i] * lightColor[i] * diffuseLight * atte ;

     //计算高光
     V = normalize(cameraPosition.xyz - P);
     H = normalize(L + V);
     specularLight = pow(max(dot(N, H), 0), shininess[i]);
 
      if (diffuseLight <= 0)
          specularLight = 0;
      specular = Ks[i] * lightColor[i] * specularLight *  atte;

      finalcolor +=  emissive + ambient + diffuse + specular;
     }

   

     相应的,在lightShaderClass.h中的struct  LightMaterialBufferType,也要做一些变化,增加一个D3DXVECTOR4分量attenuation,它的x,y,z分别表示常量、线性以及二次衰减系数。之所以用D3DVECTOR4,是因为const buffer要求是4的倍数,我曾尝试用3个float,结果程序竟然有编译错误。

D3DXVECTOR4 attenuation[NUM_LIGHTS]; //衰减系数,x,y,z对应常量,线性和二次系数

    在light.ps中,const buffer  LightMaterialBuffer,也要增加衰减因子,它和LightMaterialBufferType中的attenuation是相对应的。

float4 attenuation[NUM_LIGHTS]; //衰减系数

 

程序执行后的效果如下:

完整的代码请参考:

工程文件myTutorialD3D11_21

代码下载:

http://files.cnblogs.com/mikewolf2002/myTutorialD3D11.zip

    接下来我们再实现一个探照灯(spotlight)的效果,如下图所示,只有在圆锥内角(inner)的范围才是光照能够达到的范围。但是只考虑内角的话,我们的光照会比较生硬,内角圆锥内,有光,内角圆锥外,一片漆黑,所以我们又加了一个外角(outer),对于内角和外角之间的空间,我们使用hlsl的差值函数smoothstep来计算得到一个0-1之间的数值。

 

 

 

light.ps的主要代码:

    下面的函数通过smoothstep计算出spotlight的因子。其中cosInnerCone是内角的余弦值,cosOuterCone是外角的余弦值。如果计算出的cosDirection值大于内角余弦值,则smoothstep值为1,如果cosDirection值小于外角余弦值,则其值为0,对于在这两者之间的值,smoothstep会用多项式差值得到一个0-1之间的值。

//一个计算spot light系数的函数
float dualConeSpotlight(float3 P, float3 lightpos, float3 lightdir, float  cosInnerCone, float cosOuterCone)
{
 
  float3 V = normalize(P - lightpos);
 
  float cosDirection = dot(V, normalize(lightdir));
 
  return smoothstep(cosOuterCone,  cosInnerCone, cosDirection);
}

  for ( i = 0; i < NUM_LIGHTS; i++)
  {
       //自发射颜色
    emissive = Ke[i];
   
    //计算环境光
    ambient = Ka[i] * globalAmbient[i];
   
    //计算漫反射光
    //用LightDirection就是纯平行光,在spotlight情况下代表光的方向

    spotEffect = dualConeSpotlight(P,  lightPosition[i].xyz, lightDirection[i],spotattenuation[i].x, spotattenuation[i].y);

    //光源位置减顶点位置
     L = normalize(lightPosition[i].xyz - P);
     d = distance(lightPosition[i].xyz, P);
     //衰减系数
     atte = 1 / (attenuation[i].x + attenuation[i].y * d +attenuation[i].z * d * d);
     diffuseLight = max(dot(N, L), 0);
     diffuse =   Kd[i] * lightColor[i] * diffuseLight * atte * spotEffect;

     //计算高光
     V = normalize(cameraPosition.xyz - P);
     H = normalize(L + V);
     specularLight = pow(max(dot(N, H), 0), shininess[i]);
 
      if (diffuseLight <= 0)
          specularLight = 0;
      specular = Ks[i] * lightColor[i] * specularLight * spotEffect;

      finalcolor +=  emissive + ambient + diffuse + specular;
     }

     同样的,我们的const buffer LightMaterialBuffer中,增加了

         float4 spotattenuation[NUM_LIGHTS];

     它的x,y分别表示内角和外角余弦值。

     lightShaderClass.h中的材质光照结构中也增加了

        D3DXVECTOR4 spotattenuation[NUM_LIGHTS]; //对于spotlight,x,y分别存储内和外角cos值

程序最终执行效果如下:

 

完整的代码请参考:

工程文件myTutorialD3D11_22

代码下载:

http://files.cnblogs.com/mikewolf2002/myTutorialD3D11.zip

时间: 2024-10-14 17:53:54

Directx教程(28) 简单的光照模型(7)的相关文章

Directx教程(22) 简单的光照模型(1)

     在前面的教程中,我们在顶点属性中直接给顶点赋颜色,这样生成的三维物体缺乏真实感,如下图中两个立方体,左边的是通过光照生成物体表面颜色的,右边的则是直接给顶点赋颜色值.      首先,我们学习一下最简单的phong光照模型:       在phong光照模型中,物体表面的颜色由自发射光(emissive).环境光(ambient).漫反射光(diffuse)以及镜面高光(specular)四部分组成,每一部分又是通过物体表面的材质属性和光照属性一起来决定.用公式来表示就是:     

Directx教程(26) 简单的光照模型(5)

    在前面的工程中,我们都是在vs中实现顶点光照计算,然后再把顶点颜色传到ps中.本章中我们尝试fragment光照(或者说叫ps光照),在vs中,我们把顶点在世界坐标系中的法向和位置都直接传输到ps中.     // 世界坐标系中的顶点法向.     float3 N = mul(input.normal, (float3x3)worldMatrix);     output.worldnormal= N;         //世界坐标系顶点位置     worldPosition = m

Directx教程(29) 简单的光照模型(8)

     现在我们新建一个工程myTutorialD3D_23,在这个工程中,对前面一章的代码进行一些整理: 1.我们在顶点属性中增加材质的的漫反射系数和高光系数,前面我们放在一个光照材质结构中,这样我们能够比较灵活的定义不同顶点的材质属性,当然这也增加了顶点缓冲的大小. struct VertexType { D3DXVECTOR3 position; D3DXVECTOR3 normal; //法向 D3DXVECTOR4 Kd; //材质漫反射系数 D3DXVECTOR4 Ks; //材质

Directx教程(27) 简单的光照模型(6)

     从myTutorialD3D11_15到myTutorialD3D11_19的工程中,我们都只有一个光源,光源的位置在LightClass中我设置为m_position = D3DXVECTOR4(5.0, 5.0, -3.0,1.0),所以我们渲染的cube,在前面,右侧,上面都没有问题,但是,我们通过A键移动摄像机后,会发现右侧的面没有光照效果.这是因为右侧面的法向是-x方向,漫反射光和高光都为0,所以效果很差. 现在我们将在场景中增加第二个光源,位置放在SetLightPosit

Directx教程(25) 简单的光照模型(4)

     在本篇日志中,我们尝试用不带衰减的点光源来计算漫反射颜色.     前面的三个工程,我们都用的是方向光源(directional light),它的特点是没有光源位置或者说光源位置位于无穷远处,且光线在各个方向都是平行的,所以在工程myTutorialD3D11_17中,我们看到的程序界面上,cube每个面上的颜色都是一样的[因为diffuse光占颜色的大部分].     方向光原理可用以下图表示:      现在我们尝试点光源,点光源原理如下:     因为计算漫反射时候,每个顶点的

Directx教程(24) 简单的光照模型(3)

     在工程myTutorialD3D11_17中,我们重新定义我们的cube顶点法向,每个三角形面的顶点法向都是和这个三角形的面法向是一致的.如下图所示:     在该工程中,我们还修改了CubeModelClass文件,从一个cube.txt文件中读cube顶点位置.法向.纹理坐标.     cube.txt的格式为: Vertex Count: 36 Data: -1.0  1.0 -1.0 0.0 0.0  0.0  0.0 -1.0      数据行前三个数据为顶点位置,接着两个数

Directx教程(23) 简单的光照模型(2)

   在工程myTutorialD3D11_16中,我在文件light.vs中定义了一个材质光源属性常量缓冲. //const buffer最好为4 float的倍数,否则创建const buffer会fail cbuffer LightMaterialBuffer {     float4 lightPosition; //光源位置     float4 lightColor;  //光源颜色     float4 globalAmbient; //光源的环境光反射系数     float4

Directx11教程(29) 简单的光照模型(8)

     现在我们新建一个工程myTutorialD3D_23,在这个工程中,对前面一章的代码进行一些整理: 1.我们在顶点属性中增加材质的的漫反射系数和高光系数,前面我们放在一个光照材质结构中,这样我们能够比较灵活的定义不同顶点的材质属性,当然这也增加了顶点缓冲的大小. struct VertexType     {     D3DXVECTOR3 position;     D3DXVECTOR3 normal; //法向     D3DXVECTOR4 Kd;  //材质漫反射系数    

CSS3实例教程代码:简单制作遨游浏览器LOGO

文章简介:CSS3实例教程代码:简单制作遨游浏览器LOGO. <!DOCTYPE HTML> <html> <head> <title>LOGO</title> <style type="text/css"> * {margin:0; padding:0} .main {width:800px; margin:50px auto 0}  .main_wrapper {width:240px; height:240p