aboutsummaryrefslogtreecommitdiff
path: root/game/addons/zylann.hterrain/tools/generator/shaders/erode.gdshader
blob: 6f45e1c6f0a7fa6be17c2f042c9147c3ff9e8ac5 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
shader_type canvas_item;
render_mode blend_disabled;

#include "res://addons/zylann.hterrain/shaders/include/heightmap.gdshaderinc"

uniform vec2 u_slope_up = vec2(0, 0);
uniform float u_slope_factor = 1.0;
uniform bool u_slope_invert = false;
uniform float u_weight = 0.5;
uniform float u_dilation = 0.0;
//uniform sampler2D u_screen_texture : hint_screen_texture;
uniform sampler2D u_previous_pass;

float get_height(sampler2D tex, vec2 uv) {
   return sample_height_from_viewport(tex, uv);
}

void fragment() {
   float r = 3.0;
   
   // Divide so the shader stays neighbor dependent 1 pixel across.
   // For this to work, filtering must be enabled.
   vec2 eps = SCREEN_PIXEL_SIZE / (0.99 * r);
   
   vec2 uv = SCREEN_UV;
   float h = get_height(u_previous_pass, uv);
   float eh = h;
   float dh = h;
   
   // Morphology with circular structuring element
   for (float y = -r; y <= r; ++y) {
      for (float x = -r; x <= r; ++x) {
         
         vec2 p = vec2(float(x), float(y));
         float nh = get_height(u_previous_pass, uv + p * eps);
         
         float s = max(length(p) - r, 0);
         eh = min(eh, nh + s);

         s = min(r - length(p), 0);
         dh = max(dh, nh + s);
      }
   }
   
   eh = mix(h, eh, u_weight);
   dh = mix(h, dh, u_weight);
   
   float ph = mix(eh, dh, u_dilation);
   
   if (u_slope_factor > 0.0) {
      vec2 ps = SCREEN_PIXEL_SIZE;
      
      float left = get_height(u_previous_pass, uv + vec2(-ps.x, 0.0));
      float right = get_height(u_previous_pass, uv + vec2(ps.x, 0.0));
      float top = get_height(u_previous_pass, uv + vec2(0.0, ps.y));
      float bottom = get_height(u_previous_pass, uv + vec2(0.0, -ps.y));
      
      vec3 normal = normalize(vec3(left - right, ps.x + ps.y, bottom - top));
      vec3 up = normalize(vec3(u_slope_up.x, 1.0, u_slope_up.y));
      
      float f = max(dot(normal, up), 0);
      if (u_slope_invert) {
         f = 1.0 - f;
      }
      
      ph = mix(h, ph, mix(1.0, f, u_slope_factor));
      //COLOR = vec4(f, f, f, 1.0);
   }
   
   //COLOR = vec4(0.5 * normal + 0.5, 1.0);
   
   //eh = 0.5 * (eh + texture(SCREEN_TEXTURE, uv + mp * ps * k).r);
   //eh = mix(h, eh, (1.0 - h) / r);
   
   COLOR = encode_height_to_viewport(ph);
}