PROWAREtech
ThreeJS: A Survival Guide
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:
- Create the HTML document with at least one
<CANVAS>
tag (or create one dynamically). - Add the THREE.js JavaScript library to the HTML document before any code that requires it.
- Add a script to the HTML document.
- Create the scene container object which has a default background color of black.
- Set the scene background color (orange in the case of the following example).
- Create one or more objects with a surface material (mesh).
- 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).
- Create a perspective camera with its field of view, aspect ratio, near and far frustum (how near and far the camera can see).
- Set the camera position in space and add it to the scene.
- Create the scene renderer passing it the canvas to use.
- Set the renderer's device pixel ratio which is available from the browser.
- Set the renderer's size which is based on the size of the canvas.
- Finally, create the animation function:
- Rotate and/or move any objects if so desired in the animation function.
- Call
renderer.render()
with the scene and camera as parameters. - Call the THREE.js
requestAnimationFrame()
function passing it the animation function as a parameter. - 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
- Because almost everything is in radians,
Math.PI
is a very close friend. Use it when rotating objects. - How to increase performance.
- How to create, cast and receive shadows.
- How to make the background transparent.
- How to set the background image to cover the scene without stretching it.
- How to create transparency using Portable Network Graphics (PNG) images.
- How to make a mesh object appear to follow the mouse cursor by moving the camera.
- How to make mesh objects appear to watch the moving mouse cursor.
- How to set camera z-position based on object's size.
- Raytracer: How to access individual objects on the canvas.
-
If the canvas is to be resized when the browser window is resized then the canvas CSS width and height properties must be set and then the camera aspect ratio must be updated, as done in this code snippet:
window.addEventListener("resize", function () { var width = window.innerWidth, height = window.innerHeight; canvas.style.width = width + "px"; // NOTE: set the canvas width canvas.style.height = height + "px"; // NOTE: set the canvas height camera.aspect = width / height; camera.updateProjectionMatrix(); renderer.setPixelRatio(window.devicePixelRatio); renderer.setSize(width, height); });