Skip to content

Voxel.Renderer

JollyPixel Voxel Engine and Renderer

📌 About ​

Chunked voxel engine and renderer for Three.js and the JollyPixel engine (ECS). Add VoxelRenderer to any scene and you get multi-layer voxel worlds with tileset textures, face culling, block transforms, JSON save/load, and optional Rapier3D physics.

💡 Features ​

  • Chunked world (default 16³) — only dirty chunks are rebuilt each frame, the rest are left alone
  • Named layers composited top-down; decorative layers override base terrain without Z-fighting
  • Toggle visibility, reorder, add/remove layers, and move them in world space
  • Face culling between adjacent solid voxels to keep triangle counts low
  • Many built-in block shapes (cube, slabs, ramp, corners, pole, stairs) and a BlockShape interface for custom geometry
  • Per-block transforms via a packed byte — 90° Y rotations and X/Z flips without duplicating definitions
  • Multiple tilesets at different resolutions; tiles referenced by { tilesetId, col, row }
  • Per-face texture overrides on any block definition
  • "lambert" (default) or "standard" (PBR) material modes
  • Configurable alphaTest for foliage and sprite-style cutout blocks
  • save() / load() round-trips the full world state as plain JSON
  • TiledConverter to import Tiled .tmj maps in "stacked" or "flat" layer modes
  • Optional Rapier3D physics with "box" or "trimesh" colliders rebuilt per dirty chunk; zero extra dependency if omitted
  • Compatible with JollyPixel engine logger

NOTE

The implementation and optimization are probably far from perfect. Feel free to open a PR to help us.

💃 Getting Started ​

This package is available in the Node Package Repository and can be easily installed with npm or yarn.

bash
$ npm i @jolly-pixel/voxel.renderer
# or
$ yarn add @jolly-pixel/voxel.renderer

👀 Usage example ​

Basic — place voxels manually ​

ts
const blocks: BlockDefinition[] = [
  {
    id: 1,
    name: "Dirt",
    shapeId: "cube",
    collidable: true,
    faceTextures: {
      [Face.PosY]: {
        tilesetId: "default",
        col: 0,
        row: 2
      },
      [Face.NegX]: {
        tilesetId: "default",
        col: 0,
        row: 1
      },
      [Face.NegZ]: {
        tilesetId: "default",
        col: 0,
        row: 1
      },
      [Face.PosX]: {
        tilesetId: "default",
        col: 0,
        row: 1
      },
      [Face.PosZ]: {
        tilesetId: "default",
        col: 0,
        row: 1
      }
    },
    defaultTexture: {
      tilesetId: "default",
      col: 2,
      row: 0
    }
  }
];

const voxelMap = world.createActor("map")
  .addComponentAndGet(VoxelRenderer, {
    chunkSize: 16,
    layers: ["Ground"],
    blocks
  });

voxelMap.loadTileset({
  id: "default",
  src: "tileset/UV_cube.png",
  tileSize: 32
}).catch(console.error);

// Place a flat 8×8 ground plane
for (let x = 0; x < 8; x++) {
  for (let z = 0; z < 8; z++) {
    voxelMap.setVoxel("Ground", {
      position: { x, y: 0, z },
      blockId: 1
    });
  }
}

Tiled import — convert a .tmj map ​

ts
import { loadJSON } from "@jolly-pixel/engine";
import {
  VoxelRenderer,
  TiledConverter,
  type TiledMap
} from "@jolly-pixel/voxel.renderer";

// No blocks or layers needed here — load() restores them from the JSON snapshot
const voxelMap = world.createActor("map")
  .addComponentAndGet(VoxelRenderer, { alphaTest: 0.1, material: "lambert" });

const tiledMap = await loadJSON<TiledMap>("tilemap/map.tmj");

const worldJson = new TiledConverter().convert(tiledMap, {
  // Map Tiled .tsx source references to the PNG files served by your dev server
  resolveTilesetSrc: (src) => "tilemap/" + src.replace(/\.tsx$/, ".png"),
  layerMode: "stacked"
});

voxelMap.load(worldJson).catch(console.error);

await loadRuntime(runtime);

Rapier3D physics ​

ts
import Rapier from "@dimforge/rapier3d-compat";

await Rapier.init();
const rapierWorld = new Rapier.World({
  x: 0,
  y: -9.81,
  z: 0
});

// Step physics once per fixed tick, before the scene update
world.on("beforeFixedUpdate", () => rapierWorld.step());

const voxelMap = world.createActor("map")
  .addComponentAndGet(VoxelRenderer, {
    chunkSize: 16,
    layers: ["Ground"],
    blocks,
    rapier: { api: Rapier, world: rapierWorld }
  });

🚀 Running the examples ​

Four interactive examples live in the examples/ directory and are served by Vite. Start the dev server from the package root:

bash
npm run dev -w @jolly-pixel/voxel.renderer

Then open one of these URLs in your browser:

URLScriptWhat it shows
http://localhost:5173/demo-physics.tsA 32×32 voxel terrain with a raised platform and a Rapier3D physics sphere you can roll around with arrow keys
http://localhost:5173/tileset.htmldemo-tileset.tsEvery tile in Tileset001.png laid out as UV-mapped quads with col/row labels, plus a rotating textured cube
http://localhost:5173/shapes.htmldemo-shapes.tsAll 19 built-in block shapes rendered as coloured meshes with a wireframe overlay and labelled name
http://localhost:5173/tiled.htmldemo-tiled.tsA multi-layer Tiled .tmj map imported via TiledConverter in "stacked" mode with WASD camera navigation

All four examples use OrbitControls (left drag: rotate, right drag: pan, scroll: zoom) except the physics demo which uses Camera3DControls (WASD + mouse).

📚 API ​

  • VoxelRenderer - Main ActorComponent — options, voxel placement, tileset loading, save/load.
  • World - VoxelWorld, VoxelLayer, VoxelChunk, and related types.
  • Blocks - BlockDefinition, BlockShape, BlockRegistry, BlockShapeRegistry, and Face.
  • Tileset - TilesetManager, TilesetDefinition, TileRef, UV regions.
  • Serialization - VoxelSerializer and JSON snapshot types.
  • Collision - Rapier3D integration, VoxelColliderBuilder, and physics interfaces.
  • Built-In Shapes - All built-in block shapes and custom shape authoring.
  • TiledConverter - Converting Tiled .tmj exports to VoxelWorldJSON.

🔥 Troubleshooting ​

If something isn't working as expected, enable verbose logging to get detailed runtime output:

ts
// Enable debug logs for the entire runtime
const { world } = runtime;
world.logger.setLevel("debug");
world.logger.enableNamespace("*");

Alternatively, pass a custom Logger instance to VoxelRenderer:

ts
import { Systems } from "@jolly-pixel/engine";
import { VoxelRenderer } from "@jolly-pixel/voxel.renderer";

const vr = new VoxelRenderer({
  logger: new Systems.Logger({
    level: "trace",
    namespaces: ["*"]
  })
});

Quick tips

  • Tileset missing: verify the src path and ensure the image is being served (check browser Network tab and CORS).
  • Cutout/transparent textures look wrong: increase or decrease alphaTest (for example alphaTest: 0.1) to tune cutout thresholds.
  • Physics not working: make sure Rapier is initialized (await Rapier.init()) and you pass a Rapier World via the rapier option.
  • Chunks not updating or faces missing: face culling hides faces between adjacent solid voxels; confirm neighboring voxels are placed correctly.

Reporting issues

  • When opening an issue, include package and runtime versions, reproduction steps, and enable debug logs (see above). A minimal repro or screenshot speeds up investigation.

Contributors guide ​

If you are a developer looking to contribute to the project, you must first read the CONTRIBUTING guide.

Once you have finished your development, check that the tests (and linter) are still good by running the following script:

bash
$ npm run test
$ npm run lint

CAUTION

In case you introduce a new feature or fix a bug, make sure to include tests for it as well.

License ​

MIT