openGL实现多光源光照效果

open那个C加加 2016-08-19 12:48:00
最近正在学openGL,用的是learnOpenGL,一个网址,学到Multiple lights多光源这一章的时候遇到了问题。我想实现的是一个平行光(无光源),4个点光源,一个聚光灯源对十个箱子的光照效果。但是当我在着色器源码中将聚光灯的光照效果注释掉以后,物体呈现了黑色,也就是说平行光以及点光源对物体的光照效果没有实现。我反复看了代码,不知道错在哪里。以下是我的物体的着色器源码:

#version 330 core
//物体结构体
struct Material
{
sampler2D texture0;
sampler2D texture1;
sampler2D texture2;
float shininess;
};

//聚光灯结构体
struct FlashLight
{
vec3 ambient; // 物体ambient颜色要素的因素因子
vec3 diffuse; // 光的颜色,一般设置为白色
vec3 specular; // 物体specular颜色要素的因素因子
vec3 position; // 点的位置
vec3 directional; // 光源的方向
float cutoff; // 内余弦
float outcutoff; // 外余弦
float constant; // 光强度衰减方程式常数项
float linear; // 光强度衰减方程式一次项
float quadratic; // 光强度衰减方程式二次项
};

//点光源结构体
struct PointLight
{
vec3 ambient; // 物体ambient颜色要素的因素因子
vec3 diffuse; // 光的颜色,一般设置为白色
vec3 specular; // 物体specular颜色要素的因素因子
vec3 position; // 点的位置
float constant; //光强度衰减方程式常数项
float linear; //光强度衰减方程式一次项
float quadratic; //光强度衰减方程式二次项
};

//平行光源结构体
struct DirectLight
{
vec3 ambient; // 物体ambient颜色要素的因素因子
vec3 diffuse; // 光的颜色,一般设置为白色
vec3 specular; // 物体specular颜色要素的因素因子
vec3 directional; //光的方向
};

in vec3 Normal;
in vec3 fragPos;
in vec2 TexCoord;

out vec4 color;

uniform Material material;
//有四个点光源
uniform PointLight pointlight[4];
uniform FlashLight flashlight;
uniform DirectLight directlight;
uniform vec3 cameraPos;

//声明各个光源的光照函数
vec3 calcFlashLight(FlashLight light,vec3 fragPos,vec3 normal,vec3 viewDir);
vec3 calcPointLight(PointLight light,vec3 fragPos,vec3 normal,vec3 viewDir);
vec3 calcDirectLight(DirectLight light,vec3 normal,vec3 viewDir);

void main()
{
vec3 normal=normalize(Normal);
vec3 viewDir=normalize(cameraPos-fragPos);

vec3 result =calcDirectLight(directlight,normal,viewDir);
for(int i=0;i<4;i++)
result +=calcPointLight(pointlight[i],fragPos,normal,viewDir);
result +=calcFlashLight(flashlight,fragPos,normal,viewDir);
color=vec4(result,1.0f);
}

//聚光灯
vec3 calcFlashLight(FlashLight light,vec3 fragPos,vec3 normal,vec3 viewDir)
{
float distance=length(light.position-fragPos);
float attenuation=1.0f/(light.constant+light.linear*distance+light.quadratic*(distance*distance));

//ambient
vec3 ambient=light.ambient*vec3(texture(material.texture0,TexCoord));

//diffuse
vec3 lightDir=normalize(light.position-fragPos);
float diff=max(dot(lightDir,normal),0.0);
vec3 diffuse=light.diffuse*diff*vec3(texture(material.texture0,TexCoord));

//specular
vec3 refletDir=reflect(-lightDir,normal);
float spec=pow(max(dot(viewDir,refletDir),0.0),material.shininess);
vec3 specular=light.specular*spec*vec3(texture(material.texture1,TexCoord));

//attenuation
ambient *=attenuation; //由于我们不希望在多光源下,ambient会累积增加,所以我们这边进行一个衰减
diffuse *=attenuation;
specular *=attenuation;

//cutoff
float theta=dot(normalize(-light.directional),lightDir);
float intensity=clamp((theta-light.outcutoff)/(light.cutoff-light.outcutoff),0.0,1.0); //我们不希望强度值在0与1之外
diffuse *=intensity;
specular *=intensity;

return (ambient+diffuse+specular);
}

//点光源
vec3 calcPointLight(PointLight light,vec3 fragPos,vec3 normal,vec3 viewDir)
{
float distance=length(light.position-fragPos);
float attenuation=1.0f/(light.constant+light.linear*distance+light.quadratic*(distance*distance));

//ambient
vec3 ambient=light.ambient*vec3(texture(material.texture0,TexCoord));

//diffuse
vec3 lightDir=normalize(light.position-fragPos);
float diff=max(dot(lightDir,normal),0.0);
vec3 diffuse=light.diffuse*diff*vec3(texture(material.texture0,TexCoord));

//specular
vec3 refletDir=reflect(-lightDir,normal);
float strength=pow(max(dot(viewDir,refletDir),0.0),material.shininess);
vec3 specular=light.specular*strength*vec3(texture(material.texture1,TexCoord));

//attenuation
ambient *=attenuation; //由于我们不希望在多光源下,ambient会累积增加,所以我们这边进行一个衰减
diffuse *=attenuation;
specular *=attenuation;

return (ambient+diffuse+specular);
}

//平行光
vec3 calcDirectLight(DirectLight light,vec3 normal,vec3 viewDir)
{
//ambient
vec3 ambient=light.ambient*vec3(texture(material.texture0,TexCoord));

//diffuse
vec3 directional=normalize(-light.directional);
float diff=max(dot(directional,normal),0.0);
vec3 diffuse=light.diffuse*diff*vec3(texture(material.texture0,TexCoord));

//specular
vec3 refletDir=reflect(-directional,normal);
float spec=pow(max(dot(viewDir,refletDir),0.0),material.shininess);
vec3 specular=light.specular*spec*vec3(texture(material.texture1,TexCoord));

return (ambient+diffuse+specular);
}



我第一次发贴.有什么规矩请指教。 VS中的代码我没贴,不知道贴哪段,谢谢过来参观以及帮我解答的人。
...全文
621 2 打赏 收藏 转发到动态 举报
写回复
用AI写文章
2 条回复
切换为时间正序
请发表友善的回复…
发表回复
open那个C加加 2016-08-21
  • 打赏
  • 举报
回复
谢谢老师推荐。
赵4老师 2016-08-19
  • 打赏
  • 举报
回复
搜网络教程“学OpenGL编3D游戏”。

33,311

社区成员

发帖
与我相关
我的任务
社区描述
C/C++ 新手乐园
社区管理员
  • 新手乐园社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧