c++ - Blinn-Phong 照明 - 全色的最佳计算是什么

标签 c++ opengl glsl phong

我读过的几乎所有文章和书籍都指出最终颜色的成分是: FinalColor =ambientColor+lambertianTerm*diffuseColor(= Material 颜色)+specularIntensity*specularColor(=光颜色)

lambertianTerm = dot( surfaceNormal, normalize( lightPos - surfacePos ) );

halfVector = normalize( normalize( lightPos - surfacePos ) + normalize( eyePos - surfacePos );
specularFactor = max( dot( halfVector, surfaceNormal ), 0.0f );

specularIntensity = pow( specularFactor, surfaceShininessLevel );

以下是一些使用上述方法计算最终颜色的地方:OpenGL SuperBible 6th Edition -> Rendering Techniques -> Lighting Models -> Blinn-Phong Lighting, Wikipedia -> Blinn–Phong shading model (see the fragment shader)等。

最终颜色的计算存在问题:灯光的颜色(还要考虑多个光源的情况)不参与漫反射颜色的制定。考虑这样的情况:光泽因子是一个很大的数字 - 比如说 128.0,这样镜面反射光斑就很小,并且对象的大部分区域都使用漫反射项进行着色。还让物体的颜色为绿色,灯光的颜色为红色。如果没有环境颜色,上面等式的输出是一个部分照亮的纯绿色物体,上面有一个小黄点 = 绿色 + 红色 = 从绿色物体反射的红色光源的光: purely green diffuse color + yellow specular highlight 这是不正确的。当然,绿色+红色是黄色,但可以肯定的是,您既不会看到纯绿色的球,也不会看到黄色的镜面反射点。手里拿着一个绿色 Shiny 的球 - 例如台球比赛中的大 6 球,然后用红灯点亮它 - 我可以向你保证,你不会只看到绿色的漫反射和黄色的镜面反射。相反,您会看到混合的绿色 + 红色,用于漫反射和更红色的镜面反射点。迄今为止我发现的计算最终颜色的最佳方法 - 平均混合:

finalColor = ambient
             + lambertianTerm * ( surfaceColor + lightColor ) / 2.0 
             + specularIntensity * lightColor;

diffuse = yellow with a red accent, specular spot = almost purely red 尝试过叠加混合:

const vec4 srgbLuminanceFactor = vec4( 0.2125f, 0.7154f, 0.0721f, 1.0f );
vec4 overlay( vec4 baseColor, vec4 blendColor )
{
    float luminance = dot( baseColor, srgbLuminanceFactor );
    vec4 lowerLumOverlay = 2.0f * blendColor * baseColor;
    if ( luminance < 0.45f )
    {
        return lowerLumOverlay;
    }

    const vec4 whiteColor = vec4( 1.0 );
    vec4 higherLumOverlay = whiteColor - 2.0f * ( whiteColor - blendColor ) * ( whiteColor - baseColor );
    return luminance > 0.55f
           ? higherLumOverlay
           : mix( lowerLumOverlay, higherLumOverlay, ( luminance - 0.45f ) * 10.0f );
}

...但看起来不太好。 也许光线和物体的颜色应该以另一个线性比例混合:

mix( surfaceColor, lightColor, ratio );

...或者以我无法想到的非线性方式。

那么最后一个问题:完整颜色的最佳计算是什么? 另外,如果我遗漏了什么,请告诉我,但在我看来,纯绿色图片+纯黄色镜面高光完全不是现实世界的场景。

最佳答案

表面颜色的含义是“光的颜色可以从该表面扩散”。

镜面反射颜色的含义是“可以从该表面反射什么光的颜色”。

光色的含义是“到达表面的光的颜色”。

所以你必须将它们相乘:

   finalColor = ambient
             + lambertianTerm * surfaceColor * lightColor
             + specularIntensity * specularColor * lightColor;

关于c++ - Blinn-Phong 照明 - 全色的最佳计算是什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33600223/

相关文章:

c++ - 帧缓冲区深度伪影

opengl - 如何在OpenGL中归一化纹理空间的图像坐标?

c++ - 如何在 linux 上用 C/C++ 编写 Web 服务器

c++ - 强制机器使用专用显卡?

math - 为什么我不能平移我的飞机向下?

opengl - 我应该在统一缓冲区或着色器存储缓冲区对象内使用 `vec3` 吗?

opengl - 如何在 OpenGL 3.2 中使用 GL_TEXTURE_2D_ARRAY

c++ - 如何解释 "void(*)()"?

c++ - 在 C++ 中发现 USB 设备并与之通信

C++ 无法获取进程 ID (windows)