uniform sampler2D ntex; uniform sampler2D dtex; uniform sampler2DArrayShadow shadowtex; //uniform sampler2D warpx; ///uniform sampler2D warpy; uniform vec3 direction; uniform vec3 col; uniform mat4 invproj; uniform mat4 shadowmat[4]; //uniform int hasclouds; //uniform vec2 wind; //uniform float shadowoffset; #if __VERSION__ >= 130 in vec2 uv; out vec4 Diff; out vec4 Spec; #else varying vec2 uv; #define Diff gl_FragData[0] #define Spec gl_FragData[1] #endif vec3 DecodeNormal(vec2 n); float getShadowFactor(vec3 pos, float bias, int index) { //float a[5] = float[](3.4, 4.2, 5.0, 5.2, 1.1); vec2 shadowoffset[4] = vec2[]( vec2(-1., -1.), vec2(-1., 1.), vec2(1., -1.), vec2(1., 1.) ); vec4 shadowcoord = (shadowmat[index] * vec4(pos, 1.0)); shadowcoord /= shadowcoord.w; vec2 shadowtexcoord = shadowcoord.xy * 0.5 + 0.5; // shadowcoord = (shadowcoord * 0.5) + vec3(0.5); // float movex = decdepth(texture(warpx, shadowcoord.xy)); // float movey = decdepth(texture(warpy, shadowcoord.xy)); // float dx = movex * 2.0 - 1.0; // float dy = movey * 2.0 - 1.0; // shadowcoord.xy += vec2(dx, dy);*/ //float shadowmapz = 2. * texture(shadowtex, vec3(shadowtexcoord, shadowcoord.z).x - 1.; // bias += smoothstep(0.001, 0.1, moved) * 0.014; // According to the warping float sum = 0.; for (int i = 0; i < 4; i++) { sum += texture(shadowtex, vec4(shadowtexcoord + 0.0005 * shadowoffset[i], float(index), 0.5 * (shadowcoord.z + bias * 0.001) + 0.5)); } return sum / 4.; } void main() { float z = texture(dtex, uv).x; vec4 xpos = 2.0 * vec4(uv, z, 1.0) - 1.0; xpos = invproj * xpos; xpos.xyz /= xpos.w; vec3 norm = normalize(DecodeNormal(2. * texture(ntex, uv).xy - 1.)); // Normalized on the cpu vec3 L = direction; float NdotL = max(0.0, dot(norm, L)); vec3 R = reflect(L, norm); float RdotE = max(0.0, dot(R, normalize(xpos.xyz))); float Specular = pow(RdotE, 200); vec3 outcol = NdotL * col; // if (hasclouds == 1) // { // vec2 cloudcoord = (xpos.xz * 0.00833333) + wind; // float cloud = texture(cloudtex, cloudcoord).x; // //float cloud = step(0.5, cloudcoord.x) * step(0.5, cloudcoord.y); // outcol *= cloud; // } // Shadows float bias = 0.002 * tan(acos(NdotL)); // According to the slope bias = clamp(bias, 0.001, 0.014); float factor; if (xpos.z < 20.) factor = getShadowFactor(xpos.xyz, bias, 0); else if (xpos.z < 25.) { float a = getShadowFactor(xpos.xyz, bias, 0), b = getShadowFactor(xpos.xyz, bias, 1); factor = mix(a, b, (xpos.z - 20.) / 5.); } else if (xpos.z < 50.) factor = getShadowFactor(xpos.xyz, bias, 1); else if (xpos.z < 60.) { float a = getShadowFactor(xpos.xyz, bias, 1), b = getShadowFactor(xpos.xyz, bias, 2); factor = mix(a, b, (xpos.z - 50.) / 10.); } else if (xpos.z < 100.) factor = getShadowFactor(xpos.xyz, bias, 2); else if (xpos.z < 120.) { float a = getShadowFactor(xpos.xyz, bias, 2), b = getShadowFactor(xpos.xyz, bias, 3); factor = mix(a, b, (xpos.z - 100.) / 20.); } else factor = getShadowFactor(xpos.xyz, bias, 3); Diff = vec4(factor * NdotL * col, 1.); Spec = vec4(factor * Specular * col, 1.); return; // float moved = (abs(dx) + abs(dy)) * 0.5; // float avi = 0.002; // float abi = 0.0025; /* float avi = 0.0018; float abi = 0.002; float bias = avi * tan(acos(NdotL)); // According to the slope bias += smoothstep(0.001, 0.1, moved) * abi; // According to the warping bias = clamp(bias, 0.001, abi); // This ID, and four IDs around this must match for a shadow pixel float right = texture(shadowtex, shadowcoord.xy + vec2(shadowoffset, 0.0)).a; float left = texture(shadowtex, shadowcoord.xy + vec2(-shadowoffset, 0.0)).a; float up = texture(shadowtex, shadowcoord.xy + vec2(0.0, shadowoffset)).a; float down = texture(shadowtex, shadowcoord.xy + vec2(0.0, -shadowoffset)).a; float matching = ((right + left + up + down) * 0.25) - shadowread.a; matching = abs(matching) * 400.0; // If the ID is different, we're likely in shadow - cut the bias to cut peter panning float off = 7.0 - step(abs(shadowread.a - depthread.a) - matching, 0.004) * 6.0; bias /= off; const float softness = 8.0; // How soft is the light? float shadowed = step(shadowmapz + bias, shadowcoord.z); float dist = (shadowcoord.z / shadowmapz) - 1.0; float penumbra = dist * softness / gl_FragCoord.z; penumbra *= shadowed;*/ /* outcol.r = (shadowcoord.z - shadowmapz) * 50.0; outcol.g = moved;*/ // FragColor = vec4(outcol, 0.05); // OtherOutput = vec4(shadowed, penumbra, shadowed, shadowed); }