veloren/assets/voxygen/shaders/clouds-frag.glsl

317 lines
14 KiB
Plaintext
Raw Normal View History

#version 420 core
2020-11-15 22:18:35 +00:00
#include <constants.glsl>
#define LIGHTING_TYPE (LIGHTING_TYPE_TRANSMISSION | LIGHTING_TYPE_REFLECTION)
#define LIGHTING_REFLECTION_KIND LIGHTING_REFLECTION_KIND_SPECULAR
2022-10-23 19:03:21 +00:00
#if (FLUID_MODE == FLUID_MODE_LOW)
2020-11-15 22:18:35 +00:00
#define LIGHTING_TRANSPORT_MODE LIGHTING_TRANSPORT_MODE_IMPORTANCE
2022-09-23 14:04:47 +00:00
#elif (FLUID_MODE >= FLUID_MODE_MEDIUM)
2020-11-15 22:18:35 +00:00
#define LIGHTING_TRANSPORT_MODE LIGHTING_TRANSPORT_MODE_RADIANCE
#endif
#define LIGHTING_DISTRIBUTION_SCHEME LIGHTING_DISTRIBUTION_SCHEME_MICROFACET
#define LIGHTING_DISTRIBUTION LIGHTING_DISTRIBUTION_BECKMANN
// Must come before includes
#define IS_POSTPROCESS
2020-11-15 22:18:35 +00:00
#include <globals.glsl>
// Note: The sampler uniform is declared here because it differs for MSAA
#include <anti-aliasing.glsl>
#include <srgb.glsl>
#include <cloud.glsl>
2022-01-18 21:09:30 +00:00
#include <light.glsl>
2022-01-19 11:29:19 +00:00
// This *MUST* come after `cloud.glsl`: it contains a function that depends on `cloud.glsl` when clouds are enabled
#include <point_glow.glsl>
2022-10-04 17:11:38 +00:00
#include <random.glsl>
2020-11-15 22:18:35 +00:00
2022-03-15 18:04:21 +00:00
layout(set = 2, binding = 0)
uniform texture2D t_src_color;
2022-03-15 18:04:21 +00:00
layout(set = 2, binding = 1)
uniform sampler s_src_color;
2020-11-15 22:18:35 +00:00
2022-03-15 18:04:21 +00:00
layout(set = 2, binding = 2)
uniform texture2D t_src_depth;
2022-03-15 18:04:21 +00:00
layout(set = 2, binding = 3)
uniform sampler s_src_depth;
2020-11-15 22:18:35 +00:00
2022-03-15 18:04:21 +00:00
layout (std140, set = 2, binding = 4)
2020-11-15 22:18:35 +00:00
uniform u_locals {
mat4 all_mat_inv;
2020-11-15 22:18:35 +00:00
};
2023-01-12 15:30:13 +00:00
layout(location = 0) in vec2 uv;
layout(set = 2, binding = 5)
uniform utexture2D t_src_mat;
layout(location = 0) out vec4 tgt_color;
2020-11-15 22:18:35 +00:00
vec3 wpos_at(vec2 uv) {
2022-10-09 23:30:20 +00:00
uvec2 sz = textureSize(sampler2D(t_src_depth, s_src_depth), 0);
float buf_depth = texelFetch(sampler2D(t_src_depth, s_src_depth), clamp(ivec2(uv * sz), ivec2(0), ivec2(sz) - 1), 0).x;
//float buf_depth = texture(sampler2D(t_src_depth, s_src_depth), uv).x;
vec4 clip_space = vec4((uv * 2.0 - 1.0) * vec2(1, -1), buf_depth, 1.0);
vec4 view_space = all_mat_inv * clip_space;
2020-11-15 22:18:35 +00:00
view_space /= view_space.w;
2023-04-19 20:41:44 +00:00
return view_space.xyz;
2020-11-15 22:18:35 +00:00
}
2022-10-10 15:20:14 +00:00
float depth_at(vec2 uv) {
uvec2 sz = textureSize(sampler2D(t_src_depth, s_src_depth), 0);
float buf_depth = texelFetch(sampler2D(t_src_depth, s_src_depth), clamp(ivec2(uv * sz), ivec2(0), ivec2(sz) - 1), 0).x;
if (buf_depth == 0.0) {
return 524288.0;
} else {
vec4 clip_space = vec4((uv * 2.0 - 1.0) * vec2(1, -1), buf_depth, 1.0);
vec4 view_space = all_mat_inv * clip_space;
view_space /= view_space.w;
return -(view_mat * view_space).z;
}
}
2020-11-15 22:18:35 +00:00
void main() {
vec4 color = texture(sampler2D(t_src_color, s_src_color), uv);
2020-11-15 22:18:35 +00:00
uvec2 mat_sz = textureSize(usampler2D(t_src_mat, s_src_depth), 0);
uvec4 mat = texelFetch(usampler2D(t_src_mat, s_src_depth), clamp(ivec2(uv * mat_sz), ivec2(0), ivec2(mat_sz) - 1), 0);
#ifdef EXPERIMENTAL_VIEWNORMALS
2022-12-30 16:26:32 +00:00
tgt_color = vec4(vec3(mat.xyz) / 255.0, 1);
return;
#endif
#ifdef EXPERIMENTAL_VIEWMATERIALS
const vec3 mat_colors[5] = vec3[](
vec3(0, 1, 1), // MAT_SKY
vec3(1, 1, 0), // MAT_BLOCK
vec3(0, 0, 1), // MAT_FLUID
vec3(1, 0, 1), // MAT_FIGURE
vec3(0.5, 1, 0) // MAT_LOD
);
tgt_color = vec4(mat_colors[mat.a % 5u], 1);
return;
#endif
2023-05-31 16:07:22 +00:00
#ifdef EXPERIMENTAL_VIEWDEPTH
tgt_color = vec4(vec3(pow(clamp(depth_at(uv) / 524288.0, 0, 1), 0.3)), 1);
return;
#endif
2022-02-09 12:56:21 +00:00
#ifdef EXPERIMENTAL_BAREMINIMUM
tgt_color = vec4(color.rgb, 1);
return;
#endif
2022-01-18 21:09:30 +00:00
vec3 wpos = wpos_at(uv);
float dist = distance(wpos, cam_pos.xyz);
2022-10-23 22:54:55 +00:00
vec3 cam_dir = (wpos - cam_pos.xyz) / dist;
vec3 dir = cam_dir;
2022-01-18 21:09:30 +00:00
// Apply clouds
2022-01-21 17:29:59 +00:00
float cloud_blend = 1.0;
2022-10-23 14:15:25 +00:00
if (color.a < 1.0) {
vec2 nz = vec2(0);
2022-10-23 19:03:21 +00:00
uvec2 col_sz = textureSize(sampler2D(t_src_color, s_src_color), 0);
#if (REFLECTION_MODE >= REFLECTION_MODE_MEDIUM)
2022-10-23 14:15:25 +00:00
nz = (vec2(
noise_3d(vec3((wpos.xy + focus_off.xy) * 0.1, tick.x * 0.2 + wpos.x * 0.01)).x,
noise_3d(vec3((wpos.yx + focus_off.yx) * 0.1, tick.x * 0.2 + wpos.y * 0.01)).x
2022-10-23 19:03:21 +00:00
) - 0.5) * (dir.z < 0.0 ? color.a : 1.0);
2022-10-23 14:15:25 +00:00
const float n2 = 1.3325;
vec3 refr_dir;
// TODO: Proper refraction
// if (medium.x == MEDIUM_WATER) {
// vec3 surf_norm = normalize(vec3(nz * 0.03 / (1.0 + dist * 0.1), 1));
// refr_dir = refract(dir, surf_norm * -sign(dir.z), 1.0 / n2);
// } else {
2022-10-23 19:03:21 +00:00
refr_dir = normalize(dir + vec3(nz * 1.5 / dist, 0.0));
2022-10-23 14:15:25 +00:00
// }
vec4 clip = (all_mat * vec4(cam_pos.xyz + refr_dir, 1.0));
vec2 new_uv = (clip.xy / max(clip.w, 0)) * 0.5 * vec2(1, -1) + 0.5;
float uv_merge = clamp((1.0 - abs(new_uv.y - 0.5) * 2) * 5.0, 0, 1);
2022-10-23 19:03:21 +00:00
new_uv = mix(uv, new_uv, uv_merge);
2022-10-23 14:15:25 +00:00
2022-10-23 19:03:21 +00:00
vec4 new_col = texelFetch(sampler2D(t_src_color, s_src_color), clamp(ivec2(new_uv * col_sz), ivec2(0), ivec2(col_sz) - 1), 0);
2022-10-23 14:15:25 +00:00
if (new_col.a < 1.0) {
color = new_col;
2022-10-23 19:03:21 +00:00
dir = refr_dir;
2022-10-23 14:15:25 +00:00
}
#endif
{
cloud_blend = 1.0 - color.a;
2022-10-23 20:41:45 +00:00
#if (FLUID_MODE >= FLUID_MODE_MEDIUM || REFLECTION_MODE >= REFLECTION_MODE_MEDIUM)
if (mat.a != MAT_SKY) {
2023-01-12 15:30:13 +00:00
vec3 surf_norm = vec3(mat.xyz) / 127.0 - 1.0;
2022-10-23 14:15:25 +00:00
vec3 refl_dir = reflect(dir, surf_norm);
// Don't reflect back into the surface by snapping the reflection to the *actual* (i.e: not normal-mapped) surface plane
// TODO: Find a good way to know the *actual* surface normal, minus normal mapping
vec3 flat_norm = vec3(0, 0, 1);//round(surf_norm);
if (dot(refl_dir, flat_norm) <= 0.0) {
// TODO: This assumes that the surface is axis-aligned!
refl_dir = normalize(refl_dir.xyz * (1.0 - abs(flat_norm)));
}
2022-10-23 14:15:25 +00:00
vec4 clip = (all_mat * vec4(cam_pos.xyz + refl_dir, 1.0));
vec2 new_uv = (clip.xy / max(clip.w, 0)) * 0.5 * vec2(1, -1) + 0.5;
2022-10-23 19:03:21 +00:00
#if (REFLECTION_MODE >= REFLECTION_MODE_HIGH)
2022-10-23 14:15:25 +00:00
vec3 ray_end = wpos + refl_dir * 5.0 * dist;
// Trace through the screen-space depth buffer to find the ray intersection
const int MAIN_ITERS = 64;
for (int i = 0; i < MAIN_ITERS; i ++) {
float t = float(i) / float(MAIN_ITERS);
// TODO: Trace in screen space, not world space
vec3 swpos = mix(wpos, ray_end, t);
vec3 svpos = (view_mat * vec4(swpos, 1)).xyz;
vec4 clippos = proj_mat * vec4(svpos, 1);
vec2 suv = (clippos.xy / clippos.w) * 0.5 * vec2(1, -1) + 0.5;
float d = -depth_at(suv);
if (d < svpos.z * 0.8 && d > svpos.z * 0.999) {
// Don't cast into water!
2022-10-23 19:03:21 +00:00
if (texelFetch(sampler2D(t_src_color, s_src_color), clamp(ivec2(suv * col_sz), ivec2(0), ivec2(col_sz) - 1), 0).a >= 1.0) {
2022-10-23 14:15:25 +00:00
/* t -= 1.0 / float(MAIN_ITERS); */
// Do a bit of extra iteration to try to refine the estimate
const int ITERS = 8;
float diff = 1.0 / float(MAIN_ITERS);
for (int i = 0; i < ITERS; i ++) {
vec3 swpos = mix(wpos, ray_end, t);
svpos = (view_mat * vec4(swpos, 1)).xyz;
vec4 clippos = proj_mat * vec4(svpos, 1);
suv = (clippos.xy / clippos.w) * 0.5 * vec2(1, -1) + 0.5;
float d = -depth_at(suv);
t += ((d > svpos.z * 0.999) ? -1.0 : 1.0) * diff;
diff *= 0.5;
}
// Small offset to push us into obscured territory
new_uv = suv - vec2(0, 0.001);
break;
2022-10-10 21:07:07 +00:00
}
2022-10-10 19:44:17 +00:00
}
2022-10-10 15:20:14 +00:00
}
2022-10-23 14:15:25 +00:00
#endif
2022-12-31 19:19:52 +00:00
#ifdef EXPERIMENTAL_SMEARREFLECTIONS
const float SMEAR_FRAC = 0.2;
vec2 anew_uv = abs(new_uv - 0.5) * 2;
new_uv = mix(
anew_uv,
1.0 - SMEAR_FRAC + (1.0 - 1.0 / (1.0 + (anew_uv - 1.0 + SMEAR_FRAC))) * SMEAR_FRAC,
lessThan(vec2(1.0 - SMEAR_FRAC), anew_uv)
) * sign(new_uv - 0.5) * 0.5 + 0.5;
#else
new_uv = clamp(new_uv, vec2(0), vec2(1));
#endif
2022-10-23 14:15:25 +00:00
vec3 new_wpos = wpos_at(new_uv);
float new_dist = distance(new_wpos, cam_pos.xyz);
float merge = min(
// Off-screen merge factor
2022-12-31 19:19:52 +00:00
#ifdef EXPERIMENTAL_SMEARREFLECTIONS
1.0,
#else
clamp((1.0 - max(abs(new_uv.y - 0.5), abs(new_uv.x - 0.5)) * 2) * 6.0, 0, 1),
#endif
2022-10-23 14:15:25 +00:00
// Depth merge factor
clamp((new_dist - dist * 0.5) / (dist * 0.5), 0.0, 1.0)
);
vec3 refl_col;
float not_underground = 1.0;
// Make underground water look more correct
#if (REFLECTION_MODE >= REFLECTION_MODE_HIGH)
2022-12-30 17:24:36 +00:00
float f_alt = alt_at(wpos.xy);
not_underground = clamp((wpos.z - f_alt) / 32.0 + 1.0, 0.0, 1.0);
#endif
// Did we hit a surface during reflection?
2022-10-23 14:15:25 +00:00
if (merge > 0.0) {
// Yes: grab the new material from screen space
2022-12-30 16:01:06 +00:00
uvec4 new_mat = texelFetch(usampler2D(t_src_mat, s_src_depth), clamp(ivec2(new_uv * mat_sz), ivec2(0), ivec2(mat_sz) - 1), 0);
// If it's the sky, just go determine the sky color analytically to avoid sampling the incomplete skybox
// Otherwise, pull the color from the screen-space color buffer
vec3 sky_col = min(get_sky_color(refl_dir, time_of_day.x, wpos, vec3(-100000), 0.125, false, 0.0, true, 0.0), vec3(1)) * not_underground;
if (new_mat.a == MAT_SKY) {
2022-12-30 16:01:06 +00:00
refl_col = sky_col;
} else {
refl_col = mix(sky_col, texelFetch(sampler2D(t_src_color, s_src_color), clamp(ivec2(new_uv * col_sz), ivec2(0), ivec2(col_sz) - 1), 0).rgb, merge);
}
// Apply clouds to reflected colour
refl_col = mix(refl_col, get_cloud_color(refl_col, refl_dir, wpos, time_of_day.x, distance(new_wpos, wpos.xyz), 1.0), not_underground);
} else {
// No: assume that anything off-screen is the colour of the sky
refl_col = min(get_sky_color(refl_dir, time_of_day.x, wpos, vec3(-100000), 0.125, true, 1.0, true, 1.0) * not_underground, vec3(1));
// Apply clouds to reflection
refl_col = mix(refl_col, get_cloud_color(refl_col, refl_dir, wpos, time_of_day.x, 100000.0, 1.0), not_underground);
2022-10-10 16:30:50 +00:00
}
2022-12-31 13:04:28 +00:00
color.rgb = mix(color.rgb, refl_col, color.a);
2022-10-23 19:03:21 +00:00
cloud_blend = 1;
2022-10-22 14:01:23 +00:00
} else {
2022-10-23 14:15:25 +00:00
#else
{
#endif
cloud_blend = 1;
}
2022-10-04 17:11:38 +00:00
}
2022-01-21 17:29:59 +00:00
}
color.rgb = mix(color.rgb, get_cloud_color(color.rgb, dir, cam_pos.xyz, time_of_day.x, dist, 1.0), cloud_blend);
2022-01-21 17:29:59 +00:00
#if (CLOUD_MODE == CLOUD_MODE_NONE)
2022-01-21 00:50:14 +00:00
color.rgb = apply_point_glow(cam_pos.xyz + focus_off.xyz, dir, dist, color.rgb);
2022-07-15 07:06:11 +00:00
#else
2022-10-22 15:37:25 +00:00
if (medium.x == MEDIUM_AIR && rain_density > 0.001) {
2022-07-08 13:47:43 +00:00
vec3 cam_wpos = cam_pos.xyz + focus_off.xyz;
2022-10-23 22:54:55 +00:00
vec3 adjusted_dir = (vec4(cam_dir, 0) * rain_dir_mat).xyz;
2022-07-08 13:47:43 +00:00
vec2 dir2d = adjusted_dir.xy;
2022-07-08 17:09:13 +00:00
vec3 rorigin = cam_pos.xyz + focus_off.xyz + 0.5;
2022-07-08 14:09:38 +00:00
vec3 rpos = vec3(0.0);
2022-07-08 13:47:43 +00:00
float t = 0.0;
2022-07-10 12:53:31 +00:00
const float PLANCK = 0.01;
2022-07-15 07:06:11 +00:00
for (int i = 0; i < 14 /* log2(64) * 2 + 2 */; i ++) {
2022-07-08 19:40:08 +00:00
float scale = min(pow(2, ceil(t / 2.0)), 32);
2022-07-08 19:18:07 +00:00
vec2 deltas = (step(vec2(0), dir2d) - fract(rpos.xy / scale + 100.0)) / dir2d;
2022-07-08 14:09:38 +00:00
float jump = max(min(deltas.x, deltas.y) * scale, PLANCK);
t += jump;
2022-07-10 15:40:49 +00:00
2022-07-16 13:44:58 +00:00
#if (CLOUD_MODE >= CLOUD_MODE_MEDIUM)
if (t >= 64.0) { break; }
#else
if (t >= 16.0) { break; }
#endif
2022-07-10 15:40:49 +00:00
2022-07-08 15:07:15 +00:00
rpos = rorigin + adjusted_dir * t;
2022-07-08 13:47:43 +00:00
vec2 diff = abs(round(rpos.xy) - rpos.xy);
2022-07-08 15:09:36 +00:00
vec3 wall_pos = vec3((diff.x > diff.y) ? rpos.xy : rpos.yx, rpos.z + integrated_rain_vel);
2022-07-08 13:47:43 +00:00
wall_pos.xz *= vec2(4, 0.3);
2022-07-10 12:53:31 +00:00
wall_pos.z += hash_two(uvec2(wall_pos.xy + vec2(0, 0.5)));
2022-07-08 13:47:43 +00:00
2022-07-10 15:40:49 +00:00
float depth_adjust = fract(hash_two(uvec2(wall_pos.xz) + 500u));
2022-07-08 15:41:31 +00:00
float wpos_dist = t - jump * depth_adjust;
vec3 wpos = cam_pos.xyz + dir * wpos_dist;
2022-07-08 14:09:38 +00:00
2022-07-08 15:41:31 +00:00
if (wpos_dist > dist) { break; }
2023-01-12 15:30:13 +00:00
vec2 wall_pos_half = fract(wall_pos.xz) - 0.5;
if (dot(wall_pos_half, wall_pos_half) < 0.01 + pow(max(0.0, wpos_dist - (dist - 0.25)) / 0.25, 4.0) * 0.2) {
2022-07-10 12:53:31 +00:00
float density = rain_density * rain_occlusion_at(wpos);
2022-07-10 15:40:49 +00:00
if (fract(hash_two(uvec2(wall_pos.xz) + 1000u)) >= density) { continue; }
2022-07-10 12:53:31 +00:00
2022-07-08 19:18:07 +00:00
float alpha = 0.5 * clamp((wpos_dist - 1.0) * 0.5, 0.0, 1.0);
2022-07-10 15:40:49 +00:00
float light = dot(color.rgb, vec3(1)) + 0.05 + (get_sun_brightness() + get_moon_brightness()) * 0.2;
color.rgb = mix(color.rgb, vec3(0.3, 0.35, 0.5) * light, alpha);
2022-07-08 13:47:43 +00:00
}
}
}
#endif
2022-02-06 18:56:45 +00:00
2020-11-15 22:18:35 +00:00
tgt_color = vec4(color.rgb, 1);
}