In this lesson we’re going to move the vertices over time. The CodePen below is what we’re creating. Try adjusting the strength, scale and speed controls.

  • strength – controls the blend between the displaced vertex and the original location of the modelled vertex.
  • scale – controls the scale of the noise used for the displacment.
  • speed – controls the rate of change of the displacement.

Now press the JS button.

Notice first the list of tsl imports. Then slide down to the tsl function.

const vNoise = varying(float());
const vPosition = varying(vec3());

Notice vNoise and vPosition are declared as varyings. If you’re new to shaders then a varying will be unfamiliar. Shaders divide the job of rendering a surface into two tasks. First they move the vertices from modelled space to homogeneous clip space. For a vertex to be within the screen window each components xyz location once transformed to homogeneous clip space needs to be in the range -1 to 1. This stage is called the vertex shader.

Having transformed the vertex position the second task is to set the colour for a pixel. This stage is the fragment shader. You can pass data between the vertex and the fragment shader, this type of data in GLSL 1.0 is called a varying, because the value in the fragment shader is an interpolated version of the values of the three vertices of a triangle based on the pixel (fragment) within the triangle. TSL supports varyings if a variable is an instance of a varying.

Recall that we’re trying to reproduce a lava ball, (see the previous lesson). This used a custom GLSL function, turbulence. We need a way of reproducing this. Here’s a remix of the ThreeJS TSL Transpiler . It converts GLSL into TSL. This is a useful and quick way to get the TSL code. pnoise in the GLSL version is a Perlin noise function. The ThreeJS library comes with just such a function mx_noise_float that is an import for the CodePen example above.

Notice the line that includes void main(). This is the original code for the GLSL vertex shader. My attempt at reproducing this is the posFunc Fn function.

const posFunc = Fn(() => {
  vNoise.assign( 
    turbulenceFunc( 
      normalLocal.mul( 0.5 ).add( time )
    )
  );
  ...

First we assign the vNoise varying, using the turbulence function. It takes a modified normalLocal as the input.

...
const b = mx_noise_float(
    positionLocal.mul( noiseScale )
      .add( 
        vec3(2).mul( 
          time.mul( noiseSpeed ) 
        )
      )
    );
    
  const displacement = vNoise.add(b);

  vPosition.assign(positionLocal);
  ...

Than we create the b value, again using mx_noise_float. This takes a modified positionLocal as the input. Now we’re able to create the displacement value. We assign the vPosition value to be positionLocal. This will be used in the next lesson.

...
const pos = positionLocal.add(
    normalLocal.mul(displacement)
  ).toVar();

  vNoise.assign(vNoise.mul(noiseStrength));

  return mix(positionLocal, pos, noiseStrength);
})

pos is set to positionLocal plus normalLocal multiplied by the calculated displacement value. Finally we adjust the value of vNoise scaling it by the noiseStrength uniform and return a blend of the newly calculated position and the original position based again on the noiseStrength uniform.

Now we have the posFunc function we assign this as material.positionNode.

In the next lesson we’ll update the colours used for the rendered object.

If you find these articles useful, perhaps you can buy me a coffee by clicking the button below. It might encourage me to write more.

Signup to my mailing list.