openclonk/planet/Graphics.ocg/CommonShader.glsl

163 lines
4.8 KiB
GLSL

// uncomment the following lines for debugging light directions:
// yellow: light up, blue: light down, turqoise: light right, pink: light left
// brightness: light strength
//#define LIGHT_DEBUG
// uncomment the following lines for debugging light color:
// the light will always come from the front and have a uniform brightness.
//#define LIGHT_DEBUG_COLOR
// uncomment the following lines to set the light color to pink for all lights for debugging:
//#define LIGHT_DEBUG_PINK
// At what point of light intensity we set the "darkness" point. This
// is to compensate for the fact that the engine "smoothes" the light
// and therefore will often never arrive at 0 light intensity.
const float lightDarknessLevel = 8.0 / 256.0;
#ifdef HAVE_LIGHT
uniform sampler2D ambientTex;
uniform mat3x2 ambientTransform;
uniform float ambientBrightness;
uniform sampler2D lightTex;
#endif
// Gamma uniforms
uniform vec3 gamma;
// 0 if backface culling is enabled, 1 if it is disabled
uniform float cullMode;
// Light dot product, taking cull mode into account
float lightDot(vec3 normal, vec3 lightDir) {
return abs(max(-cullMode, dot(normalize(normal), lightDir)));
}
vec3 extend_normal(vec2 v)
{
// the higher the second value, the further away the light source from the landscape
return normalize(vec3(v, 0.45));
}
// Converts the pixel range 0.0..1.0 into the integer range 0..255
int f2i(float x) {
return int(x * 255.9);
}
slice(texture+5)
{
#ifdef HAVE_LIGHT
// Query light texture
vec2 lightDirCoord = lightCoord.st;
vec4 lightPx = texture2D(lightTex, lightDirCoord);
float lightBright = 2.0*max(0.0, lightPx.a-lightDarknessLevel);
vec3 lightDir = extend_normal(vec2(1.0, 1.0) - lightPx.yz * 3.0);
// Query light color texture (part of the light texture)
vec2 lightColorCoord = lightCoord.st - vec2(0.0, 0.5); // subtract offset for the color texture
vec3 lightColor = texture2D(lightTex, lightColorCoord.st).rgb;
// Normalise light colour
#ifdef LIGHT_DEBUG_COLOR
lightBright = 0.5;
lightColor = vec4(1.0, 0.0, 1.0, 1.0);
#endif
#else
// When lighting is disabled, put a light source coming from the camera.
// Note that in most cases this does not actually matter, since in the
// case with lighting disabled, ambient lighting takes fully over.
float lightBright = 0.5;
vec3 lightDir = vec3(0.0, 0.0, 1.0);
vec3 lightColor = vec3(1.0, 1.0, 1.0);
#endif
#ifdef HAVE_LIGHT
// Ambient light
// Extra .xy since some old intel drivers return a vec3
float ambient = texture2D(ambientTex, (ambientTransform * vec3(gl_FragCoord.xy, 1.0)).xy).r * ambientBrightness;
#else
// Lighting disabled: Ambient light everywhere
float ambient = 1.0;
#endif
}
slice(light)
{
// Light dot product, taking backface culling into account
normal = normalize(normal);
float light = lightDot(normal, lightDir);
// Amount of reflection, depending on the angle where material reflects most
light = min(light / matAngle, 2.0 - light / matAngle);
#ifdef LANDSCAPE
normal2 = normalize(normal2);
float light2 = lightDot(normal2, lightDir);
light2 = min(light2 / matAngle2, 2.0 - light2 / matAngle2);
#endif
}
slice(light+1)
{
#ifdef LANDSCAPE
// For landscape, ambient brightness is coming from top
vec3 ambientDir = normalize(vec3(0.0, -1.0, 1.0));
#else
// For objects, ambient brightness is coming from the front
vec3 ambientDir = vec3(0.0, 0.0, 1.0);
#endif
// Add ambience to brightness
lightBright = mix(lightBright, 1.0, ambient);
light = mix(light, 0.5 * (ambient + lightDot(normal, ambientDir)), ambient);
#ifdef LANDSCAPE
light2 = mix(light2, 0.5 * (ambient + lightDot(normal2, ambientDir)), ambient);
#endif
lightColor = mix(lightColor, vec3(1.0,1.0,1.0), ambient);
}
slice(color+5)
{
// Normalize light colour
vec3 lightColorNorm = sqrt(3.0) * normalize(lightColor);
// Add light. Depending on material properties, we make it more
// "spotty" and allow the material to self-illuminate. The light
// brightness overrules everything though (= FoW is last factor).
vec3 spotLight = pow(vec3(light,light,light), matSpot);
color.rgb = lightBright * color.rgb * (matEmit + lightColorNorm * spotLight);
#ifdef LANDSCAPE
vec3 spotLight2 = pow(vec3(light2,light2,light2), matSpot2);
color2.rgb = lightBright * color2.rgb * (matEmit2 + lightColorNorm * spotLight2);
#endif
}
slice(finish+5)
{
#ifdef LIGHT_DEBUG
#ifdef HAVE_LIGHT
float lightYDir = lightPx.b - 1.0/3.0;
float lightXDir = lightPx.g - 1.0/3.0;
float lightStrength = lightPx.a;
color =
vec4(lightStrength * vec3(1.0-1.5*(max(0.0, lightYDir) + max(0.0,lightXDir)),
1.0-1.5*(max(0.0, lightYDir) + max(0.0,-lightXDir)),
1.0-1.5*max(0.0, -lightYDir)),
1.0);
#else
color = vec4(0.0, 0.0, 0.0, 0.0); // invisible
#endif
#endif
}
slice(finish+10) {
color = vec4(pow(color.rgb, gamma), color.a);
}