2014-11-30 20:11:22 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2014 Tim Anema
|
|
|
|
light shadow, shine and normal shader all in one
|
|
|
|
*/
|
2014-11-29 19:52:05 +00:00
|
|
|
#define PI 3.1415926535897932384626433832795
|
|
|
|
|
2014-11-30 20:11:22 +00:00
|
|
|
extern Image shadowMap; //a canvas containing shadow data only
|
2014-12-14 06:14:42 +00:00
|
|
|
extern vec3 lightPosition; //the light position on the screen(not global)
|
|
|
|
extern vec3 lightColor; //the rgb color of the light
|
2014-11-30 20:11:22 +00:00
|
|
|
extern float lightRange; //the range of the light
|
|
|
|
extern float lightSmooth; //smoothing of the lights attenuation
|
2014-12-14 06:14:42 +00:00
|
|
|
extern vec2 lightGlow = vec2(0.5, 0.5); //how brightly the light bulb part glows
|
2014-11-30 20:11:22 +00:00
|
|
|
extern bool invert_normal; //if the light should invert normals
|
2014-11-29 19:52:05 +00:00
|
|
|
|
|
|
|
vec4 effect(vec4 color, Image texture, vec2 texture_coords, vec2 pixel_coords) {
|
|
|
|
vec4 pixelColor = Texel(texture, texture_coords);
|
|
|
|
|
2014-12-11 02:42:04 +00:00
|
|
|
float dist = distance(lightPosition, vec3(pixel_coords, 1.0));
|
|
|
|
//if the pixel is within this lights range
|
|
|
|
if(dist > lightRange) {
|
2014-11-30 19:35:18 +00:00
|
|
|
return vec4(0.0, 0.0, 0.0, 1.0);
|
2014-12-11 02:42:04 +00:00
|
|
|
}else{
|
2014-11-30 20:11:22 +00:00
|
|
|
//calculater attenuation of light based on the distance
|
2014-11-29 19:52:05 +00:00
|
|
|
float att = clamp((1.0 - dist / lightRange) / lightSmooth, 0.0, 1.0);
|
2014-11-30 20:11:22 +00:00
|
|
|
// if not on the normal map draw attenuated shadows
|
2014-11-29 19:52:05 +00:00
|
|
|
if(pixelColor.a == 0.0) {
|
2014-12-14 06:14:42 +00:00
|
|
|
vec3 pixel = lightColor * pow(att, lightSmooth) + pow(smoothstep(lightGlow.x, 1.0, att), lightSmooth) * lightGlow.y;
|
2014-11-30 20:11:22 +00:00
|
|
|
//If on the shadow map add the shadow color
|
2014-12-14 06:14:42 +00:00
|
|
|
vec4 shadowColor = Texel(shadowMap, texture_coords);
|
2014-11-30 00:10:15 +00:00
|
|
|
if(shadowColor.a > 0.0) {
|
2014-11-30 19:35:18 +00:00
|
|
|
pixel.rgb = pixel.rgb * shadowColor.rgb;
|
2014-11-30 00:10:15 +00:00
|
|
|
}
|
2014-12-14 06:14:42 +00:00
|
|
|
return vec4(pixel, 1.0);
|
2014-11-29 19:52:05 +00:00
|
|
|
} else {
|
2014-11-30 20:11:22 +00:00
|
|
|
//on the normal map, draw normal shadows
|
2014-12-14 06:14:42 +00:00
|
|
|
vec3 lightDir = vec3((lightPosition.xy - pixel_coords.xy) / love_ScreenSize.xy, lightPosition.z);
|
|
|
|
lightDir.x *= love_ScreenSize.x / love_ScreenSize.y;
|
|
|
|
vec3 normal = normalize(vec3(pixelColor.r,(invert_normal ? 1 - pixelColor.g : pixelColor.g), pixelColor.b) * 2.0 - 1.0);
|
|
|
|
vec3 diffuse = lightColor * max(dot(normalize(normal), normalize(lightDir)), 0.0);
|
2014-11-30 20:11:22 +00:00
|
|
|
//return the light that is effected by the normal and attenuation
|
2014-12-14 06:14:42 +00:00
|
|
|
return vec4(diffuse * att, 1.0);
|
2014-11-29 19:52:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|