PROWAREtech

articles » current » javascript » three-js » survival-guide

ThreeJS: A Survival Guide

A tutorial on the basics of THREE.js and a guide to more advanced features including examples, information about mesh materials both reflective and not, lighting, groups, textures and how to load them, displacement and alpha maps, and how to modify a scene during the window scroll event.

This guide requires a basic understanding of JavaScript.

The THREE.js library used in the following examples is version 147 and may be downloaded here: THREEJS.zip. It is incompatible with Internet Explorer 11 (IE11).

This is a guide to the basics for getting started. For the THREE.js full documentation see the THREE.js site.

What is THREE.js

THREE.js is a WebGL plug-in based on JavaScript. It works straight from the browser and is used to power games, advertisements or any other GPU-powered apps. JavaScript is an interpreted language and as a result, THREE.js cannot perform as well as a language like C/C++ which is compiled and linked to a hardware platform.

The Basics of THREE.js

WebGL and THREE.js require at least one of the special <CANVAS> HTML tags to be in the HTML document. The <CANVAS> tag is used to draw graphics, on the fly, via JavaScript scripting.

THREE.js is based around a single scene which is a container with objects like a camera, planes, cubes, spheres, and much more... Every THREE.js application requires a renderer and an function that animates the scene, but lights are optional depending on whether or not the other objects in the scene reflect light (some objects do not reflect light and therefore do not need it to be seen). Every object has a mesh, which is simply a material associated with it.

Three dimensional means having or appearing to have length, breadth, and depth, so all objects on the scene have x, y and z positions as well as x, y and z rotations.

Here are all the steps, one by one, to creating a very basic THREE.js scene using JavaScript:

  1. Create the HTML document with at least one <CANVAS> tag (or create one dynamically).
  2. Add the THREE.js JavaScript library to the HTML document before any code that requires it.
  3. Add a script to the HTML document.
  4. Create the scene container object which has a default background color of black.
  5. Set the scene background color (orange in the case of the following example).
  6. Create one or more objects with a surface material (mesh).
  7. Rotate/position the object(s) as needed for the scene (this can be done during the animation of the scene, or as the window is scrolled, which is covered near the end of this guide).
  8. Create a perspective camera with its field of view, aspect ratio, near and far frustum (how near and far the camera can see).
  9. Set the camera position in space and add it to the scene.
  10. Create the scene renderer passing it the canvas to use.
  11. Set the renderer's device pixel ratio which is available from the browser.
  12. Set the renderer's size which is based on the size of the canvas.
  13. Finally, create the animation function:
    1. Rotate and/or move any objects if so desired in the animation function.
    2. Call renderer.render() with the scene and camera as parameters.
    3. Call the THREE.js requestAnimationFrame() function passing it the animation function as a parameter.
    4. Call the animation function once to get it started.

Here is a full example of the above steps:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Basic Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		// NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);
		var canvas = document.getElementsByTagName("canvas")[0];

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFF8000); // NOTE: color 0xFF8000 is the same as the CSS color #FF8000 which is simply orange in this case

		var cubeSize = 2;
		var cubeGeometry = new THREE.BoxGeometry(cubeSize, cubeSize, cubeSize);
		var materials = {}; // NOTE: use an object to store mesh objects
		materials.white = new THREE.MeshBasicMaterial({ color: 0xFFFFFF });
		materials.silver = new THREE.MeshBasicMaterial({ color: 0xCCCCCC });
		materials.lightgray = new THREE.MeshBasicMaterial({ color: 0x999999 });
		materials.gray = new THREE.MeshBasicMaterial({ color: 0x666666 });
		materials.darkgray = new THREE.MeshBasicMaterial({ color: 0x333333 });
		materials.black = new THREE.MeshBasicMaterial({ color: 0x000000 });
		var cubeSides = [];
		cubeSides.push(materials.white); // NOTE: right side of cube
		cubeSides.push(materials.silver); // NOTE: left side of cube
		cubeSides.push(materials.lightgray); // NOTE: top of cube 
		cubeSides.push(materials.gray); // NOTE: bottom of cube
		cubeSides.push(materials.darkgray); // NOTE: front of the cube
		cubeSides.push(materials.black); // NOTE: back of cube
		var cube = new THREE.Mesh(cubeGeometry, cubeSides);
		scene.add(cube);

		var fieldOfView = 75; // NOTE: this is in degrees and is the vertical field of view
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		// NOTE: "near" and "far" define the frustum which is the region in the 3D space that is going to be projected to the screen
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 5; // NOTE: position the camera back from the scene some so as to "see" the objects
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: required animation function
		function animate() {

			// NOTE: move the cube around
			cube.rotation.x += 0.001;
			cube.rotation.y += 0.002;
			cube.rotation.z += 0.003;

			renderer.render(scene, camera); // NOTE: render the scene
			requestAnimationFrame(animate); // NOTE: call the animate function when the next frame is ready, typically 60 per second
		}
		animate(); // NOTE: call animation function once to get it started

	</script>
</body>
</html>

Reflective Materials and Lights

Besides the THREE.MeshBasicMaterial which is non-reflective, there are several materials available with different reflective qualities. THREE.MeshStandardMaterial is used for any surface. THREE.MeshLambertMaterial is used to simulate surfaces without shine, like a natural, unvarnished wood surface. THREE.MeshPhongMaterial is used to simulate shiny surfaces, like polished stone or wood with varnish. These last three surfaces require a light be added to the scene. THREE.MeshBasicMaterial requires no light.

Lights are required when using materials with reflective qualities. There are several lights all with different qualities. THREE.AmbientLight shines from every direction. THREE.DirectionalLight is light like that of the sun, directional, and its rays are all parallel. It can cast shadows. THREE.PointLight is like a lightbulb, which is also directional and can cast shadows, but for shadows the THREE.DirectionalLight seems best. Lights can emit any color, not just white.

Here is an example using these lights and reflective materials (notice that there can be subtle differences of each light on the spheres):


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Lights and Materials Example</title>
</head>
<body style="background-color:#FFFFFF; font-size:10px;">
	<table style="width:300px;"><tr><th>BasicMaterial</th><th>LambertMaterial</th><th>PhongMaterial</th></tr></table>
	<canvas style="display: inline-block; width: 300px; height: 300px;"></canvas>
	<table style="display:inline-block;"><tr style="height: 100px;"><td>spheres</td></tr><tr style="height: 100px;"><td>cubes</td></tr><tr style="height: 100px;"><td>planes</td></tr></table>
	<div>
		<label><input type="radio" name="light" onclick="setLight(['ambient']);" />AmbientLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['directional']);" />DirectionalLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['point']);" />PointLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['hemisphere']);" />HemisphereLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['spot']);" />SpotLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['ambient','directional']);" checked />AmbientLight+DirectionalLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['ambient','point']);" />AmbientLight+PointLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['directional','hemisphere']);" />DirectionalLight+HemisphereLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['ambient','hemisphere']);" />AmbientLight+HemisphereLight</label><br />
		<label><input type="radio" name="light" onclick="setLight(['ambient','spot']);" />AmbientLight+SpotLight</label><br />
		<label><input type="radio" name="light" onclick="setLight([]);" />None</label><br />
	</div>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		var canvas = document.getElementsByTagName("canvas")[0]; // NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFF8000); // NOTE: color 0xFF8000 is the same as the CSS color #FF8000 which is simply orange in this case

		function createCube(size, material) {
			var geometry = new THREE.BoxGeometry(size, size, size);
			return new THREE.Mesh(geometry, material);
		}
		function createSphere(size, material) {
			var geometry = new THREE.SphereGeometry(size);
			return new THREE.Mesh(geometry, material);
		}
		function createPlane(size, material) {
			var geometry = new THREE.PlaneGeometry(size, size);
			return new THREE.Mesh(geometry, material);
		}
		var materials = {};
		materials.basic = new THREE.MeshBasicMaterial({ color: 0x6699CC });
		materials.lambert = new THREE.MeshLambertMaterial({ color: 0x6699CC });
		materials.phong = new THREE.MeshPhongMaterial({ color: 0x6699CC, shininess: 20 });

		var objects = {};
		objects.basicCube = createCube(1, materials.basic);
		objects.basicCube.position.x = -2;
		scene.add(objects.basicCube);
		objects.lambertCube = createCube(1, materials.lambert);
		scene.add(objects.lambertCube);
		objects.phongCube = createCube(1, materials.phong);
		objects.phongCube.position.x = 2;
		scene.add(objects.phongCube);

		objects.basicSphere = createSphere(.67, materials.basic);
		objects.basicSphere.position.x = -2;
		objects.basicSphere.position.y = 2;
		scene.add(objects.basicSphere);
		objects.lambertSphere = createSphere(.67, materials.lambert);
		objects.lambertSphere.position.y = 2;
		scene.add(objects.lambertSphere);
		objects.phongSphere = createSphere(.67, materials.phong);
		objects.phongSphere.position.x = 2;
		objects.phongSphere.position.y = 2;
		scene.add(objects.phongSphere);

		objects.basicPlane = createPlane(1, materials.basic);
		objects.basicPlane.position.x = -2;
		objects.basicPlane.position.y = -2;
		scene.add(objects.basicPlane);
		objects.lambertPlane = createPlane(1, materials.lambert);
		objects.lambertPlane.position.y = -2;
		scene.add(objects.lambertPlane);
		objects.phongPlane = createPlane(1, materials.phong);
		objects.phongPlane.position.x = 2;
		objects.phongPlane.position.y = -2;
		scene.add(objects.phongPlane);

		for(var object in objects) {
			objects[object].castShadow = objects[object].receiveShadow = true;
		}

		var back = createPlane(1000, new THREE.MeshLambertMaterial({ color: 0xCCCCCC }));
		back.receiveShadow = true;
		back.position.z = -3;
		back.rotation.x = Math.PI * -0.142857143;
		scene.add(back);

		var lights = {};

		lights.ambient = new THREE.AmbientLight(0xffffff, .5);

		lights.hemisphere = new THREE.HemisphereLight(0xffffff, 0x080820, 1);

		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);

		lights.point = new THREE.PointLight(0xffffff, 1);
		lights.point.position.set(10, 25, 15);

		lights.spot = new THREE.SpotLight(0xffffff);
		lights.spot.position.set(10, 25, 15);

		function setLight(lightNames) {
			for(var light in lights) {
				scene.remove(lights[light]);
			}
			for(var i = 0; i < lightNames.length; i++) {
				scene.add(lights[lightNames[i]]);
			}
		}
		setLight(['ambient','directional']);

		var fieldOfView = 10; // NOTE: this is in degrees and is the vertical field of view
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 40; // NOTE: position the camera back from the scene some so as to "see" the objects
		scene.add(camera);

		var setShadow = function (light) {
			light.castShadow = true;
			light.shadow.mapSize.width = 1024;
			light.shadow.mapSize.height = 1024;
			light.shadow.camera.near = near;
			light.shadow.camera.far = far;
		};
		setShadow(lights.directional);
		setShadow(lights.point);
		setShadow(lights.spot);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.shadowMap.enabled = true;
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: animation function which rotates the cube
		function animate() {
			for(var object in objects) {
				objects[object].rotation.x += 0.006;
				objects[object].rotation.y += 0.009;
				objects[object].rotation.z += 0.012;
			}
			renderer.render(scene, camera); // NOTE: render the scene
			requestAnimationFrame(animate); // NOTE: call the animate function when the next frame is ready, typically 60 frames-per-second
		}
		animate(); // NOTE: call animation function once to get it started

	</script>
</body>
</html>

Object Groups

Groups are designed to make it simple to create a single object out of several basic objects by grouping them together into one. They may still be controlled individually but they can be rotated and positioned as a whole by setting the group's rotation and position properties. Groups make it possible to create a city, for example, with all the individual objects contained within a city group including people, cars, houses, etc. which are also groups themselves.

In this example, a group is used to add all the objects together so that they may be rotated in space as one unit:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Grouping Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		var canvas = document.getElementsByTagName("canvas")[0]; // NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFF8000); // NOTE: color 0xFF8000 is the same as the CSS color #FF8000 which is simply orange in this case

		var group = new THREE.Group(); // NOTE: create a group to add objects to
		
		function createCube(size, material) {
			var geometry = new THREE.BoxGeometry(size, size, size);
			return new THREE.Mesh(geometry, material);
		}
		function createSphere(size, material) {
			var geometry = new THREE.SphereGeometry(size);
			return new THREE.Mesh(geometry, material);
		}
		function createTorus(size, material) {
			var geometry = new THREE.TorusGeometry(size, size / 2, 16, 100);
			return new THREE.Mesh(geometry, material);
		}
		var materials = {};
		materials.lambert = new THREE.MeshLambertMaterial({ color: 0x6699CC });
		materials.phong = new THREE.MeshPhongMaterial({ color: 0x6699CC });

		var objects = {};
		objects.lambertCube = createCube(1, materials.lambert);
		objects.lambertCube.position.x = -1;
		group.add(objects.lambertCube);
		objects.phongCube = createCube(1, materials.phong);
		objects.phongCube.position.x = 1;
		group.add(objects.phongCube);

		objects.lambertSphere = createSphere(.67, materials.lambert);
		objects.lambertSphere.position.x = -1;
		objects.lambertSphere.position.y = 2;
		group.add(objects.lambertSphere);
		objects.phongSphere = createSphere(.67, materials.phong);
		objects.phongSphere.position.x = 1;
		objects.phongSphere.position.y = 2;
		group.add(objects.phongSphere);

		objects.lambertPlane = createTorus(.5, materials.lambert);
		objects.lambertPlane.position.x = -1;
		objects.lambertPlane.position.y = -2;
		group.add(objects.lambertPlane);
		objects.phongPlane = createTorus(.5, materials.phong);
		objects.phongPlane.position.x = 1;
		objects.phongPlane.position.y = -2;
		group.add(objects.phongPlane);

		scene.add(group);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);

		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);

		function setLight(lightNames) {
			for(var light in lights) {
				scene.remove(lights[light]);
			}
			for(var i = 0; i < lightNames.length; i++) {
				scene.add(lights[lightNames[i]]);
			}
		}
		setLight(['ambient','directional']);

		var fieldOfView = 10; // NOTE: this is in degrees and is the vertical field of view
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 40; // NOTE: position the camera back from the scene some so as to "see" the objects
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: animation function which rotates the cube
		function animate() {

			// NOTE: rotate each object
			for(var object in objects) {
				objects[object].rotation.x += 0.006;
				objects[object].rotation.y += 0.009;
				objects[object].rotation.z += 0.012;
			}

			// NOTE: rotate the whole group of objects
			group.rotation.x += 0.01;
			group.rotation.y += 0.02;

			renderer.render(scene, camera); // NOTE: render the scene
			requestAnimationFrame(animate); // NOTE: call the animate function when the next frame is ready, typically 60 frames-per-second
		}
		animate(); // NOTE: call animation function once to get it started

	</script>
</body>
</html>

Using Textures and the TextureLoader

The THREE.TextureLoader is used for loading textures which are then set to objects.

Settings the Scene Background

In this example, the scene background (scene.background) is set using the texture loader.


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Background Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		// NOTE: this is an image in base64 format; a url to an image can be used such as:
		// var stars = "/images/stars.jpg";
		var stars = "";

		// NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);
		var canvas = document.getElementsByTagName("canvas")[0];

		// NOTE: create the texture loader
		var loader = new THREE.TextureLoader();

		var scene = new THREE.Scene();
		// NOTE: set the background to a JPEG image of stars
		// could also be something like scene.background = loader.load("/images/stars.jpg");
		scene.background = loader.load(stars);

		var cubeSize = 2;
		var cubeGeometry = new THREE.BoxGeometry(cubeSize, cubeSize, cubeSize);
		var materials = {}; // NOTE: use an object to store mesh objects
		materials.blue = new THREE.MeshPhongMaterial({ color: 0x6699CC });
		var cube = new THREE.Mesh(cubeGeometry, materials.blue);
		scene.add(cube);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);
		scene.add(lights.ambient);
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 75; // NOTE: this is in degrees and is the vertical field of view
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		// NOTE: "near" and "far" define the frustum which is the region in the 3D space that is going to be projected to the screen
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 5; // NOTE: position the camera back from the scene some so as to "see" the objects
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: required animation function
		function animate() {

			// NOTE: move the cube around
			cube.rotation.x += 0.001;
			cube.rotation.y += 0.002;
			cube.rotation.z += 0.003;

			renderer.render(scene, camera); // NOTE: render the scene
			requestAnimationFrame(animate); // NOTE: call the animate function when the next frame is ready, typically 60 per second
		}
		animate(); // NOTE: call animation function once to get it started

	</script>
</body>
</html>
Adding Textures to Objects

Textures can also be applied to object surfaces by "mapping" them to the mesh material.

In this example, a texture is mapped to the sides of the cube:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Add Texture to Objects Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		// NOTE: these images are in base64 format; a url to an image can be used such as:
		// var stars = "/images/stars.jpg";
		var stars = "";
		var starDust = "";

		var canvas = document.getElementsByTagName("canvas")[0];

		// NOTE: create the texture loader
		var loader = new THREE.TextureLoader();

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFFFFFF); // NOTE: make background white

		var cubeSize = 2;
		var cubeGeometry = new THREE.BoxGeometry(cubeSize, cubeSize, cubeSize);
		var materials = {}; // NOTE: use an object to store mesh objects
		materials.stars = new THREE.MeshPhongMaterial({ map: loader.load(stars) }); // NOTE: load the stars image and map it
		materials.starDust = new THREE.MeshPhongMaterial({ map: loader.load(starDust) }); // NOTE: load the star dust image and map it
		var sides = [];
		sides.push(materials.stars);
		sides.push(materials.stars);
		sides.push(materials.stars);
		sides.push(materials.stars);
		sides.push(materials.starDust); // NOTE: front of cube
		sides.push(materials.starDust); // NOTE: back of cube
		var cube = new THREE.Mesh(cubeGeometry, sides);
		scene.add(cube);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);
		scene.add(lights.ambient);
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 75;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 5;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		function animate() {

			// NOTE: move the cube around
			cube.rotation.x += 0.001;
			cube.rotation.y += 0.002;
			cube.rotation.z += 0.003;

			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	</script>
</body>
</html>

Displacement Maps

A texture can have a displacement map. A displacement map is a grayscale image that can be used to alter geometry. The lighter areas have greater displacement. The darker areas less so. Each pixel's value in the displacement map image is used to change the position of the vertices of the mesh.

These are the displacement maps use in the following example:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Two Planes with Displacement Map Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		// NOTE: these images are in base64 format, a url to an image can be used such as:
		// var stars = "/images/stars.jpg";
		var stars = "";
		var starDust = "";
		// NOTE: these displacement maps are just the original image converted to grayscale, the lighter areas will displace more than the darker areas
		var starsDisplacementMap = "";
		var starDustDisplacementMap = "";

		var canvas = document.getElementsByTagName("canvas")[0];

		// NOTE: create the texture loader
		var loader = new THREE.TextureLoader();

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFFFFFF); // NOTE: make background white

		var planeSize = 4;

		var materials = {}; // NOTE: use an object to store mesh objects
		materials.stars = new THREE.MeshPhongMaterial({
			map: loader.load(stars),                               // NOTE: load the stars image
			displacementMap: loader.load(starsDisplacementMap),    // NOTE: load the displacementMap
			displacementScale: 1,                                  // NOTE: this changes the amount of the displacement, starts at 0
			side: THREE.DoubleSide                                 // NOTE: make both sides visible
		});
		materials.starDust = new THREE.MeshPhongMaterial({
			map: loader.load(starDust),                            // NOTE: load the star dust image
			displacementMap: loader.load(starDustDisplacementMap), // NOTE: load the displacementMap
			displacementScale: 1,                                  // NOTE: this changes the amount of the displacement, starts at 0
			side: THREE.DoubleSide                                 // NOTE: make both sides visible
		});
		materials.plane = new THREE.MeshPhongMaterial({ color: 0x6699CC });

		var planeBufferGeometry = new THREE.PlaneBufferGeometry(planeSize, planeSize, 100, 100); // NOTE: create a plane buffer, 100 vertices

		var planeBuffer1 = new THREE.Mesh(planeBufferGeometry, materials.stars);
		planeBuffer1.rotation.x = -Math.PI / 2.5;
		scene.add(planeBuffer1);

		var planeBuffer2 = new THREE.Mesh(planeBufferGeometry, materials.starDust);
		planeBuffer2.rotation.x = -Math.PI / 2.5;
		planeBuffer2.position.y = -1;
		scene.add(planeBuffer2);

		var lights = {};
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 75;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 5;
		camera.position.y = -1;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		function animate() {

			// NOTE: move the objects around
			planeBuffer1.rotation.z += 0.003;
			planeBuffer2.rotation.z = planeBuffer1.rotation.z;

			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	</script>
</body>
</html>

Alpha Maps

A texture can have an alpha map, which is a grayscale texture that controls the opacity across the surface. Black is fully transparent while white is fully opaque.

This is the alpha map used in the following example:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Plane with Alpha Map and Displacement Map Example</title>
</head>
<body style="background-color:#FFFFFF;">
	<canvas style="display: block; width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		// NOTE: these are images in base64 format, a url to an image can be used such as:
		// var stars = "/images/stars.jpg";
		var stars = "";
		var starsDisplacementMap = "";
		var alphaMap = "";

		var canvas = document.getElementsByTagName("canvas")[0];

		// NOTE: create the texture loader
		var loader = new THREE.TextureLoader();

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0x6699CC); // NOTE: make the background blue

		var planeSize = 4;

		var materials = {}; // NOTE: use an object to store mesh objects
		materials.stars = new THREE.MeshPhongMaterial({
			map: loader.load(stars),                            // NOTE: load the stars image
			displacementMap: loader.load(starsDisplacementMap), // NOTE: load the displacementMap (not required for alpha maps)
			displacementScale: 1,                               // NOTE: this changes the amount of the displacement, starts at 0 (not required for alpha maps)
			alphaMap: loader.load(alphaMap),                    // NOTE: load the alphaMap
			transparent: true,                                  // NOTE: must specify this
			side: THREE.DoubleSide                              // NOTE: make both sides visible
		});

		var planeBufferGeometry = new THREE.PlaneBufferGeometry(planeSize, planeSize, 100, 100); // NOTE: create a plane buffer, 100 vertices

		var planeBuffer = new THREE.Mesh(planeBufferGeometry, materials.stars);
		planeBuffer.rotation.x = -Math.PI / 2.5;
		scene.add(planeBuffer);

		var lights = {};
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 75;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 5;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		function animate() {

			// NOTE: move the object around
			planeBuffer.rotation.z += 0.003;

			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	</script>
</body>
</html>

Moving Objects as the User Scrolls

To move an object around as the user scrolls, the parent element of the canvas must have a CSS height value higher than the browser window, such as 400vh which is 4 times the window height, and the canvas must have a position of sticky or fixed with a value set for its CSS top property, usually set to 0.

A basic example for moving an object during the window scroll event is adding a constant to the rotation of the mesh object as in this example:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Basic Scroll Example</title>
</head>
<body style="margin: 0; padding: 0;background-color:#FFFFFF; min-height: 400vh;">
	<canvas style="display: block; position: fixed; top: 0; left: calc(50% - 150px); width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		var canvas = document.getElementsByTagName("canvas")[0]; // NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFFFFFF); // NOTE: make background white like the HTML page

		function createCube(size, material) {
			var geometry = new THREE.BoxGeometry(size, size, size);
			return new THREE.Mesh(geometry, material);
		}
		var materials = {}; // NOTE: use an object to store mesh objects (object materials)
		materials.lambert = new THREE.MeshLambertMaterial({ color: 0x6699CC });

		var objects = {};
		objects.lambertCube = createCube(3, materials.lambert);
		scene.add(objects.lambertCube);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);
		scene.add(lights.ambient);
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 10;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 40;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: animation function which only updates the canvas and does not modify any objects
		function animate() {
			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	window.addEventListener("scroll", function () {
		
		// NOTE: rotate the objects in the scene a fixed amount and forward direction only
		for(var object in objects) {
			objects[object].rotation.x += 0.009;
			objects[object].rotation.y += 0.018;
			objects[object].rotation.z += 0.036;
		}

	});

	</script>
</body>
</html>
Using Scroll Position to Set Objects

A better example uses the position of scroll to set the rotation as in this example:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Rotate Element on Scroll Example</title>
</head>
<body style="margin: 0; padding: 0;background-color:#FFFFFF; min-height: 400vh;">
	<canvas style="display: block; position: fixed; top: 0; left: calc(50% - 150px); width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		var canvas = document.getElementsByTagName("canvas")[0]; // NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFFFFFF); // NOTE: make background white like the HTML page

		function createCube(size, material) {
			var geometry = new THREE.BoxGeometry(size, size, size);
			return new THREE.Mesh(geometry, material);
		}
		var materials = {}; // NOTE: use an object to store mesh objects (object materials)
		materials.lambert = new THREE.MeshLambertMaterial({ color: 0x6699CC });

		var objects = {};
		objects.lambertCube = createCube(3, materials.lambert);
		scene.add(objects.lambertCube);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);
		scene.add(lights.ambient);
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 10;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		camera.position.z = 40;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: animation function which only updates the canvas and does not modify any objects
		function animate() {
			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	window.addEventListener("scroll", function () {

		// NOTE: get the position of the scroll value (0 < position < 1)
		var position = window.scrollY / (canvas.parentElement.offsetHeight - window.innerHeight);

		// NOTE: rotate the objects in the scene
		for(var object in objects) {
			objects[object].rotation.x = position * 2;
			objects[object].rotation.y = position * 4;
			objects[object].rotation.z = position * 8;
		}

	});

	</script>
</body>
</html>
Using Scroll Position to Set Camera Position

In this example the camera is moved forward as the user scrolls:


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<meta http-equiv="X-UA-Compatible" content="IE=edge">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>THREE.js - Zoom on Scroll Example</title>
</head>
<body style="margin: 0; padding: 0;background-color:#FFFFFF; min-height: 400vh;">
	<canvas style="display: block; position: fixed; top: 0; left: calc(50% - 150px); width: 300px; height: 300px;"></canvas>
	<script src="/js/three.min.js"></script>
	<script type="text/javascript">

		var canvas = document.getElementsByTagName("canvas")[0]; // NOTE: get the canvas element - this can be created with document.createElement("canvas") and added with document.body.appendChild(createdCanvasObject);

		var scene = new THREE.Scene();
		scene.background = new THREE.Color(0xFFFFFF); // NOTE: make background white like the HTML page

		function createCube(size, material) {
			var geometry = new THREE.BoxGeometry(size, size, size);
			return new THREE.Mesh(geometry, material);
		}
		var materials = {}; // NOTE: use an object to store mesh objects (object materials)
		materials.lambert = new THREE.MeshLambertMaterial({ color: 0x6699CC });

		var objects = {};
		var cubeSize = 3;
		objects.lambertCube = createCube(cubeSize, materials.lambert);
		scene.add(objects.lambertCube);

		var lights = {};
		lights.ambient = new THREE.AmbientLight(0xffffff, .5);
		scene.add(lights.ambient);
		lights.directional = new THREE.DirectionalLight(0xffffff, 1);
		lights.directional.position.set(10, 25, 15);
		scene.add(lights.directional);

		var fieldOfView = 10;
		var aspectRatio = canvas.offsetWidth / canvas.offsetHeight;
		var near = 1;
		var far = 10000;
		var camera = new THREE.PerspectiveCamera(fieldOfView, aspectRatio, near, far);
		var cameraPositionZ = 50;
		camera.position.z = cameraPositionZ;
		scene.add(camera);

		var renderer = new THREE.WebGLRenderer({ canvas: canvas, antialias: true });
		renderer.setPixelRatio(window.devicePixelRatio);
		renderer.setSize(canvas.offsetWidth, canvas.offsetHeight);

		// NOTE: animation function which only updates the canvas and does not modify any objects
		function animate() {
			renderer.render(scene, camera);
			requestAnimationFrame(animate);
		}
		animate();

	window.addEventListener("scroll", function () {

		// NOTE: get the position of the scroll value (0 < position < 1)
		var position = window.scrollY / (canvas.parentElement.offsetHeight - window.innerHeight);

		// NOTE: rotate the objects in the scene
		for(var object in objects) {
			objects[object].rotation.x = position * (Math.PI / 2);
			objects[object].rotation.y = position * Math.PI;
			objects[object].rotation.z = position * (Math.PI * 1.5);
		}

		// NOTE: this will zoom the camera into the scene
		// cubeSize is added to make the camera not go beyond the cube as it zooms
		camera.position.z = cameraPositionZ - position * cameraPositionZ + cubeSize;
	});

	</script>
</body>
</html>

Other Things to Consider

This site uses cookies. Cookies are simple text files stored on the user's computer. They are used for adding features and security to this site. Read the privacy policy.
CLOSE