Examples

Breakdown

The Equation

These animations were achieved using the following equation:

float h = A * sin(k*p - w*t);

A = amplitude;
k = spatial_angular_frequency; // (a.k.a. wavenumber)
p = position in space;
w = temporal_angular_frequency;
t = time;

This equation is a sinusoidal function; it represents periodic oscillations or waves. We can use the result of this function as a height to some vertex we are drawing in our world.

In a 3D world where +X is right, +Y is up, and +Z is towards the viewer, I create an xz-plane with a bunch of vertices. Then in the vertex shader, I calculate the height of each vertex (y-coordinate) based on this equation.

It’s more intuitive to parametrize a wave like so:

amplitude:  Half-height of wave, i.e., distance from crest/trough to still-water line.
wavelength: Length of one wave cycle.
speed:      How fast the wave moves.

From these parameters, we can derive:

// Wavenumber / Spatial angular frequency.
k = 2 * PI / wavelength;

// Temporal angular frequency.
w = k * speed;

// We can use it directly.
A = amplitude;

// Time is a value we increase every frame in the CPU by delta_time and we pass it to GPU in a constant buffer.
t = time;

// Position of the current vertex in some dimension. We can do cool patterns by modifying this value as I'll show later.
p = vertex.position.x and/or vertex.position.z;

With this, we can calculate the height of each vertex (of the plane) using that equation and we’d get something that resembles a sine wave moving on the surface of the plane.

The cool part is that we can combine the result of multiple sinusoidal waves where each wave has different parameters:

So, in HLSL, we have something like the following:

struct Water_Wave
{
    float amplitude;
    float spatial_angular_frequency;  // TAU / wavelength;
    float temporal_angular_frequency; // spatial_angular_frequency * speed;
};

cbuffer Constants : register(b0)
{
    float time;
    int wave_count;

    // ...
}

// We pass a bunch of random Waves from the CPU.
StructuredBuffer<Water_Wave> waves : register(t0);

float3 sine(float3 v_pos, Water_Wave wave)
{
    float k = wave.spatial_angular_frequency;
    float w = wave.temporal_angular_frequency;
    float p = v_pos.x; // -/+v_pos.x: wave moves left/right... -/+v_pos.z: wave moves forward/backward.
    float h = wave.amplitude * sin(k * p - w * time);

    // Displace along +Y axis.
    return float3(0, h, 0);
}

PS_Input vs(float3 position : POSITION)
{
    // ...

    // Position perturbation.
    float3 offset = 0.0f;
    for (int wave_index = 0; wave_index < wave_count; wave_index++) {
        offset += sine(position, waves[wave_index]);
    }

    // Final pos = original position + position perturbation. 
    float3 final_pos = position + offset;
    
    // final_pos is in local space here, we can transform it now.

    // ...
}

Different Wave Patterns

Instead of directly using the vertex position in the sine() function, we can achieve different wave patterns by interpreting the vertex position differently:

Directional

float get_wave_coord(float3 v_pos, Water_Wave wave)
{
    // Project vertex position on desired direction.
    float result = 0.0f;
    result = dot(v_pos.xz, wave.direction);
    return result;
}

float3 sine(float3 v_pos, Water_Wave wave)
{
    float k = wave.spatial_angular_frequency;
    float w = wave.temporal_angular_frequency;
    float p = get_wave_coord(v_pos, wave);
    float h = wave.amplitude * sin(k * p - w * time);

    // Displace along +Y axis.
    return float3(0, h, 0);
}

Radial

{
    // Distance from origin.
    result = length(v_pos);
}

Oscillating Radial

{
    // Oscillating wave along a circular path.
    result = sin(length(v_pos));
}

Spiral

{
    float angle  = atan2(v_pos.x, v_pos.z);
    float radius = length(v_pos);
    result       = sin(radius - angle);
}

Elliptical

{
    // Elliptical wave with different axis lengths.
    result = length(v_pos.xz / float2(2.0, 1.0f));
}

Latticed

{
    result = sin(v_pos.x * 2.0) + sin(v_pos.z * 2.0);
}

Calculating Normals

Normals are direction vectors that point directly away from a surface. We need normals for light calculations that make our waves look nice.

The way we calculate a normal of a surface at some point ‘p’ is by doing a cross product between two other vectors: the tangent and bitangent. The tangent and bitangent are vectors that “just touch” the surface at the point ‘p’. They are perpendicular to one another and doing a cross product between them gives us a third vector that is perpendicular to both (the normal):

Note: There are infinite tangent and bitangent lines for a given surface, I just have to ensure that whichever ones I choose, it must be always true that normal = cross(tangent, bitangent), noting that I use a right-handed system.

To retrieve the tangent and bitangent of a surface at some point, we need to do some calculus, so let’s take a step back:
Imagine a hill that rises and falls as you walk in the x-direction. The height value varies based on ‘x’. The rate of change of height tells us how steeply the surface rises or falls in the x-direction. This is the same thing as saying:

  • The rate of change of ‘y’ with respect to ‘x’
  • The derivative of ‘y’ with respect to ‘x’
  • dy/dx
  • The slope of the height of the surface at point ‘p’ in direction ‘x’

The direction of the tangent is along the surface of the hill.
So, the tangent at a point where the height is zero is: float3(1, 0, 0)
Otherwise the tangent will be tilted based on the rate of change of height with respect to the x-direction.

We know the function we’re using to calculate the height (assume simple sine wave propagating in x-direction):

float h = A * sin(k*pos.x - w*t);

The derivative of that is:

float dh_dx = A * k * cos(k*pos.x - w*t);

The tangent is then:

float3 tangent = normalize(float3(1, dh_dx, 0));

Now let’s do the same in 3D:

We’re dealing with the xz-plane; the normal of that plane is the y-axis/up-vector. There are infinite possibilities to choose from when it comes to the direction of the tangent and bitangent vectors. But remember, I just have to ensure that N = cross(T, B).

Let’s choose (T = z-axis) and (B = x-axis).
Now doing cross(T, B) correctly gives the y-axis (use the right-hand rule to see).

So:

float dh_dx = A * k * cos(k*pos.x - w*t);
float dh_dz = A * k * cos(k*pos.z - w*t);

float3 tangent   = normalize(float3(0, dh_dz, 1)); // Along z-axis
float3 bitangent = normalize(float3(1, dh_dz, 0)); // Along x-axis

float3 normal = cross(tangent, bitangent) = float3(-dh_dx, 1, -dh_dz);

We can omit the cross product calculation by directly doing float3 normal = float3(-dh_dx, 1.0, -dh_dz);
However, it’s super important to note that we have multiple waves, so we want to accumulate the “tilt” of the normal from all the waves:

float4 ps(PS_Input input) : SV_TARGET
{
    // ...

    // Normal perturbation.
    float3 offset  = 0.0f;
    for (int wave_index = 0; wave_index < wave_count; wave_index++) {
        offset += sine_normal(input.local_pos, waves[wave_index]);
    }
    
    // Final nor = original normal + normal perturbation.
    float3 final_nor = float3(0,1,0) + offset;

    // final_nor is in local space here, we can transform and normalize it now.

    // ... 
}

Basically, sine_normal() should return only float3(-dh_dx, 0, -dh_dz);


Different wave patterns will have different derivatives.

Directional

float3 sine_normal(float3 v_pos, Water_Wave wave)
{
    float k  = wave.spatial_angular_frequency;
    float w  = wave.temporal_angular_frequency;
    float A  = wave.amplitude;

    // This is the height rate of change in the wave's propagation direction.
    float dHeight_dr = A * k * cos(k * get_wave_coord(v_pos, wave) - w * time);

    // Directional wave pattern.
    float dHeight_dx = dHeight_dr * wave.direction.x;
    float dHeight_dz = dHeight_dr * wave.direction.y;

    return float3(-dHeight_dx, 0.0f, -dHeight_dz);
}

Radial

    // Radial waves - derivative with respect to radial distance.
    float2 grad_r = normalize(v_pos.xz);
    dHeight_dx    = dHeight_dr * grad_r.x;
    dHeight_dz    = dHeight_dr * grad_r.y;

Oscillating Radial

    float radius  = length(v_pos);
    float2 grad_r = normalize(v_pos.xz);
    dHeight_dx    = dHeight_dr * grad_r.x * cos(radius);
    dHeight_dz    = dHeight_dr * grad_r.y * cos(radius);

Spiral

    // Spiral wave - polar coordinates (angle and radius).
    float radius  = length(v_pos);
    float angle   = atan2(v_pos.x, v_pos.z);
    float2 grad_r = normalize(v_pos.xz);
    dHeight_dx    = dHeight_dr * grad_r.x * cos(radius - angle);
    dHeight_dz    = dHeight_dr * grad_r.y * cos(radius - angle);

Elliptical

    // Elliptical wave - scaled coordinates.
    float2 scaled_pos  = v_pos.xz / float2(2.0, 1.0);
    float2 grad_scaled = normalize(scaled_pos) / float2(2.0, 1.0);
    dHeight_dx         = dHeight_dr * grad_scaled.x;
    dHeight_dz         = dHeight_dr * grad_scaled.y;

Latticed

    // Lattice wave - derivatives in both x and z directions.
    dHeight_dx = dHeight_dr * 2.0 * cos(2.0 * v_pos.x);
    dHeight_dz = dHeight_dr * 2.0 * cos(2.0 * v_pos.z);

The main resource I can recommend is Acerola’s awesome videos:
https://www.youtube.com/watch?v=PH9q0HNBjT4

Note: There’s a small mistake in the video where the spatial angular frequency is shown to be 2 / wavelength when in reality it’s meant to be 2 * PI / wavelength

Wave simulation can be improved with Fractional Brownian Motion (FBM) and Fast Fourier Transforms (FFT), which is what I’m planning to look at next.

Thanks for reading ^^