As part of my audience building for Watt Interactive, I created a few shaders and related tutorial videos that I shared on the Godot Shaders website.
Full list of shared shaders can be found here: https://godotshaders.com/author/watt-interactive/
For a quick example, here is a screen space science fiction scanning pulse shader, and what it looks like in action.
/*
Basic sci-fi pulse post-processing effect.
Required to be put on a quad mesh that is rendered in the scene.
Required associated script to increase the radius over time.
Video tutorial on YouTube: https://youtu.be/x1dIJdz8Uj8
Written by Michael Watt
Thanks to Inigo Quilez for the SDF (https://iquilezles.org/articles/distfunctions/)
Thanks to nonunknown for the conditional statement replacements (https://godotshaders.com/shader/optimize-your-shaders/)
*/
shader_type spatial;
render_mode unshaded;
// Settings to play with
uniform mat4 start_point = mat4(1.0);
uniform float pulse_width = 2.0;
uniform vec4 color : source_color = vec4(1.0);
// Updated by Script
uniform float radius = 5.0;
// Access the screen and depth buffers
uniform sampler2D SCREEN_TEXTURE : hint_screen_texture, filter_linear_mipmap;
uniform sampler2D DEPTH_TEXTURE : hint_depth_texture, filter_linear_mipmap;
// Necessary for rebuilding the world coordinates
varying mat4 CAMERA;
// Function from Inigo Quilez https://iquilezles.org/articles/distfunctions/
float sdSphere( vec3 p, float s ) {
return length(p)-s;
}
// Replacements for < and > because math on GPU is fast. They return 1 or 0
float when_lt(float left_side, float right_side) {
return max(sign(right_side - left_side), 0.0);
}
float when_gt(float left_side, float right_side) {
return max(sign(left_side - right_side), 0.0);
}
void vertex() {
POSITION = vec4(VERTEX, 1.0);
CAMERA = INV_VIEW_MATRIX;
}
void fragment() {
// Get the original screen rendered texture at the screen uv coordinates.
vec4 original = textureLod(SCREEN_TEXTURE, SCREEN_UV, 0.0);
// Get the depth value form the depth buffer.
float depth = textureLod(DEPTH_TEXTURE, SCREEN_UV, 0.0).x;
vec3 ndc = vec3(SCREEN_UV * 2.0 - 1.0, depth);
// Unecessary for this effect, but to get the linear depth value,
// use the following code.
// vec4 view = INV_PROJECTION_MATRIX * vec4(ndc, 1.0);
// view.xyz /= view.w;
// float linear_depth = -view.z;
// Calculate the fragment's world position
vec4 world = CAMERA * INV_PROJECTION_MATRIX * vec4(ndc, 1.0);
vec3 world_position = world.xyz / world.w;
// Use the provided start transform and shift the world position to match
// for the SDF to work as expected.
vec3 adjusted_position = (inverse(start_point) * vec4(world_position, 1.0)).xyz;
float dist = sdSphere(adjusted_position, radius);
// Extra calculations to get the correct gradient direction.
// Using calculation functions in place of if statements.
float mix_ratio = 0.0;
float check = when_lt(dist, 0.0) * when_gt(dist, -pulse_width);
float percentage = abs(dist) / abs(pulse_width);
mix_ratio = 1.0 * check - percentage;
mix_ratio = clamp(mix_ratio, 0.0, 1.0);
// Set the albedo to the mix between the original screen and the added
// pulse color.
ALBEDO = mix(original.rgb, color.rgb, mix_ratio);
}