Generating Voxel Minecraft Worlds - Ray-Marching and HeightMaps.
Voxels to Create Minecraft-Like Terrains (Boxy World)
With the new Minecraft movie hitting the cinemas - and my love of the game - I thought I'd write a simple tutorial on how you'd create simple minecraft-like worlds that look nice and are interactive.
There are two ways of creating minecraft scenes - either brick by brick (builds the scene up from blocks) - like you do with lego - or converting an existing scene into a 'minecraft-like' one.
This tutorial will focus on 'converting' existing worlds and geometry into a minecraft-like looking ones (voxels and blocks).
We'll 'voxelate' a scene - which means we'll take a smooth surface and pass it through an extra calculation to round the values to the nearest voxel edge (flattening them) and generate blocks.
We'll then mix in some color - so the result looks nice (e.g., lighting and colors).
The tutorial will start of simple - using 2d visualizations and minimal working code - then incrementally keep adding new things (pointing out any pitfalls/artifacts or issues along the way).
Voxelate is like 'pixelate' - but instead of working with an image of pixels in 2d - we do it in 3d and work with the world positions (xyz coordinates). In 2d pixel space it's called pixelation and in 3d world coordinates it's called voxelation.
For the code examples - we'll use a fragment shader for all of the graphical outputs - the examples will use WebGPU and WGSL so you can run/interact with the examples in the web browser without needing to install anything - you can access the source code for the section at the end (and other links for related projects/code).
Fragment Shader (Full Screen Quad)
First thing first, let's setup a vanilla fragment shader - that has the texture coordinates for a full screen quad. If we draw the texture coordinates as the 'red' and 'green' color components - it should look like this:
Full screen quad - draw the texture coordinates as the red and green color in the fragment shader.
The fragment shader for the full screen quad is given below:
@fragment fn main(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { // uv range [0,0] to [1,1] top-left to bottom-right return vec4( uv, 0.0, 1.0 ); }
Simple Height Map Function
Let's define a height map function - so any position in the world, if we pass the
x
and
z
value to the function it will give us the height of the terrain at that location.
We want a hilly terrain with some variation - which we can do by combining a few trigonmetric functions together.
function takes a vec3 but only uses the 'x' and 'z' values in the calculation of the height (does not use the 'y' value). Also the return value is a single 'float' - which is the 'height' of the terrain at the 'x' and 'z' location.
2D Visualization (Smooth) - Top Down
Top down visualization of the 'height map' function to see what sort of pattern it generates - we'll mix between blue and red for the height (blue bottom and red height).
Top down 2d visualization of the height map function (color represents the height value).
@fragment fn main(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { // uv range [0,0] to [1,1] top-left to bottom-right // Normalize UV coordinates to range [-1,1] let nuv: vec2<f32> = -1.0 + 2.0 * uv.xy;
let z = sin( mytimer * 0.1 ); // slice location for the z
// set 'y' - height to 0 as we're going to 'find' the height using the height function let pos = vec3( nuv.x, 0.0, z ) * 20.0;
let y = abs( getGroundHeight( pos ) );
var red = vec3(1.0, 0.0, 0.0); var blue = vec3(0.0, 0.0, 1.0); var color = blue;
if ( abs(nuv.y-y)<0.01 ) { color = red; }
return vec4<f32>( color, 1.0 ); }
Side View as 'Gradient'
We can modify the previous example to draw a gradient color (distance) from the surface instead of using a threshold.
Draw the 2d side view (slice) as a 'gradient' instead of a hard threshold tolerance.
@fragment fn main(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { // uv range [0,0] to [1,1] top-left to bottom-right // Normalize UV coordinates to range [-1,1] let nuv: vec2<f32> = -1.0 + 2.0 * uv.xy;
let z = sin( mytimer * 0.1 ); // slice location for the z
// set 'y' - height to 0 as we're going to 'find' the height using the height function let pos = vec3( nuv.x, 0.0, z ) * 20.0;
let y = abs( getGroundHeight( pos ) );
var red = vec3(1.0, 0.0, 0.0); var blue = vec3(0.0, 0.0, 1.0);
// d is the distance from the surface let d = abs(nuv.y-y); var color = mix( red, blue, pow(d,0.5) );
return vec4<f32>( color, 1.0 ); }
Voxelize 2D (Pixelate)
This is a key step on our journey to developing a `minecraft` scene - as we use the two key functions floor and fract - which will round the numbers and also get us the fractional part of the number.
The 'floor' function will clamp our x, y and z position - while the 'fract' function will be used to detect the 'edges' of each square (so we can mix in color information).
For this example, we'll voxelate the terrain and draw it as a 2d visualization (slice - fixed z). We'll also 'animate' the z-value - so you can see how the result changes for different z-values. We draw the 'fract' value on top - to confirm the edges of each block - and to make sure they're all the same size.
2d visualization of the terrain after voxelization.
@group(0) @binding(2) var <uniform> mytimer : f32;
// Smooth ground height function fn getGroundHeight(pos: vec3<f32>) -> f32 { var v = 0.2 * sin(pos.x * 0.4) + 0.3 * cos(pos.z * 0.45) + 0.15 * sin(pos.z * 1.7) * cos(pos.x * 0.4); return v * 15.0; // amplify the height of the terrain }
@fragment fn main(@location(0) uv: vec2<f32>) -> @location(0) vec4<f32> { // uv range [0,0] to [1,1] top-left to bottom-right // Normalize UV coordinates to range [-1,1] let nuv: vec2<f32> = -1.0 + 2.0 * uv.xy;
let z = sin( mytimer * 0.1 )*3.0; // slice location for the z
// Going to find the surface 'height' at this location let pos = vec3( nuv.x, nuv.y, z ) * 10.0;
// voxelate the input - which will correct our 'x' and 'z' input values // so they're on the correct boundaries let sizeVoxels = 0.8; var floorPos = floor( pos * sizeVoxels ) / sizeVoxels; var fractPos = fract( pos * sizeVoxels );
// We have the height using the voxel x-z input locations let y = getGroundHeight( floorPos );
// This little nugget - we want the 'y' values to also lay on the // correct boundary - so we need to 'floor' the y value (if not they'll be offset wrong) floorPos.y = y; floorPos = floor( floorPos * sizeVoxels ) / sizeVoxels;
// visualize the result var red = vec3(1.0, 0.0, 0.0); var blue = vec3(0.0, 0.0, 1.0); var color = blue;
if ( pos.y > floorPos.y ) { color = red;
// color the edges - check the values are 'square' and they're on // the correct boundaries - using the 'fract' part of the value color *= fractPos.y; color *= fractPos.x; }
return vec4<f32>( color, 1.0 ); }
We can visualize the steps from the 'smooth' height function to the 'voxelated' grid:
Show the steps going from the smooth height function to the voxelated grid.
2D Boxes (Lines)
Use the 'fract' which gives us the fractional part of the number which goes from 0.0 to 1.0 - we use this to draw the edges (boxes). We'll also show the scene in different voxel resolutions (how big or small we want our boxes).
Visualize the voxelized terrain with diffrent size blocks.
If the pixel is on the lower edge (i.e., less than 0.05) we draw it darker (so the edges of the voxels) stand out as lines in the 2d visualization.
Visitor:
Copyright (c) 2002-2025 xbdev.net - All rights reserved.
Designated articles, tutorials and software are the property of their respective owners.