// Copyright 2018 Google Inc. precision mediump float; uniform vec4 u_ambient_light_color; uniform vec4 u_SceneLight_0_color; uniform vec4 u_SceneLight_1_color; uniform float u_Shininess; uniform float u_RimIntensity; uniform float u_RimPower; uniform vec4 u_Color; varying vec4 v_color; varying vec3 v_normal; varying vec3 v_position; varying vec3 v_light_dir_0; varying vec3 v_light_dir_1; varying vec2 v_texcoord0; const float PI = 3.141592654; const float INV_PI = 0.318309886; const vec3 GAMMA_DIELECTRIC_SPEC = vec3(0.220916301, 0.220916301, 0.220916301); const float GAMMA_ONE_MINUS_DIELECTRIC = (1.0 - 0.220916301); float Pow5(float x) { return x * x * x * x * x; } float DisneyDiffuseTerm(float NdotV, float NdotL, float LdotH, float perceptualRoughness) { float fd90 = 0.5 + 2.0 * LdotH * LdotH * perceptualRoughness; float lightScatter = 1.0 + (fd90 - 1.0) * Pow5(1.0 - NdotL); float viewScatter = 1.0 + (fd90 - 1.0) * Pow5(1.0 - NdotV); return lightScatter * viewScatter; } float SmithJointVisibilityTerm(float NdotL, float NdotV, float roughness) { float lambdaV = NdotL * mix(NdotV, 1.0, roughness); float lambdaL = NdotV * mix(NdotL, 1.0, roughness); return 0.5 / (lambdaV + lambdaL + 1e-5); } float GgxDistributionTerm(float NdotH, float roughness) { float a2 = roughness * roughness; float d = (NdotH * a2 - NdotH) * NdotH + 1.0; return INV_PI * a2 / (d * d + 1e-7); } vec3 FresnelTerm (vec3 F0, float cosA) { float t = Pow5(1.0 - cosA); return F0 + (vec3(1.0) - F0) * t; } vec3 SurfaceShaderInternal( vec3 normal, vec3 lightDir, vec3 eyeDir, vec3 lightColor, vec3 diffuseColor, vec3 specularColor, float perceptualRoughness) { float NdotL = clamp(dot(normal, lightDir), 0.0, 1.0); float NdotV = abs(dot(normal, eyeDir)); vec3 halfVector = normalize(lightDir + eyeDir); float NdotH = clamp(dot(normal, halfVector), 0.0, 1.0); float LdotH = clamp(dot(lightDir, halfVector), 0.0, 1.0); float diffuseTerm = NdotL * DisneyDiffuseTerm(NdotV, NdotL, LdotH, perceptualRoughness); if (length(specularColor) < 1e-5) { return diffuseColor * (lightColor * diffuseTerm); } float roughness = perceptualRoughness * perceptualRoughness; float V = GgxDistributionTerm(NdotH, roughness); float D = SmithJointVisibilityTerm(NdotL, NdotV, roughness); float specularTerm = V * D * PI; specularTerm = sqrt(max(1e-4, specularTerm)); specularTerm *= NdotL; vec3 fresnelColor = FresnelTerm(specularColor, LdotH); return lightColor * (diffuseTerm * diffuseColor + specularTerm * fresnelColor); } vec3 SurfaceShaderSpecularGloss( vec3 normal, vec3 lightDir, vec3 eyeDir, vec3 lightColor, vec3 albedoColor, vec3 specularColor, float gloss) { float oneMinusSpecularIntensity = 1.0 - clamp(max(max(specularColor.r, specularColor.g), specularColor.b), 0., 1.); vec3 diffuseColor = albedoColor * oneMinusSpecularIntensity; float perceptualRoughness = 1.0 - gloss; return SurfaceShaderInternal( normal, lightDir, eyeDir, lightColor, diffuseColor, specularColor, perceptualRoughness); } vec3 SurfaceShaderMetallicRoughness( vec3 normal, vec3 lightDir, vec3 eyeDir, vec3 lightColor, vec3 albedoColor, float metallic, float perceptualRoughness) { vec3 specularColor = mix(GAMMA_DIELECTRIC_SPEC, albedoColor, metallic); float oneMinusReflectivity = GAMMA_ONE_MINUS_DIELECTRIC - metallic * GAMMA_ONE_MINUS_DIELECTRIC; vec3 diffuseColor = albedoColor * oneMinusReflectivity; return SurfaceShaderInternal( normal, lightDir, eyeDir, lightColor, diffuseColor, specularColor, perceptualRoughness); } vec3 ShShaderWithSpec( vec3 normal, vec3 lightDir, vec3 lightColor, vec3 diffuseColor, vec3 specularColor) { float specularGrayscale = dot(specularColor, vec3(0.3, 0.59, 0.11)); float NdotL = clamp(dot(normal, lightDir), 0.0, 1.0); float shIntensityMultiplier = 1. - specularGrayscale; shIntensityMultiplier *= shIntensityMultiplier; return diffuseColor * lightColor * NdotL * shIntensityMultiplier; } vec3 ShShader( vec3 normal, vec3 lightDir, vec3 lightColor, vec3 diffuseColor) { return ShShaderWithSpec(normal, lightDir, lightColor, diffuseColor, vec3(0.,0.,0.)); } vec3 LambertShader( vec3 normal, vec3 lightDir, vec3 lightColor, vec3 diffuseColor) { float NdotL = clamp(dot(normal, lightDir), 0.0, 1.0); return diffuseColor * lightColor * NdotL; } vec3 computeGlassReflection() { vec3 normal = normalize(v_normal); float backfaceDimming = 1.; if (!gl_FrontFacing) { normal *= -1.0; backfaceDimming = .25; } vec3 lightDir0 = normalize(v_light_dir_0); vec3 lightDir1 = normalize(v_light_dir_1); vec3 eyeDir = -normalize(v_position); vec3 diffuseColor = vec3(0.,0.,0.); vec3 specularColor = vec3(u_Color.r, u_Color.g, u_Color.b); vec3 lightOut0 = SurfaceShaderSpecularGloss(normal, lightDir0, eyeDir, u_SceneLight_0_color.rgb, diffuseColor, specularColor, u_Shininess); float viewAngle = clamp(dot(eyeDir, normal),0.,1.); float rim = pow(1. - viewAngle, u_RimPower) * u_RimIntensity; vec3 rimColor = vec3(rim,rim,rim); return (lightOut0 + rimColor) * backfaceDimming; } void main() { gl_FragColor.rgb = computeGlassReflection(); gl_FragColor.a = 1.0; }