72 lines
2.2 KiB
Text
72 lines
2.2 KiB
Text
|
shader_type spatial;
|
||
|
|
||
|
// Development shader used to debug or help authoring.
|
||
|
|
||
|
#include "include/heightmap.gdshaderinc"
|
||
|
|
||
|
uniform sampler2D u_terrain_heightmap;
|
||
|
uniform sampler2D u_terrain_normalmap;
|
||
|
uniform sampler2D u_terrain_colormap;
|
||
|
uniform sampler2D u_map; // This map will control color
|
||
|
uniform mat4 u_terrain_inverse_transform;
|
||
|
uniform mat3 u_terrain_normal_basis;
|
||
|
|
||
|
varying float v_hole;
|
||
|
|
||
|
|
||
|
vec3 unpack_normal(vec4 rgba) {
|
||
|
// If we consider texture space starts from top-left corner and Y goes down,
|
||
|
// then Y+ in pixel space corresponds to Z+ in terrain space,
|
||
|
// while X+ also corresponds to X+ in terrain space.
|
||
|
vec3 n = rgba.xzy * 2.0 - vec3(1.0);
|
||
|
// Had to negate Z because it comes from Y in the normal map,
|
||
|
// and OpenGL-style normal maps are Y-up.
|
||
|
n.z *= -1.0;
|
||
|
return n;
|
||
|
}
|
||
|
|
||
|
void vertex() {
|
||
|
vec4 wpos = MODEL_MATRIX * vec4(VERTEX, 1);
|
||
|
vec2 cell_coords = (u_terrain_inverse_transform * wpos).xz;
|
||
|
// Must add a half-offset so that we sample the center of pixels,
|
||
|
// otherwise bilinear filtering of the textures will give us mixed results (#183)
|
||
|
cell_coords += vec2(0.5);
|
||
|
|
||
|
// Normalized UV
|
||
|
UV = cell_coords / vec2(textureSize(u_terrain_heightmap, 0));
|
||
|
|
||
|
// Height displacement
|
||
|
float h = sample_heightmap(u_terrain_heightmap, UV);
|
||
|
VERTEX.y = h;
|
||
|
wpos.y = h;
|
||
|
|
||
|
// Putting this in vertex saves 2 fetches from the fragment shader,
|
||
|
// which is good for performance at a negligible quality cost,
|
||
|
// provided that geometry is a regular grid that decimates with LOD.
|
||
|
// (downside is LOD will also decimate tint and splat, but it's not bad overall)
|
||
|
vec4 tint = texture(u_terrain_colormap, UV);
|
||
|
v_hole = tint.a;
|
||
|
|
||
|
// Need to use u_terrain_normal_basis to handle scaling.
|
||
|
NORMAL = u_terrain_normal_basis * unpack_normal(texture(u_terrain_normalmap, UV));
|
||
|
}
|
||
|
|
||
|
void fragment() {
|
||
|
if (v_hole < 0.5) {
|
||
|
// TODO Add option to use vertex discarding instead, using NaNs
|
||
|
discard;
|
||
|
}
|
||
|
|
||
|
vec3 terrain_normal_world =
|
||
|
u_terrain_normal_basis * unpack_normal(texture(u_terrain_normalmap, UV));
|
||
|
terrain_normal_world = normalize(terrain_normal_world);
|
||
|
vec3 normal = terrain_normal_world;
|
||
|
|
||
|
vec4 value = texture(u_map, UV);
|
||
|
// TODO Blend toward checker pattern to show the alpha channel
|
||
|
|
||
|
ALBEDO = value.rgb;
|
||
|
ROUGHNESS = 0.5;
|
||
|
NORMAL = (VIEW_MATRIX * (vec4(normal, 0.0))).xyz;
|
||
|
}
|