The Object Loader in threejs

The Object Loader in threejs is a loader option that is built into the core of the library itself that can be used to load JSON files that follow the object format. Many other loaders for object formats must be added to threejs by making use of an additional add on file beyond just threejs itself so this alone is one reason why one might be interested in the format. However another nice thing about it is that it is also easy to work with when it comes to creating this kind of json data as just simply calling the toJSON method of the object that I want to convert will create the data in an object format, and then I can just pass that to the JSON.stringify method.

The Object Loader and what to know first

In this post I am writing about the object loader that is built into threejs than can be used to load a single object, or a whole scene in the from of a JSON file. The Object Loader will typically be used to load a whole other external file that will need to be loaded over HTTP, however the parse method of the loader can also be used to create a workable object3d class based object from an object form of the data. In addition the object form of the data can be created from a JSON string from any source by just passing that string to the JSON.parse method. Anyway this is not a post for people that are new to threejs, or javaScript in general. If you are at a loss all ready there are some more things you might want to read about more first.

Know a thing out two about JSON

It would be a good idea to learn at least a thing or two about JSON in general if you have not done so before hand. Mainly the JSON.stringify method that can be used to create a data object created with the toJSON method of the object3d class to a JSON string, as well as the JSON.parse method that can do the inverse of this.

There is the base Loader and Loader Manager classes

The Object Loader extends the Base Loader class, and as such there are various features in that class that a worth checking out. The Object Loader is then just one kind of loader that extends from this common class, as such there are many other options for loading various assets for a threejs project. The object loader is one example of a loader that is built into the core of the threejs library itself, however there is a wide range of other options for loaders that can be added on top of threejs by itself in the form of add on files that can be found in the examples folder of the threejs repository on Github.

Source Code Examples are also on Github

The source code examples that I am writing about in this post can also be found in my test threejs repository on Github. This is the location where I also store the source code exmaples for every other blog post that I have wrote on threejs thus far.

Version Numbers Matter

When I first wrote the blog post I was following my r152 style rules for the code examples. This means that I am using javaScript Module type script tags along with the use of import in the demos for the main threejs module. If you are sticking with an older revision of threejs the demos might still work by doing away with the use of import at the top of the code examples and make sure that you load the plain javaScript form of the library first. However I was very much using the r152 module form at this time so if things are not working for you that would be the first thing to check.

1 - Some Basic Examples of the Object Loader in threejs

To start out with the Object Loader in this section I will be going over a few basic examples. There are several ways to go about getting started with this, but the first thing is to find out how to go about creating JSON data to begin with. So many of the examples here will be focusing on that and the use of the parse method of the Object Loader as a way to parse this JSON data into an object that can then be used in a project. Because we are talking about a plain text format here this is a good format to get started with as just hand coded raw text is one way to get started. However in the long run it would be best to make use of options that can be used to generate the json data by just calling the toJSON method of the source object that I want to convert to this standard.

1.1 - Creating an object from a hard coded JSON string using the parse method

One way to get started with this Object Loader, and also to get familiar with the format, would be to just start hand coding a JSON string by just studied the format of the JSON and then doing it that way. One thing about this is that I will need to have a way to generate UUID strings for the uuid keys of the various objects. For this task there is the generateUUID method in the MathUtils object of threejs which I can call for each ID that I need.

So now I can work out what the main JSON object will be by just using back ticks so I can just go ahead and have End Of Line chars in the JSON string. The format should include a metadata object and I will want to go with version 4.5 as of this writing. The type key for the metadata object should be \”Object\”, and for the generator key I just typed \”Hand Coded\” just to make it clear as to how this data was created. After the meta data object I will then want to add a geometries, and materials arrays that will contain data for all the geometry and materials used for the main object as well as all the children as well. However for this very first demo I am not going to do anything that advanced and just stick with making a single mesh object that make used of just one geometry and material.

When it comes to working out a geometry for the object I can use one of the built in constructor function options for this such as BoxGeometry by just setting that as the value for the type key in place of BufferGeometry. This way I can just pass some key values that will be used for the various parameters when calling the constructor function for that. For the material I am just setting the type key to the MeshNormalMaterial as that is a nice option that does not require anything fancy with light and textures to show depth.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
// ---------- ----------
// IMPORT - threejs and any addons I want to use
// ---------- ----------
import * as THREE from 'three';
// ---------- ----------
// SCENE, CAMERA, RENDERER
// ---------- ----------
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(50, 32 / 24, 0.1, 1000);
const renderer = new THREE.WebGL1Renderer();
renderer.setSize(640, 480, false);
(document.querySelector('#demo') || document.body).appendChild(renderer.domElement);
// ---------- ----------
// JSON STRING IN OBJECT FORMAT
// ---------- ----------
const uuid_geometry = new THREE.MathUtils.generateUUID();
const uuid_material = new THREE.MathUtils.generateUUID();
const uuid_mesh = new THREE.MathUtils.generateUUID();
const mesh_json = `
{
"metadata": {
"version": 4.5,
"type": "Object",
"generator": "Hand Coded"
},
"geometries": [{
"uuid": "` + uuid_geometry + `",
"type": "BoxGeometry",
"width": 1,
"height": 1,
"depth": 1
}
],
"materials": [{
"uuid": "` + uuid_material + `",
"type": "MeshNormalMaterial"
}
],
"object": {
"uuid": "` + uuid_mesh + `",
"type": "Mesh",
"geometry": "` + uuid_geometry + `",
"material": "` + uuid_material + `"
}
}
`
const mesh = new THREE.ObjectLoader().parse( JSON.parse( mesh_json ) );
scene.add(mesh);
// ---------- ----------
// RENDER
// ---------- ----------
camera.position.set(2, 2, 2);
camera.lookAt(0, 0, 0);
renderer.render(scene, camera);

Now we come to the part of the JSON string that will be the Object3D class based object to store this way. For now this object is just a Single Mesh Object, however there is getting into children for this mesh object, and yes the type of the Object can be Scene rather than mesh. However this is still just the very first example of this basic section of the post, so lets refrain from anything fancy for the moment.

1.2 - The toJSON method of Objects

So hand coding JSON files is all find and good, but lets get real here this is not how to go about creating these kinds of assets right? Well no there is of course the toJSON method that can be called off of any object3d class based object to which I want an object in the format that is used for the object loader, and then I can pass that to JSON.stringify. However I have found that the results that this kind of generator creates is not always what I might want, so it still seems like I need to hack over it a little. However depending on how you aim to use the object loader you might not need to bother with that. In this demo I will write about one issue that I have found when it comes to the position of objects and the use of the Object3d.toJSON generator. Well for some it might be an issue, but for others its just the way it should be done, I’ll explain.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
// ---------- ----------
// IMPORT - threejs and any addons I want to use
// ---------- ----------
import * as THREE from 'three';
// ---------- ----------
// SCENE, CAMERA, RENDERER
// ---------- ----------
const camera = new THREE.PerspectiveCamera(50, 32 / 24, 0.1, 1000);
const renderer = new THREE.WebGL1Renderer();
renderer.setSize(640, 480, false);
(document.querySelector('#demo') || document.body).appendChild(renderer.domElement);
// ---------- ----------
// SOURCE SCENE OBJECT
// ---------- ----------
const scene_source = new THREE.Scene();
const geometry = new THREE.BufferGeometry();
geometry.copy( new THREE.BoxGeometry(1,1,1) );
const material = new THREE.MeshPhongMaterial({ color: 0xff0000, emissive: 0xffffff, emissiveIntensity: 0.05 });
const mesh_source = new THREE.Mesh( geometry, material );
scene_source.add(mesh_source);
const light_source = new THREE.PointLight(0xffffff, 1);
light_source.position.set( 1, 3, 2);
light_source.name = 'point_light'
scene_source.add(light_source);
const scene_json = JSON.stringify( scene_source.toJSON() );
// logging the JSON text
console.log( scene_json );
// ---------- ----------
// CREATING A SCENE OBJECT FROM THE JSON TEXT
// ---------- ----------
// creating a new mesh from the JSON text
const scene = new THREE.ObjectLoader().parse( JSON.parse( scene_json ) );
// !!! looks like the 'position' prop of child objects is not getting stored.
// that is a problem
const pl = scene.getObjectByName('point_light');
pl.position.set(1, 3, 2);
// ---------- ----------
// RENDER
// ---------- ----------
camera.position.set(2, 2, 2);
camera.lookAt(0, 0, 0);
renderer.render(scene, camera);

For this demo I am creating a scene object, along with a mesh object, and a point light. I am then positioning the point light away from the mesh object and then calling the toJSON method off of the scene object. Now for the most part this works fine, but with one little problem it would seem that the change to the position attribute that I made is not being stored into the json data. So the way that I had to fix that here when it comes to converting the JSON string back to an object was to get a reference to the point light, and then set the position again.

The good news here is that if I aim to use the object loader to load one object at a time, and then do something else to position things then this way of creating the json data will work just fine. However if I want to create a whole static scene with a bunch of child objects all over the place this presets a problem. The good news though is that this is a limitation of the Object3d.toJSON generator and not the parse method of the Object loader. When I look at the source code for the Object loader it does look like it will process a key value pair for the position. However if the position is key is not there, or if it is there but a matrix key is there then it will not work. Also additional good news is that this is not even a problem at all if one is willing to get into how to work with Matrix4 objects as they can be used to store position, rotation, and scale as one array of values. It would seem that is what is expected rather than the use of verctor3 objects for position and scale, and Euler objects for rotation.

1.3 - A load one JSON file Demo

Well I have wrote a thing or two about the parse method of the Object Loader, and also some ways of how to go about creating the JSON data to begin with. However there Object loader is very much, well a loader, so I should have at east one example where I am loading an external JOSN file then. For this demo then I came up with a JSON file that is a hacked over export of what I worked out in the above example with the scene where I have a bunch of objects that form a very basic scene object. I cleaned up things with the geometry, materials, and well all of the objects really.

However the major change that I did is I removed the matrix keys and in place of those I uses simple position keys rather than bothering with Matrix4 objects. This is very much a basic section after all, and also I often prefer to go with the use of the actually. I can see why things are set up this way though as Matrix4 objects can be used as a way to store position, rotation and scale all in one array of values.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
{
"metadata": {
"version": 4.5,
"type": "Object",
"generator": "Hacked Over Object3D.toJSON Export"
},
"geometries": [{
"uuid": "5f3a2c7e-e238-4f8c-a68b-fe2190c4d7e5",
"type": "BoxGeometry",
"width": 1,
"height": 1,
"depth": 1
}
],
"materials": [{
"uuid": "e12a0bbf-4405-4fb8-a000-f417d24c01c6",
"type": "MeshPhongMaterial",
"color": 16711680,
"emissive": 16777215,
"emissiveIntensity": 0.05,
"specular": 1118481,
"shininess": 30
}
],
"object": {
"uuid": "d9157825-c7c5-4bd9-9c8c-21d21087f883",
"type": "Scene",
"position": [0,0,0],
"children": [{
"uuid": "87313a12-d689-403f-88d2-ad29743e7d2c",
"type": "Mesh",
"position": [0,0,0],
"geometry": "5f3a2c7e-e238-4f8c-a68b-fe2190c4d7e5",
"material": "e12a0bbf-4405-4fb8-a000-f417d24c01c6"
}, {
"uuid": "953c384f-755c-4658-94cd-a5bd246e8df7",
"type": "PointLight",
"position": [3,2,1],
"color": 16777215,
"intensity": 1
}
]
}
}

So now that I have a JSON file to load I can then use the Object Loader to load that JSON file and parse it into a scene object. To do that I just need to create an instance of the Object Loader, and then call the load method off of that instance. When I call the load method I will just need to parse the URL to the json file as the first argument, and then a call back function that I want to run when that file is done loading.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
// ---------- ----------
// IMPORT - threejs and any addons I want to use
// ---------- ----------
import * as THREE from 'three';
// ---------- ----------
// SCENE, CAMERA, RENDERER
// ---------- ----------
const camera = new THREE.PerspectiveCamera(50, 32 / 24, 0.1, 1000);
const renderer = new THREE.WebGL1Renderer();
renderer.setSize(640, 480, false);
(document.querySelector('#demo') || document.body).appendChild(renderer.domElement);
// ---------- ----------
// LOAD, RENDER
// ---------- ----------
const loader = new THREE.ObjectLoader();
const url = '/forpost/threejs-object-loader/s1-3-basic-load-one/scene.json';
const on_done = (obj) => {
camera.position.set(2, 2, 2);
camera.lookAt(0, 0, 0);
renderer.render(obj, camera);
};
loader.load(url, on_done);

There is a whole lot more to write about when it comes this this topic of course as a lot of things will pop up when it comes to things like loading more than one file, and displaying process. Much of that has to do with loaders in general though, and I am thinking that maybe much of that should be handing in a section on the loading manager or maybe even a whole other blog post on that topic.

2 - The tri12 project

One Project that I started that is relevant to the use of this object loader is my tri12 project. As the same suggests this is a collection of JSON file assets where the goal is to create just very simple models that are composed of no more than 12 triangles. This then makes the idea of creating these kinds of models by just hand coding data not to far out of reach then which is one reason why I started doing this. However I have set no limits of any kind when it comes to everything else mind you, so everything that has to do with materials, animations and so forth is still on the table.

2.1 - hard coded string example based on 2.json from tri12-butterfly

For this example I have a hard coded string form of 2.json from my tri12-butterfly project. This json assets has just about it all when it comes to making these kinds of modules as I see it. So there is the position, normal, and uv attribute worked out just fine as I want it for this. However I also have a morph attribute and an animation clip backed into this as well. I also have an image for the texture that I am using for the map option of the basic material that I am using for it the data of it can be baled into the json itself by way of data urls. So then I can parse this string in if I have it as a hard coded string format, and then I just need to set up an animation mixer for the deal, and get this going in a loop.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
// ---------- ----------
// IMPORT - threejs and any addons I want to use
// ---------- ----------
import * as THREE from 'three';
// ---------- ----------
// SCENE, CAMERA, RENDERER
// ---------- ----------
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(50, 32 / 24, 0.1, 1000);
const renderer = new THREE.WebGL1Renderer();
renderer.setSize(640, 480, false);
(document.querySelector('#demo') || document.body).appendChild(renderer.domElement);
// ---------- ----------
// TRI12 butterfly - /json/tri12-butterfly/set1-object/2.json as hard coded string with much that is not needed cut out
// ---------- ----------
const mesh_json = `
{
"metadata": {
"version": 4.5,
"type": "Object",
"generator": "Hand Coded"
},
"geometries": [{
"uuid": "07d25b15-9d12-43f1-8322-5fc6794af50b",
"type": "BufferGeometry",
"data": {
"attributes": {
"position": {
"itemSize": 3,
"type": "Float32Array",
"array": [0, 0, 0, -1, 2, 0, -1, 0.5, 0, 1, 2, 0, 1, 0.5, 0, -0.7, -1, 0, -0.7, -0.1, 0, 0.7, -1, 0, 0.7, -0.1, 0],
"normalized": false
},
"normal": {
"itemSize": 3,
"type": "Float32Array",
"array": [0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1],
"normalized": false
},
"uv": {
"itemSize": 2,
"type": "Float32Array",
"array": [0, 0, 1, 0, 1, 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 0, 1, 1],
"normalized": false
}
},
"index": {
"type": "Uint16Array",
"array": [0, 1, 2, 0, 4, 3, 0, 6, 5, 0, 7, 8]
},
"morphAttributes": {
"position": [{
"itemSize": 3,
"type": "Float32Array",
"array": [0, 0, 0, 0, 0, -0.8, 0, 0, -0.8, 0, 0, -0.8, 0, 0, -0.8, 0, 0, -0.4, 0, 0, -0.4, 0, 0, -0.4, 0, 0, -0.4],
"normalized": false
}
]
},
"morphTargetsRelative": true
}
}
],
"materials": [{
"uuid": "fcac10b3-e11a-49c9-94b8-575042930338",
"type": "MeshBasicMaterial",
"color": 16777215,
"map": "6e281008-fce7-4145-b7e5-afb64421ba51",
"side": 2
}
],
"animations": [{
"name": "flap",
"duration": 1,
"tracks": [{
"name": ".morphTargetInfluences[0]",
"times": [0, 0.25, 0.5, 0.75, 1],
"values": [0, 0.30000001192092896, 0.5, 0.15000000596046448, 0],
"type": "number"
}
],
"uuid": "024eb45e-f782-443b-9fde-522a2897b585",
"blendMode": 2500
}
],
"textures": [{
"uuid": "6e281008-fce7-4145-b7e5-afb64421ba51",
"image": "336e0dfd-4b72-4589-91e6-5c7d26498267"
}
],
"images": [{
"uuid": "336e0dfd-4b72-4589-91e6-5c7d26498267",
"url": ""
}
],
"object": {
"uuid": "ea2f929d-85b7-4167-b5fd-c841702e0d1e",
"type": "Mesh",
"geometry": "07d25b15-9d12-43f1-8322-5fc6794af50b",
"material": "fcac10b3-e11a-49c9-94b8-575042930338",
"animations": ["024eb45e-f782-443b-9fde-522a2897b585"]
}
}
`
const mesh = new THREE.ObjectLoader().parse( JSON.parse( mesh_json ) );
scene.add(mesh);
//-------- ----------
// LOOP
//-------- ----------
camera.position.set(3, 3, 3);
camera.lookAt(0, 0.5, 0);
const state = {
mesh: mesh,
mixer: new THREE.AnimationMixer(mesh)
};
state.action = state.mixer.clipAction( state.mesh.animations[0] );
state.action.play();
let frame = 0;
const frame_max = 30;
const loop = () => {
requestAnimationFrame(loop);
const a1 = frame / frame_max;
state.mixer.setTime(1 * a1);
renderer.render(scene, camera);
frame += 1;
frame %= frame_max;
};
loop();

Conclusion

That will be it for the Object Loader thus far between now and then next time I do a little editing for this post. Although there are a lot of options for loaders not just built in but also with respect to add on loaders such as the DAE Loader just to name one, I do like the built in object loader for many reasons. For one reason it is a plain text format which is nice because although it might be time consuming to do so, the data can be created and edited with any plain text editor. However it is not just that it is a plain text format, it is also because the plain text is JSON rather than some other plain text format like XML.