QT+OpenGL高级光照1

本篇完整工程见gitee:QtOpenGL 对应点的tag,由turbolove提供技术支持,您可以关注博主或者私信博主

Blinn-Phong

  • 冯氏光照:视线与反射方向之间的夹角不小于90度,镜面光分量会变成0.0(不是很合理,会有清晰的分界线)

  • Blinn-Phone模型采用了半程向量,即光线与视线夹角一般方向上的一个单位向量。当半程向量与法线向量越接近,镜面光分量就越大。

半程向量公式如下:
H ⃗ = L ⃗ + V ⃗ ∣ ∣ L ⃗ + V ⃗ ∣ ∣ \vec{H} = \frac {\vec{L} + \vec{V}}{||\vec{L} + \vec{V}||} H =∣∣L +V ∣∣L +V ​

vec3 lightDir = normalize(lightPos - FragPos);
vec3 viewDir = normalize(viewPos - FragPos);
vec3 halfWayDir = normalize(lightDir + viewDir);
......
float spec = pow(max(dot(normal halfWayDir), 0.0), shininess)

效果图:冯氏光照

Gammar校正

  • 阴极射线管显示器(CRT):输入电压翻倍与亮度提高的关系是与Gammar次方相关

    • 设备输入亮度 = 电压的Gamma次幂
  • 这正好与人眼的感知是类似(相反)的

对于CRT,Gamma通常为2.2

人眼对黑夜的环境更加敏感(把更多的精度分配给了低灰度)

点线:线性的理想情况, Gamma为1

实线:CRT的实际情况,Gamma为2.2

(0.5,0.0,0.0)*2 = (1.0, 0.0,0.0)

对于CRT来说,实际上亮度提高了: 1 / 0.218 约等于 4.587

所以到目前位置,我们之前配置的颜色和光照变量从物理角度来看都是不正确的

  • 由于颜色是基于显示器的输出进行配置的,因此线性空间中的所有中间值计算在物理上都是不正确的。随着更先进的照明算法的加入,这一点变得更加明显

有两种方法可以将gamma校正应用于场景:

  • 通过使用OpenGL内置的sRGB帧缓冲支持(glEnable(GL_FRAMEBUFFER_SRGB))

  • 通过自己在片段着色器中执行gamma校正

float gamma = 2.2;
FragColor.rgb = pow(fragColor.rgb, vec3(1.0/gamma));

sRGB(standard Red GreenBlue)

微软联合HP、三菱、爱普生等厂商联合开发的通用色彩标准

  • Gamma值为2.2的颜色空间称为sRGB颜色空间(不是100%精确,但是接近)。每个监视器都有自己的Gamma曲线, Gamma值为2.2时候在大多数监视器上显示良好。游戏通常允许玩家更改游戏的Gamma值,因为每个显示器的Gamma设置略有不同。
  • 由于监视器显示应用了Gamma的颜色,因此创建或编辑的所有图片都不是在线性空间中,而是在sRGB空间中。

纹理在sRGB空间创建和展示,在sRGB空间中使用,不必关心gamma校正纹理,显示也没问题。然而,如果把所有的东西都放在线性空间中展示,纹理颜色就会出现问题。

实际上进行了两次Gamma校正!

  • 基于显示器看到的情况创建一个图像,我们就已经对颜色进行了Gamma校正
  • 代码又进行了一次Gamma校正

sRGB代码实现

为了解决重复校正的问题,把这些sRGB纹理在在进行任何颜色值的计算之前变回线性空间:

float gamma = 2.2;
vec3 diffuseColor = pow(texture(diffuse, texCoords).rgb, vec3(gamma));

或者通过OpenGL,自动把颜色校正到线性空间中

glTexImage2D(GL_TEXTURE2D, 0, GL_SRGB, width, height, 0, GL_RGB, GL_UNSIGNED_BYTE, data);

如果在纹理中引入了alpha元素,必须将纹理的内部格式指定为GL_SRGB_ALPHA

不是所有的纹理都是在sRGB空间中,当把纹理转换成sRGB时要格外小心:

  • 比如diffuse纹理,这种为物体上色的纹理几乎都是在sRGB空间中的
  • 像specular贴图和法线贴图几乎都是在线性空间中
QImage image(":/resources/wall.jpg");
specular_texture_ = new QOpenGLTexture(QOpenGLTexture::Target2D);
glBindTexture(GL_TEXTURE_2D, specular_texture_->textureId());
glTexImage2D(GL_TEXTURE_2D, GL_SRGB, image.width(), image.height(), 0, GL_RGB, GL_UNSIGNED_BYTE, image.bits());
glGenerateMipmap(GL_TEXTURE_2D);

衰减

在真正的物理世界,光线的衰减与距离的平方成反比:

float attenuation = 1.0 / (distance * distance);

但是当距离小的时候,上面的公式效果会很不对劲,用下面的公式会更好:

float attenuation = 1.0 / distance;

所以以前一直使用下面的公式:
F a t t = 1.0 K c + K l ∗ d + K q ∗ d 2 F_{att} = \frac{1.0}{K_c + K_l * d + K_q *d^2} Fatt​=Kc​+Kl​∗d+Kq​∗d21.0​

#version 330 corestruct Material {sampler2D diffuse;sampler2D specular;float shininess;
};
uniform Material material;struct Light {vec3 pos;vec3 ambient;vec3 diffuse;vec3 specular;
};
uniform Light light;uniform bool gamma;out vec4 FragColor;in vec2 TexCoords;
in vec3 Normal;
in vec3 FragPos;uniform vec3 viewPos;
vec3 BlinnPhong(vec3 normal, vec3 fragPos, vec3 lightPos, vec3 lightColor) {// diffusevec3 lightDir = normalize(lightPos - fragPos);float diff = max(dot(lightDir, normal), 0.0);vec3 diffuse = diff * lightColor;// specularvec3 viewDir = normalize(viewPos - fragPos);vec3 reflectDir = reflect(-lightDir, normal);float spec = 0.0;vec3 halfwayDir = normalize(lightDir + viewDir);spec = pow(max(dot(normal, halfwayDir), 0.0), 64.0);vec3 specular = spec * lightColor;// simple attenuationfloat max_distance = 1.5;float distance = length(lightPos - fragPos);float attenuation = 1.0 / (gamma ? distance * distance : distance);diffuse *= attenuation;specular *= attenuation;return diffuse + specular;
}void main() {vec3 diffuseTexColor=vec3(texture(material.diffuse,TexCoords));vec3 ambient=light.ambient;//........//if(gamma) diffuseTexColor = pow(diffuseTexColor, vec3(1.0/2.2));vec3 norm = normalize(Normal);vec3 result ;for(int i = -2; i < 2; ++i){vec3 lightColor=(2-i)*vec3(0.25);result+= BlinnPhong(norm,FragPos,light.pos+i*vec3(2,0.0,0.0),lightColor);}if(gamma)ambient=pow(ambient, vec3(2.2));result+=ambient;result*=diffuseTexColor*result;if(gamma) result = pow(result, vec3(1.0/2.2));if(gl_FrontFacing==false)FragColor = vec4(result, 1.0);
}

QT+OpenGL高级光照 Blinn-Phong和Gamma校正相关推荐

  1. OpenGL 高级光照Advanced Lighting

    OpenGL高级光照Advanced Lighting 高级光照Advanced Lighting简介 Blinn-Phong 高级光照Advanced Lighting简介 在光照小节中,我们简单地 ...

  2. QT+OpenGL高级数据和高级GLSL

    QT+OpenGL高级数据和高级GLSL 本篇完整工程见gitee:QtOpenGL 对应点的tag,由turbolove提供技术支持,您可以关注博主或者私信博主 高级数据 OpenGL中的缓冲区 对 ...

  3. opengl高级光照之gamma校正

    官方文章 gamma校正 gamma校正概念 一个渐变的效果 通过以下网站调整Gamma值可以观察到效果 色彩管理网 gamma校正 Gamma校正(Gamma Correction)的思路是在最终的 ...

  4. QT openGL环境光照

    先看上一篇文章:Qt Opengl 给两个立方体设置两中不同颜色 使用环境光非常的简单,我们只需要设置一个环境光强度,用这个强度值乘上光源的颜色得到环境光颜色.最后,用环境光颜色乘上物体的颜色,得到物 ...

  5. opengl高级光照之延迟渲染以及光体积

    延迟着色法 延迟着色法官方文档 我们现在一直使用的光照方式叫做正向渲染(Forward Rendering)或者正向着色法(Forward Shading),它是我们渲染物体的一种非常直接的方式,在场 ...

  6. opengl高级光照之法线贴图

    法线贴图 opengl官方文档 核心修改的就是片段着色器中的normal值 uniform sampler2D normalMap; void main() { // 从法线贴图范围[0,1]获取法线 ...

  7. opengl高级光照之视差贴图(陡峭视差贴图以及视差遮蔽映射)

    视差贴图 视差贴图官方文档 视差贴图 视差贴图(Parallax Mapping)技术和法线贴图差不多,但它有着不同的原则.和法线贴图一样视差贴图能够极大提升表面细节,使之具有深度感.它也是利用了视错 ...

  8. QT+OpenGL光照

    QT+OpenGL光照 本篇完整工程见gitee:QtOpenGL 对应点的tag,由turbolove提供技术支持,您可以关注博主或者私信博主 颜色 现实生活中看到的物体的颜色并不是这个物体真正拥有 ...

  9. OpenGL基础45:光照矫正(下)之Gamma校正

    接上文:OpenGL基础44:光照矫正(上) 四.Gamma矫正 4.1.人的视觉特性 和很多错视图一样,对于下面这张灰阶图,如果1表示纯白,0表示纯黑,那么这张图片的哪个位置代表的是0.5,也就是自 ...

最新文章

  1. 报名 | 大数据打造智慧的政府门户讲座
  2. Java练习:用IF()进行数字排序
  3. Android自定义spinner下拉框实现的实现
  4. 基于TextRank的关键词提取算法
  5. Luogu P4709 信息传递 (群论、生成函数、多项式指数函数)
  6. Python FastApi:post文件与数据/本地端测试
  7. java图片上传(mvc)
  8. php获得昨天零时的时间戳,php 获取时间今天明天昨天时间戳
  9. Jenkins的安装和卸载(转载)
  10. 在VC++中使用Tab Control控件
  11. link标签中的integrity和crossorigin字段---web前端工作笔记015
  12. 8.11 NOIP模拟测试17 入阵曲+将军令+星空
  13. 除非万不得已,否则绝不谈判
  14. jquery 图像滑块_10个很棒的jQuery图像滑块插件
  15. BZOJ2592: [Usaco2012 Feb]Symmetry
  16. 慕课软件质量保证与测试(第八章.软件评审概述)
  17. win10输入法变成繁体字的快捷键
  18. 手机电脑浏览器抓取京东Cookies教程
  19. CodeForces - 3B Lorry【贪心】
  20. Facebook攻略--注册流程

热门文章

  1. InfiniBand, RDMA, iWARP, RoCE , CNA, FCoE, TOE, RDMA, iWARP, iSCSI等概念
  2. windows下修改hosts文件不需要重启也能生效。
  3. 在word使用中,如何将word中的单独一页设置成横向或纵向
  4. 大数据分析案例-基于决策树算法构建信用卡违约预测模型
  5. 周阳老师JUC并发编程
  6. Mysql入门【Mysql基础】
  7. HHU 《算法与数据结构》复习指南
  8. word2vec三种保存模型方式
  9. MaterialDesigner(三)
  10. 分享卡通儿童中小学班干竞选PPT模板