TDME2 1.9.121
GLTFReader.cpp
Go to the documentation of this file.
2
3#include <map>
4#include <set>
5#include <string>
6#include <vector>
7
8#define TINYGLTF_IMPLEMENTATION
9#define STB_IMAGE_IMPLEMENTATION
10#define STB_IMAGE_WRITE_IMPLEMENTATION
11
12#include <ext/libpng/png.h>
13#include <ext/tinygltf/tiny_gltf.h>
14
15#include <tdme/tdme.h>
34#include <tdme/math/Matrix4x4.h>
36#include <tdme/math/Vector3.h>
46
47using std::map;
48using std::set;
49using std::string;
50using std::to_string;
51using std::vector;
52
54
85
86Model* GLTFReader::read(const string& pathName, const string& fileName)
87{
88 // load model
89 vector<uint8_t> glftBinaryData;
90 FileSystem::getInstance()->getContent(pathName, fileName, glftBinaryData);
91 // parse model
92 string err;
93 string warn;
94 tinygltf::Model gltfModel;
95 tinygltf::TinyGLTF gltfLoader;
96 auto ret = gltfLoader.LoadBinaryFromMemory(&gltfModel, &err, &warn, glftBinaryData.data(), glftBinaryData.size());
97 if (warn.empty() == false) Console::println("GLTFReader::read(): warnings: " + warn);
98 if (err.empty() == false) Console::println("GLTFReader::read(): errors: " + err);
99 if (ret == false){
100 Console::println("GLTFReader::read(): Failed to load model: " + pathName + "/" + fileName);
101 return nullptr;
102 }
103
104 // create model
105 auto model = new Model(
106 fileName,
107 fileName,
108 UpVector::Y_UP,
109 RotationOrder::ZYX,
110 nullptr,
112 );
113 model->setShaderModel(ShaderModel::PBR);
114
115 // parse nodes aka scene
116 for (auto& gltfScene: gltfModel.scenes) {
117 for (auto gltfNodeIdx: gltfScene.nodes) {
118 auto& glTfNode = gltfModel.nodes[gltfNodeIdx];
119 auto node = parseNode(pathName, gltfModel, gltfNodeIdx, model, nullptr);
120 model->getNodes()[node->getId()] = node;
121 if (model->getSubNodes().find(node->getId()) != model->getSubNodes().end()) {
122 Console::println("GLTFReader::read(): node already exists: " + node->getId());
123 }
124 model->getSubNodes()[node->getId()] = node;
125 if (glTfNode.children.empty() == false) parseNodeChildren(pathName, gltfModel, glTfNode.children, node);
126 }
127 }
128
129 // animations
130 auto maxFrames = 0;
131 {
132 set<string> animationNodes;
133 map<string, vector<Matrix4x4>> animationScaleMatrices;
134 map<string, vector<Matrix4x4>> animationRotationMatrices;
135 map<string, vector<Matrix4x4>> animationTranslationMatrices;
136 for (auto& gltfAnimation: gltfModel.animations) {
137 auto frames = 0;
138 for (auto& gltfChannel: gltfAnimation.channels) {
139 auto node = model->getNodeById(gltfModel.nodes[gltfChannel.target_node].name);
140 animationNodes.insert(node->getId());
141 auto& gltfSample = gltfAnimation.samplers[gltfChannel.sampler];
142 // animation input: key frame time stamps
143 auto& animationInputAccessor = gltfModel.accessors[gltfSample.input];
144 auto& animationInputBufferView = gltfModel.bufferViews[animationInputAccessor.bufferView];
145 auto& animationInputBuffer = gltfModel.buffers[animationInputBufferView.buffer];
146 auto animationInputBufferData = (const float*)(animationInputBuffer.data.data() + animationInputAccessor.byteOffset + animationInputBufferView.byteOffset);
147 if (animationInputAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
148 Console::println("GLTFReader::read(): " + node->getId() + ": animation: " + gltfAnimation.name + ": Invalid input attributes component: " + getComponentTypeString(animationInputAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(animationInputAccessor.componentType)));
149 continue;
150 }
151 auto channelFrames = static_cast<int32_t>(Math::ceil(animationInputBufferData[animationInputAccessor.count - 1] * 30.0f));
152 // animation output: translation, rotation, scale
153 auto& animationOutputAccessor = gltfModel.accessors[gltfSample.output];
154 auto& animationOutputBufferView = gltfModel.bufferViews[animationOutputAccessor.bufferView];
155 auto& animationOutputBuffer = gltfModel.buffers[animationOutputBufferView.buffer];
156 auto animationOutputBufferData = (const float*)(animationOutputBuffer.data.data() + animationOutputAccessor.byteOffset + animationOutputBufferView.byteOffset);
157 if (animationOutputAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
158 Console::println("GLTFReader::read(): " + node->getId() + ": animation: " + gltfAnimation.name + ": Invalid output attributes component: " + getComponentTypeString(animationOutputAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(animationOutputAccessor.componentType)));
159 continue;
160 }
161 if (maxFrames + channelFrames > animationScaleMatrices[node->getId()].size()) {
162 animationScaleMatrices[node->getId()].resize(maxFrames + channelFrames);
163 }
164 if (maxFrames + channelFrames > animationRotationMatrices[node->getId()].size()) {
165 animationRotationMatrices[node->getId()].resize(maxFrames + channelFrames);
166 }
167 if (maxFrames + channelFrames > animationTranslationMatrices[node->getId()].size()) {
168 animationTranslationMatrices[node->getId()].resize(maxFrames + channelFrames);
169 }
170 // Console::println("xxx: " + node->getId() + ": " + gltfChannel.target_path + " / maxFrames = " + to_string(maxFrames) + ", output frames: " + to_string(animationOutputAccessor.count) + ", " + to_string(animationInputBufferData[0]) + ", channel frames: " + to_string(channelFrames));
171 if (gltfChannel.target_path == "translation") {
172 if (animationOutputAccessor.type != TINYGLTF_TYPE_VEC3) {
173 Console::println("GLTFReader::read(): " + node->getId() + ": animation: " + gltfAnimation.name + ": Invalid translation channel output type: " + getTypeString(animationOutputAccessor.type) + ", expected: Vector3");
174 continue;
175 }
176 vector<Matrix4x4> keyFrameMatrices(animationOutputAccessor.count);
177 for (auto i = 0; i < animationOutputAccessor.count; i++) {
178 keyFrameMatrices[i].identity();
179 keyFrameMatrices[i].translate(Vector3(animationOutputBufferData[i * 3 + 0], animationOutputBufferData[i * 3 + 1], animationOutputBufferData[i * 3 + 2]));
180 }
181 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationTranslationMatrices[node->getId()], maxFrames);
182 } else
183 if (gltfChannel.target_path == "rotation") {
184 if (animationOutputAccessor.type != TINYGLTF_TYPE_VEC4) {
185 Console::println("GLTFReader::read(): " + node->getId() + ": animation: " + gltfAnimation.name + ": Invalid rotation channel output type: " + getTypeString(animationOutputAccessor.type) + ", expected: Vector4");
186 continue;
187 }
188 vector<Matrix4x4> keyFrameMatrices(animationOutputAccessor.count);
189 Quaternion rotationQuaternion;
190 for (auto i = 0; i < animationOutputAccessor.count; i++) {
191 rotationQuaternion.set(animationOutputBufferData[i * 4 + 0], animationOutputBufferData[i * 4 + 1], animationOutputBufferData[i * 4 + 2], animationOutputBufferData[i * 4 + 3]);
192 keyFrameMatrices[i] = rotationQuaternion.computeMatrix();
193 }
194 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationRotationMatrices[node->getId()], maxFrames);
195 } else
196 if (gltfChannel.target_path == "scale") {
197 if (animationOutputAccessor.type != TINYGLTF_TYPE_VEC3) {
198 Console::println("GLTFReader::read(): " + node->getId() + ": animation: " + gltfAnimation.name + ": Invalid scale channel output type: " + getTypeString(animationOutputAccessor.type) + ", expected: Vector3");
199 continue;
200 }
201 vector<Matrix4x4> keyFrameMatrices(animationOutputAccessor.count);
202 for (auto i = 0; i < animationOutputAccessor.count; i++) {
203 keyFrameMatrices[i].identity();
204 keyFrameMatrices[i].scale(Vector3(animationOutputBufferData[i * 3 + 0], animationOutputBufferData[i * 3 + 1], animationOutputBufferData[i * 3 + 2]));
205 }
206 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationScaleMatrices[node->getId()], maxFrames);
207 } else {
208 Console::println("GLTFReader::GLTFReader(): " + gltfAnimation.name + ": Invalid target path:" + gltfChannel.target_path);
209 }
210 if (channelFrames > frames) frames = channelFrames;
211 }
212 model->addAnimationSetup(gltfAnimation.name, maxFrames, maxFrames + frames - 1, false);
213 maxFrames+= frames;
214 }
215
216 // extend all animation matrices to max frames
217 for (auto& it: animationScaleMatrices) {
218 auto& animationMatrices = it.second;
219 animationMatrices.resize(maxFrames);
220 }
221 for (auto& it: animationRotationMatrices) {
222 auto& animationMatrices = it.second;
223 animationMatrices.resize(maxFrames);
224 }
225 for (auto& it: animationTranslationMatrices) {
226 auto& animationMatrices = it.second;
227 animationMatrices.resize(maxFrames);
228 }
229
230 // set up nodes animations if we have frames
231 for (auto& animationNode: animationNodes) {
232 auto node = model->getNodeById(animationNode);
233 if (node == nullptr) {
234 Console::println("GLTFReader::GLTFReader(): animation: node not found:" + animationNode);
235 continue;
236 }
237 vector<Matrix4x4> animationFinalMatrices(maxFrames);
238 Matrix4x4 emptyTransformations;
239 for (auto i = 0; i < maxFrames; i++) {
240 if (animationScaleMatrices[node->getId()][i].equals(emptyTransformations) == true &&
241 animationRotationMatrices[node->getId()][i].equals(emptyTransformations) == true &&
242 animationTranslationMatrices[node->getId()][i].equals(emptyTransformations) == true) {
243 animationFinalMatrices[i] = model->getNodeById(node->getId())->getTransformationsMatrix();
244 } else {
245 animationFinalMatrices[i].identity();
246 if (animationScaleMatrices[node->getId()][i].equals(emptyTransformations) == false) animationFinalMatrices[i].multiply(animationScaleMatrices[node->getId()][i]);
247 if (animationRotationMatrices[node->getId()][i].equals(emptyTransformations) == false) animationFinalMatrices[i].multiply(animationRotationMatrices[node->getId()][i]);
248 if (animationTranslationMatrices[node->getId()][i].equals(emptyTransformations) == false) animationFinalMatrices[i].multiply(animationTranslationMatrices[node->getId()][i]);
249 }
250 }
251 auto animation = new Animation();
252 animation->setTransformationsMatrices(animationFinalMatrices);
253 node->setAnimation(animation);
254 }
255 }
256
257 // create default animations
258 ModelTools::createDefaultAnimation(model, maxFrames);
259 // set up joints
260 ModelTools::setupJoints(model);
261 // fix animation length
262 ModelTools::fixAnimationLength(model);
263
264 //
265 return model;
266}
267
268void GLTFReader::interpolateKeyFrames(int frameTimeCount, const float* frameTimes, const vector<Matrix4x4>& keyFrameMatrices, int interpolatedMatrixCount, vector<Matrix4x4>& interpolatedMatrices, int frameStartIdx) {
269 auto tansformationsMatrixLast = &keyFrameMatrices[0];
270 auto keyFrameIdx = 0;
271 auto frameIdx = 0;
272 auto timeStampLast = 0.0f;
273 for (auto i = 0; i < frameTimeCount; i++) {
274 auto keyFrameTime = frameTimes[i];
275 auto transformationsMatrixCurrent = &keyFrameMatrices[(keyFrameIdx) % keyFrameMatrices.size()];
276 float timeStamp;
277 for (timeStamp = timeStampLast; timeStamp < keyFrameTime; timeStamp += 1.0f / 30.0f) {
278 if (frameIdx >= interpolatedMatrixCount) {
279 // TODO: check me again!
280 // Console::println(string("Warning: skipping frame: ") + to_string(frameIdx));
281 frameIdx++;
282 continue;
283 }
284 // Console::println("yyy: " + to_string(frameStartIdx + frameIdx) + ": key frame idx: " + to_string(keyFrameIdx) + ", interpolation t: " + to_string((timeStamp - timeStampLast) / (keyFrameTime - timeStampLast)));
285 interpolatedMatrices[frameStartIdx + frameIdx] = Matrix4x4::interpolateLinear(*tansformationsMatrixLast, *transformationsMatrixCurrent, (timeStamp - timeStampLast) / (keyFrameTime - timeStampLast));
286 frameIdx++;
287 }
288 timeStampLast = timeStamp;
289 tansformationsMatrixLast = transformationsMatrixCurrent;
290 keyFrameIdx++;
291 }
292}
293
294Node* GLTFReader::parseNode(const string& pathName, const tinygltf::Model& gltfModel, int gltfNodeIdx, Model* model, Node* parentNode) {
295 auto& gltfNode = gltfModel.nodes[gltfNodeIdx];
296 auto node = new Node(model, parentNode, gltfNode.name, gltfNode.name);
297 if (gltfNode.matrix.size() == 16) {
298 node->setTransformationsMatrix(
299 Matrix4x4(
300 static_cast<float>(gltfNode.matrix[0]),
301 static_cast<float>(gltfNode.matrix[1]),
302 static_cast<float>(gltfNode.matrix[2]),
303 static_cast<float>(gltfNode.matrix[3]),
304 static_cast<float>(gltfNode.matrix[4]),
305 static_cast<float>(gltfNode.matrix[5]),
306 static_cast<float>(gltfNode.matrix[6]),
307 static_cast<float>(gltfNode.matrix[7]),
308 static_cast<float>(gltfNode.matrix[8]),
309 static_cast<float>(gltfNode.matrix[9]),
310 static_cast<float>(gltfNode.matrix[10]),
311 static_cast<float>(gltfNode.matrix[11]),
312 static_cast<float>(gltfNode.matrix[12]),
313 static_cast<float>(gltfNode.matrix[13]),
314 static_cast<float>(gltfNode.matrix[14]),
315 static_cast<float>(gltfNode.matrix[15])
316 )
317 );
318 } else {
319 Matrix4x4 nodeScaleMatrices;
320 Matrix4x4 nodeRotationMatrices;
321 Matrix4x4 nodeTranslationMatrices;
322 nodeScaleMatrices.identity();
323 nodeRotationMatrices.identity();
324 nodeTranslationMatrices.identity();
325 if (gltfNode.scale.size() == 3) {
326 nodeScaleMatrices.scale(Vector3(gltfNode.scale[0], gltfNode.scale[1], gltfNode.scale[2]));
327 }
328 if (gltfNode.rotation.size() == 4) {
329 Quaternion rotationQuaternion(gltfNode.rotation[0], gltfNode.rotation[1], gltfNode.rotation[2], gltfNode.rotation[3]);
330 nodeRotationMatrices = rotationQuaternion.computeMatrix();
331 }
332 if (gltfNode.translation.size() == 3) {
333 nodeTranslationMatrices.translate(Vector3(gltfNode.translation[0], gltfNode.translation[1], gltfNode.translation[2]));
334 }
335 Matrix4x4 nodeTransformationsMatrix;
336 nodeTransformationsMatrix.set(nodeScaleMatrices);
337 nodeTransformationsMatrix.multiply(nodeRotationMatrices);
338 nodeTransformationsMatrix.multiply(nodeTranslationMatrices);
339 node->setTransformationsMatrix(nodeTransformationsMatrix);
340 }
341 if (gltfNode.mesh == -1) return node;
342 vector<int> joints;
343 vector<float> weights;
344 vector<Vector3> vertices;
345 vector<Vector3> normals;
346 vector<TextureCoordinate> textureCoordinates;
347 vector<FacesEntity> facesEntities;
348 auto& mesh = gltfModel.meshes[gltfNode.mesh];
349 int facesEntityIdx = 0;
350 for (auto& gltfPrimitive: mesh.primitives) {
351 Material* material = nullptr;
352 if (gltfPrimitive.material != -1) {
353 auto& gltfMaterial = gltfModel.materials[gltfPrimitive.material];
354 auto& gltfMaterialName = gltfMaterial.name;
355 auto materialIt = model->getMaterials().find(gltfMaterialName);
356 if (materialIt != model->getMaterials().end()) {
357 material = materialIt->second;
358 } else {
359 material = new Material(gltfMaterial.name);
360 material->setDoubleSided(false/*TODO: enable me: gltfMaterial.doubleSided*/);
361 auto pbrMaterialProperties = new PBRMaterialProperties();
362 pbrMaterialProperties->setEmbedTextures(true);
363 auto specularMaterialProperties = new SpecularMaterialProperties();
364 specularMaterialProperties->setEmbedTextures(true);
365 // some adjustment, lets see if we can extract this later
366 specularMaterialProperties->setAmbientColor(Color4(0.8f, 0.8f, 0.8f, 1.0f));
367 specularMaterialProperties->setDiffuseColor(Color4(0.2f, 0.2f, 0.2f, 1.0f));
368 if (gltfMaterial.values.find("baseColorFactor") != gltfMaterial.values.end()) {
369 auto& gltfMaterialBaseColorFactor = gltfMaterial.values.find("baseColorFactor")->second;
370 Console::println(
371 "GLTFReader::parseNode(): " +
372 node->getId() + ": " +
373 "have base color factor with " +
374 to_string(gltfMaterialBaseColorFactor.number_array[0]) + ", " +
375 to_string(gltfMaterialBaseColorFactor.number_array[1]) + ", " +
376 to_string(gltfMaterialBaseColorFactor.number_array[2]) + ", " +
377 to_string(gltfMaterialBaseColorFactor.number_array[3])
378 );
379 pbrMaterialProperties->setBaseColorFactor(
380 Color4(
381 gltfMaterialBaseColorFactor.number_array[0],
382 gltfMaterialBaseColorFactor.number_array[1],
383 gltfMaterialBaseColorFactor.number_array[2],
384 gltfMaterialBaseColorFactor.number_array[3]
385 )
386 );
387 specularMaterialProperties->setAmbientColor(
388 Color4(
389 specularMaterialProperties->getAmbientColor().getRed() * gltfMaterialBaseColorFactor.number_array[0],
390 specularMaterialProperties->getAmbientColor().getGreen() * gltfMaterialBaseColorFactor.number_array[1],
391 specularMaterialProperties->getAmbientColor().getBlue() * gltfMaterialBaseColorFactor.number_array[2],
392 gltfMaterialBaseColorFactor.number_array[3]
393 )
394 );
395 specularMaterialProperties->setDiffuseColor(
396 Color4(
397 specularMaterialProperties->getDiffuseColor().getRed() * gltfMaterialBaseColorFactor.number_array[0],
398 specularMaterialProperties->getDiffuseColor().getGreen() * gltfMaterialBaseColorFactor.number_array[1],
399 specularMaterialProperties->getDiffuseColor().getBlue() * gltfMaterialBaseColorFactor.number_array[2],
400 gltfMaterialBaseColorFactor.number_array[3]
401 )
402 );
403 }
404 if (gltfMaterial.values.find("metallicFactor") != gltfMaterial.values.end()) {
405 auto& gltfMaterialMatallicFactor = gltfMaterial.values.find("metallicFactor")->second;
406 Console::println(
407 "GLTFReader::parseNode(): " +
408 node->getId() + ": " +
409 "have metallic factor with " +
410 to_string(gltfMaterialMatallicFactor.number_value)
411 );
412 pbrMaterialProperties->setMetallicFactor(gltfMaterialMatallicFactor.number_value);
413 }
414 if (gltfMaterial.values.find("roughnessFactor") != gltfMaterial.values.end()) {
415 auto& gltfMaterialRoughnessFactor = gltfMaterial.values.find("roughnessFactor")->second;
416 Console::println(
417 "GLTFReader::parseNode(): " +
418 node->getId() + ": " +
419 "have roughness factor with " +
420 to_string(gltfMaterialRoughnessFactor.number_value)
421 );
422 pbrMaterialProperties->setRoughnessFactor(gltfMaterialRoughnessFactor.number_value);
423 }
424 // we ignore for now Factor, ColorFactor, TextureScale, TextureStrength, TextureTexCoord as I do not see them feasible in Blender exported GLTF files
425 if (gltfMaterial.values.find("baseColorTexture") != gltfMaterial.values.end() &&
426 gltfMaterial.values.find("baseColorTexture")->second.TextureIndex() != -1) {
427 auto& gltfMaterialBaseColorTexture = gltfMaterial.values.find("baseColorTexture")->second;
428 auto& gltfTexture = gltfModel.textures[gltfMaterialBaseColorTexture.TextureIndex()];
429 auto& image = gltfModel.images[gltfTexture.source];
430 try {
431 if (image.component != 3 && image.component != 4) throw ExceptionBase("We only support RGB or RGBA textures for now");
432 if (image.bits != 8) throw ExceptionBase("We only support 8 bit channels for now");
433 auto fileName = determineTextureFileName(image.name);
434 Console::println("GLTFReader::parseNode(): " + node->getId() + ": have base color texture with " + to_string(image.width) + " x " + to_string(image.height) + " x " + to_string(image.component) + " x " + to_string(image.bits) + ": " + fileName);
435 auto textureData = ByteBuffer::allocate(image.width * image.height * image.component * image.bits / 8);
436 for (int y = image.height - 1; y >= 0; y--) {
437 textureData->put(&image.image[y * image.width * image.component * image.bits / 8], image.width * image.component * image.bits / 8);
438 }
439 auto texture = new Texture(
440 fileName,
441 image.bits * image.component,
442 image.width,
443 image.height,
444 image.width,
445 image.height,
446 textureData
447 );
448 pbrMaterialProperties->setBaseColorTexture(texture);
449 if (pbrMaterialProperties->hasBaseColorTextureTransparency() == true) pbrMaterialProperties->setBaseColorTextureMaskedTransparency(true);
450 specularMaterialProperties->setDiffuseTexture(texture);
451 if (specularMaterialProperties->hasDiffuseTextureTransparency() == true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(true);
452 } catch (Exception& exception) {
453 Console::println("GLTFReader::parseNode(): " + node->getId() + ": An error occurred: " + exception.what());
454 }
455 }
456 if (gltfMaterial.values.find("metallicRoughnessTexture") != gltfMaterial.values.end() &&
457 gltfMaterial.values.find("metallicRoughnessTexture")->second.TextureIndex() != -1) {
458 auto& gltfMetallicRoughnessTexture = gltfMaterial.values.find("metallicRoughnessTexture")->second;
459 auto& gltfTexture = gltfModel.textures[gltfMetallicRoughnessTexture.TextureIndex()];
460 auto& image = gltfModel.images[gltfTexture.source];
461 try {
462 if (image.component != 3 && image.component != 4) throw ExceptionBase("We only support RGB or RGBA textures for now");
463 if (image.bits != 8) throw ExceptionBase("We only support 8 bit channels for now");
464 auto fileName = determineTextureFileName(image.name);
465 Console::println("GLTFReader::parseNode(): " + node->getId() + ": have metallic roughness texture with " + to_string(image.width) + " x " + to_string(image.height) + " x " + to_string(image.component) + " x " + to_string(image.bits) + ": " + fileName);
466 auto textureData = ByteBuffer::allocate(image.width * image.height * image.component * image.bits / 8);
467 for (int y = image.height - 1; y >= 0; y--) {
468 textureData->put(&image.image[y * image.width * image.component * image.bits / 8], image.width * image.component * image.bits / 8);
469 }
470 auto texture = new Texture(
471 fileName,
472 image.bits * image.component,
473 image.width,
474 image.height,
475 image.width,
476 image.height,
477 textureData
478 );
479 pbrMaterialProperties->setMetallicRoughnessTexture(texture);
480 } catch (Exception& exception) {
481 Console::println("GLTFReader::parseNode(): " + node->getId() + ": An error occurred: " + exception.what());
482 }
483 }
484 if (gltfMaterial.additionalValues.find("normalTexture") != gltfMaterial.additionalValues.end() &&
485 gltfMaterial.additionalValues.find("normalTexture")->second.TextureIndex() != -1) {
486 auto& gltfNormalTexture = gltfMaterial.additionalValues.find("normalTexture")->second;
487 auto& gltfTexture = gltfModel.textures[gltfNormalTexture.TextureIndex()];
488 auto& image = gltfModel.images[gltfTexture.source];
489 try {
490 if (image.component != 3 && image.component != 4) throw ExceptionBase("We only support RGB or RGBA textures for now");
491 if (image.bits != 8) throw ExceptionBase("We only support 8 bit channels for now");
492 auto fileName = determineTextureFileName(image.name);
493 Console::println("GLTFReader::parseNode(): " + node->getId() + ": have normal texture with " + to_string(image.width) + " x " + to_string(image.height) + " x " + to_string(image.component) + " x " + to_string(image.bits) + ": " + fileName);
494 auto textureData = ByteBuffer::allocate(image.width * image.height * image.component * image.bits / 8);
495 for (int y = image.height - 1; y >= 0; y--) {
496 textureData->put(&image.image[y * image.width * image.component * image.bits / 8], image.width * image.component * image.bits / 8);
497 }
498 auto texture = new Texture(
499 fileName,
500 image.bits * image.component,
501 image.width,
502 image.height,
503 image.width,
504 image.height,
505 textureData
506 );
507 pbrMaterialProperties->setNormalTexture(texture);
508 } catch (Exception& exception) {
509 Console::println("GLTFReader::parseNode(): " + node->getId() + ": An error occurred: " + exception.what());
510 }
511 }
512 material->setSpecularMaterialProperties(specularMaterialProperties);
513 material->setPBRMaterialProperties(pbrMaterialProperties);
514 model->getMaterials()[material->getId()] = material;
515 }
516 }
517 if (gltfPrimitive.mode != 4) {
518 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid primitive mode: " + to_string(gltfPrimitive.mode));
519 continue;
520 }
521 vector<int> indices;
522 {
523 auto& indicesAccessor = gltfModel.accessors[gltfPrimitive.indices];
524 auto& indicesBufferView = gltfModel.bufferViews[indicesAccessor.bufferView];
525 auto& indicesBuffer = gltfModel.buffers[indicesBufferView.buffer];
526 if (indicesBufferView.byteStride != 0) {
527 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid stride: " + to_string(indicesBufferView.byteStride));
528 } else
529 switch (indicesAccessor.componentType) {
530 case TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT:
531 {
532 // TODO: stride
533 const uint16_t* indicesBufferData = (const uint16_t*)(indicesBuffer.data.data() + indicesAccessor.byteOffset + indicesBufferView.byteOffset);
534 for (auto i = 0; i < indicesAccessor.count; i++) {
535 indices.push_back(indicesBufferData[i]);
536 }
537 break;
538 }
539 case TINYGLTF_COMPONENT_TYPE_UNSIGNED_INT:
540 {
541 // TODO: stride
542 const uint32_t* indicesBufferData = (const uint32_t*)(indicesBuffer.data.data() + indicesAccessor.byteOffset + indicesBufferView.byteOffset);
543 for (auto i = 0; i < indicesAccessor.count; i++) {
544 indices.push_back(indicesBufferData[i]);
545 }
546 break;
547 }
548 default:
549 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid indices component: " + to_string(indicesAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(indicesAccessor.componentType)));
550 }
551 }
552 auto start = 0;
553 bool haveVertices = false;
554 bool haveNormals = false;
555 bool haveTextureCoordinates = false;
556 for (auto& gltfAttributeIt: gltfPrimitive.attributes) {
557 auto gltfBufferType = gltfAttributeIt.first;
558 auto& attributeAccessor = gltfModel.accessors[gltfAttributeIt.second];
559 auto& attributeBufferView = gltfModel.bufferViews[attributeAccessor.bufferView];
560 auto& attributeBuffer = gltfModel.buffers[attributeBufferView.buffer];
561 if (attributeBufferView.byteStride != 0) {
562 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid attributes stride: " + to_string(attributeBufferView.byteStride));
563 } else {
564 if (gltfBufferType == "POSITION") {
565 if (attributeAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
566 Console::println("GLTFReader::parseNode(): " + node->getId() + ": POSITION: Invalid attributes component: " + to_string(attributeAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(attributeAccessor.componentType)));
567 continue;
568 }
569 haveVertices = true;
570 start = vertices.size();
571 if (start + attributeAccessor.count > vertices.size()) vertices.resize(start + attributeAccessor.count);
572 auto bufferData = (const float*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
573 for (auto i = 0; i < attributeAccessor.count; i++) {
574 vertices[start + i] = Vector3(bufferData[i * 3 + 0], bufferData[i * 3 + 1], bufferData[i * 3 + 2]);
575 }
576 } else
577 if (gltfBufferType == "NORMAL") {
578 if (attributeAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
579 Console::println("GLTFReader::parseNode(): " + node->getId() + ": NORMAL: Invalid attributes component: " + to_string(attributeAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(attributeAccessor.componentType)));
580 continue;
581 }
582 haveNormals = true;
583 auto start = normals.size();
584 if (start + attributeAccessor.count > normals.size()) normals.resize(start + attributeAccessor.count);
585 auto bufferData = (const float*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
586 for (auto i = 0; i < attributeAccessor.count; i++) {
587 normals[start + i] = Vector3(bufferData[i * 3 + 0], bufferData[i * 3 + 1], bufferData[i * 3 + 2]);
588 }
589 } else
590 if (gltfBufferType == "TEXCOORD_0") {
591 if (attributeAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
592 Console::println("GLTFReader::parseNode(): " + node->getId() + ": TEXTCOORD_0: Invalid attributes component: " + to_string(attributeAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(attributeAccessor.componentType)));
593 continue;
594 }
595 haveTextureCoordinates = true;
596 auto start = textureCoordinates.size();
597 if (start + attributeAccessor.count > textureCoordinates.size()) textureCoordinates.resize(start + attributeAccessor.count);
598 auto bufferData = (const float*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
599 for (auto i = 0; i < attributeAccessor.count; i++) {
600 textureCoordinates[start + i] = TextureCoordinate(bufferData[i * 2 + 0], bufferData[i * 2 + 1]);
601 }
602 } else
603 if (gltfBufferType == "COLOR_0") {
604 // ignored for now
605 } else
606 if (gltfBufferType == "WEIGHTS_0") {
607 if (attributeAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
608 Console::println("GLTFReader::parseNode(): " + node->getId() + ": WEIGHTS_0: Invalid attributes component: " + to_string(attributeAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(attributeAccessor.componentType)));
609 continue;
610 }
611 auto start = weights.size();
612 if (start + attributeAccessor.count * 4 > weights.size()) weights.resize(start + attributeAccessor.count * 4);
613 auto bufferData = (const float*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
614 for (auto i = 0; i < attributeAccessor.count * 4; i++) {
615 weights[start + i] = bufferData[i];
616 }
617 } else
618 if (gltfBufferType == "JOINTS_0") {
619 if (attributeAccessor.componentType == TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE) {
620 auto start = joints.size();
621 if (start + attributeAccessor.count * 4 > joints.size()) joints.resize(start + attributeAccessor.count * 4);
622 auto bufferData = (const uint8_t*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
623 for (auto i = 0; i < attributeAccessor.count * 4; i++) {
624 joints[start + i] = bufferData[i];
625 }
626 } else
627 if (attributeAccessor.componentType == TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT) {
628 auto start = joints.size();
629 if (start + attributeAccessor.count * 4 > joints.size()) joints.resize(start + attributeAccessor.count * 4);
630 auto bufferData = (const uint16_t*)(attributeBuffer.data.data() + attributeAccessor.byteOffset + attributeBufferView.byteOffset);
631 for (auto i = 0; i < attributeAccessor.count * 4; i++) {
632 joints[start + i] = bufferData[i];
633 }
634 } else {
635 Console::println("GLTFReader::parseNode(): " + node->getId() + ": JOINTS_0: Invalid attributes component: " + to_string(attributeAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(attributeAccessor.componentType)));
636 continue;
637 }
638 } else {
639 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid buffer type: " + gltfBufferType);
640 }
641 }
642 }
643 FacesEntity facesEntity(node, node->getId() + "-" + to_string(facesEntityIdx));
644 facesEntity.setMaterial(material);
645 vector<Face> faces;
646 if (haveVertices == false || haveNormals == false) throw ModelFileIOException("Missing vertices or normals");
647 if (haveTextureCoordinates == true) {
648 for (auto i = 0; i < indices.size() / 3; i++) {
649 faces.push_back(
650 Face(
651 node,
652 start + indices[i * 3 + 0], start + indices[i * 3 + 1], start + indices[i * 3 + 2],
653 start + indices[i * 3 + 0], start + indices[i * 3 + 1], start + indices[i * 3 + 2],
654 start + indices[i * 3 + 0], start + indices[i * 3 + 1], start + indices[i * 3 + 2]
655 )
656 );
657 }
658 } else {
659 for (auto i = 0; i < indices.size() / 3; i++) {
660 faces.push_back(
661 Face(
662 node,
663 start + indices[i * 3 + 0], start + indices[i * 3 + 1], start + indices[i * 3 + 2],
664 start + indices[i * 3 + 0], start + indices[i * 3 + 1], start + indices[i * 3 + 2]
665 )
666 );
667 }
668 }
669 facesEntity.setFaces(faces);
670 facesEntities.push_back(facesEntity);
671 facesEntityIdx++;
672 }
673
674 // skinning
675 if (gltfNode.skin != -1) {
676 auto& gltfSkin = gltfModel.skins[gltfNode.skin];
677 auto& inverseBindMatricesAccessor = gltfModel.accessors[gltfSkin.inverseBindMatrices];
678 auto& inverseBindMatricesBufferView = gltfModel.bufferViews[inverseBindMatricesAccessor.bufferView];
679 auto& inverseBindMatricesBuffer = gltfModel.buffers[inverseBindMatricesBufferView.buffer];
680 const float* inverseBindMatricesBufferData = nullptr;
681 if (inverseBindMatricesBufferView.byteStride != 0) {
682 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Invalid attributes stride: " + to_string(inverseBindMatricesBufferView.byteStride));
683 } else
684 if (inverseBindMatricesAccessor.componentType != TINYGLTF_COMPONENT_TYPE_FLOAT) {
685 Console::println("GLTFReader::parseNode(): " + node->getId() + ": Inverse bind matrices: Invalid attributes component: " + to_string(inverseBindMatricesAccessor.componentType) + ", with size: " + to_string(getComponentTypeByteSize(inverseBindMatricesAccessor.componentType)));
686 } else {
687 inverseBindMatricesBufferData = (const float*)(inverseBindMatricesBuffer.data.data() + inverseBindMatricesAccessor.byteOffset + inverseBindMatricesBufferView.byteOffset);
688 }
689 if (inverseBindMatricesBufferData != nullptr) {
690 auto skinning = new Skinning();
691 {
692 auto i = 0;
693 vector<Joint> skinningJoints(gltfSkin.joints.size());
694 for (auto gltfJointNodeIdx: gltfSkin.joints) {
695 Joint joint(gltfModel.nodes[gltfJointNodeIdx].name);
696 joint.setBindMatrix(
697 Matrix4x4(
698 inverseBindMatricesBufferData[i * 16 + 0],
699 inverseBindMatricesBufferData[i * 16 + 1],
700 inverseBindMatricesBufferData[i * 16 + 2],
701 inverseBindMatricesBufferData[i * 16 + 3],
702 inverseBindMatricesBufferData[i * 16 + 4],
703 inverseBindMatricesBufferData[i * 16 + 5],
704 inverseBindMatricesBufferData[i * 16 + 6],
705 inverseBindMatricesBufferData[i * 16 + 7],
706 inverseBindMatricesBufferData[i * 16 + 8],
707 inverseBindMatricesBufferData[i * 16 + 9],
708 inverseBindMatricesBufferData[i * 16 + 10],
709 inverseBindMatricesBufferData[i * 16 + 11],
710 inverseBindMatricesBufferData[i * 16 + 12],
711 inverseBindMatricesBufferData[i * 16 + 13],
712 inverseBindMatricesBufferData[i * 16 + 14],
713 inverseBindMatricesBufferData[i * 16 + 15]
714 )
715 );
716 skinningJoints[i++] = joint;
717 }
718 skinning->setJoints(skinningJoints);
719 }
720 {
721 vector<float> skinningWeights;
722 vector<vector<JointWeight>> skinningJointWeights(vertices.size());
723 for (auto i = 0; i < vertices.size(); i++) {
724 for (auto j = 0; j < 4; j++) {
725 // TODO: reuse weights
726 if (weights[i * 4 + j] > Math::EPSILON) {
727 skinningJointWeights[i].push_back(JointWeight(joints[i * 4 + j], skinningWeights.size()));
728 skinningWeights.push_back(weights[i * 4 + j]);
729 }
730 }
731 }
732 skinning->setWeights(skinningWeights);
733 skinning->setVerticesJointsWeights(skinningJointWeights);
734 }
735 node->setSkinning(skinning);
736 }
737 }
738
739 // set up node
740 node->setVertices(vertices);
741 node->setNormals(normals);
742 node->setTextureCoordinates(textureCoordinates);
743 node->setFacesEntities(facesEntities);
744
745 // create tangents and bitangets, as they seem not be delivered in GLTF files but needed for PBR
746 if (vertices.empty() == false && normals.empty() == false) {
747 ModelTools::createTangentsAndBitangents(node);
748 };
749
750 //
751 return node;
752}
753
754void GLTFReader::parseNodeChildren(const string& pathName, const tinygltf::Model& gltfModel, const vector<int>& gltfNodeChildrenIdx, Node* parentNode) {
755 for (auto gltfNodeIdx: gltfNodeChildrenIdx) {
756 auto& gltfNode = gltfModel.nodes[gltfNodeIdx];
757 auto node = parseNode(pathName, gltfModel, gltfNodeIdx, parentNode->getModel(), parentNode);
758 parentNode->getModel()->getNodes()[node->getId()] = node;
759 if (parentNode->getSubNodes().find(node->getId()) != parentNode->getSubNodes().end()) {
760 Console::println("GLTFReader::parseNodeChildren(): node already exists: " + node->getId());
761 }
762 parentNode->getSubNodes()[node->getId()] = node;
763 if (gltfNode.children.empty() == false) parseNodeChildren(pathName, gltfModel, gltfNode.children, node);
764 }
765}
766
767
768string GLTFReader::determineTextureFileName(const string& imageName) {
769 /*
770 // try to avoid double parts in names that can happen when having 2 maps, 1 for colors and 1 for transparency
771 string doubleFileNamePart = "";
772 for (auto i = 3; i < imageName.size(); i++) {
773 auto doubleFileNamePartTest = StringTools::substring(imageName, 0, i);
774 if (imageName.rfind(doubleFileNamePartTest) > i) {
775 doubleFileNamePart = doubleFileNamePartTest;
776 } else {
777 break;
778 }
779 }
780 if (doubleFileNamePart.empty() == false) {
781 return StringTools::replace(imageName, doubleFileNamePart, "", imageName.rfind(doubleFileNamePart)) + ".png";
782 } else {
783 return imageName + ".png";
784 }
785 */
786 return imageName + ".png";
787}
static string getComponentTypeString(int type)
Definition: GLTFReader.h:72
static string getTypeString(int type)
Definition: GLTFReader.h:98
static size_t getComponentTypeByteSize(int type)
Definition: GLTFReader.h:49
static string determineTextureFileName(const string &imageName)
Determine texture file name.
Definition: GLTFReader.cpp:768
static Node * parseNode(const string &pathName, const tinygltf::Model &gltfModel, int gltfNodeIdx, Model *model, Node *parentNode)
Parse GLTF node.
Definition: GLTFReader.cpp:294
static void parseNodeChildren(const string &pathName, const tinygltf::Model &gltfModel, const vector< int > &gltfNodeChildrenIdx, Node *parentNode)
Parse GLTF node children into TDME node.
Definition: GLTFReader.cpp:754
static void interpolateKeyFrames(int frameTimeCount, const float *frameTimes, const vector< Matrix4x4 > &keyFrameMatrices, int interpolatedMatrixCount, vector< Matrix4x4 > &interpolatedMatrices, int frameStartIdx)
Interpolate key frames to our internal 30fps format.
Definition: GLTFReader.cpp:268
Color 4 base definition class.
Definition: Color4Base.h:19
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
void setMaterial(Material *material)
Set up the entity's material.
Definition: FacesEntity.h:72
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Definition: FacesEntity.cpp:47
Joint / Bone.
Definition: Joint.h:19
void setBindMatrix(const Matrix4x4 &bindMatrix)
Bind matrix.
Definition: Joint.h:55
Represents a material.
Definition: Material.h:21
void setSpecularMaterialProperties(SpecularMaterialProperties *specularMaterialProperties)
Set specular material properties.
Definition: Material.cpp:22
void setPBRMaterialProperties(PBRMaterialProperties *pbrMaterialProperties)
Set PBR material properties.
Definition: Material.cpp:29
const string & getId() const
Definition: Material.h:57
void setDoubleSided(bool doubleSided)
Set double sided.
Definition: Material.h:112
Representation of a 3d model.
Definition: Model.h:32
map< string, Node * > & getNodes()
Returns all object's nodes.
Definition: Model.h:179
map< string, Material * > & getMaterials()
Returns all object materials.
Definition: Model.h:171
Model node.
Definition: Node.h:31
Model * getModel()
Definition: Node.h:70
map< string, Node * > & getSubNodes()
Definition: Node.h:289
Represents specular material properties.
Represents rotation orders of a model.
Definition: RotationOrder.h:23
Skinning definition for nodes.
Definition: Skinning.h:27
Represents specular material properties.
Class representing texture UV coordinates data.
Model up vector.
Definition: UpVector.h:20
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & identity()
Setup identity matrix.
Definition: Matrix4x4.h:326
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Definition: Matrix4x4.h:95
Matrix4x4 & translate(const Vector3 &v)
Sets up a translation matrix.
Definition: Matrix4x4.h:460
Matrix4x4 & scale(float s)
Scales this matrix.
Definition: Matrix4x4.h:418
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
Quaternion class.
Definition: Quaternion.h:22
Matrix4x4 computeMatrix() const
Computes a matrix from given.
Definition: Quaternion.h:299
Quaternion & set(float x, float y, float z, float w)
Set up this quaternion by components.
Definition: Quaternion.h:68
3D vector 3 class
Definition: Vector3.h:22
File system singleton class.
Definition: FileSystem.h:14
Byte buffer class.
Definition: ByteBuffer.h:24
Console class.
Definition: Console.h:26
Exception base class.
Definition: ExceptionBase.h:20
Model tools functions class.
Definition: ModelTools.h:38
String tools class.
Definition: StringTools.h:20
std::exception Exception
Exception base class.
Definition: Exception.h:19