8#define TINYGLTF_IMPLEMENTATION
9#define STB_IMAGE_IMPLEMENTATION
10#define STB_IMAGE_WRITE_IMPLEMENTATION
12#include <ext/libpng/png.h>
13#include <ext/tinygltf/tiny_gltf.h>
86Model* GLTFReader::read(
const string& pathName,
const string& fileName)
89 vector<uint8_t> glftBinaryData;
90 FileSystem::getInstance()->getContent(pathName, fileName, glftBinaryData);
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);
100 Console::println(
"GLTFReader::read(): Failed to load model: " + pathName +
"/" + fileName);
105 auto model =
new Model(
113 model->setShaderModel(ShaderModel::PBR);
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());
124 model->getSubNodes()[node->getId()] = node;
125 if (glTfNode.children.empty() ==
false)
parseNodeChildren(pathName, gltfModel, glTfNode.children, node);
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) {
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];
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)));
151 auto channelFrames =
static_cast<int32_t
>(Math::ceil(animationInputBufferData[animationInputAccessor.count - 1] * 30.0f));
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)));
161 if (maxFrames + channelFrames > animationScaleMatrices[node->getId()].size()) {
162 animationScaleMatrices[node->getId()].resize(maxFrames + channelFrames);
164 if (maxFrames + channelFrames > animationRotationMatrices[node->getId()].size()) {
165 animationRotationMatrices[node->getId()].resize(maxFrames + channelFrames);
167 if (maxFrames + channelFrames > animationTranslationMatrices[node->getId()].size()) {
168 animationTranslationMatrices[node->getId()].resize(maxFrames + 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");
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]));
181 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationTranslationMatrices[node->getId()], maxFrames);
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");
188 vector<Matrix4x4> keyFrameMatrices(animationOutputAccessor.count);
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]);
194 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationRotationMatrices[node->getId()], maxFrames);
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");
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]));
206 interpolateKeyFrames(animationInputAccessor.count, animationInputBufferData, keyFrameMatrices, channelFrames, animationScaleMatrices[node->getId()], maxFrames);
208 Console::println(
"GLTFReader::GLTFReader(): " + gltfAnimation.name +
": Invalid target path:" + gltfChannel.target_path);
210 if (channelFrames > frames) frames = channelFrames;
212 model->addAnimationSetup(gltfAnimation.name, maxFrames, maxFrames + frames - 1,
false);
217 for (
auto& it: animationScaleMatrices) {
218 auto& animationMatrices = it.second;
219 animationMatrices.resize(maxFrames);
221 for (
auto& it: animationRotationMatrices) {
222 auto& animationMatrices = it.second;
223 animationMatrices.resize(maxFrames);
225 for (
auto& it: animationTranslationMatrices) {
226 auto& animationMatrices = it.second;
227 animationMatrices.resize(maxFrames);
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);
237 vector<Matrix4x4> animationFinalMatrices(maxFrames);
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();
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]);
252 animation->setTransformationsMatrices(animationFinalMatrices);
253 node->setAnimation(animation);
258 ModelTools::createDefaultAnimation(model, maxFrames);
260 ModelTools::setupJoints(model);
262 ModelTools::fixAnimationLength(model);
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;
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()];
277 for (timeStamp = timeStampLast; timeStamp < keyFrameTime; timeStamp += 1.0f / 30.0f) {
278 if (frameIdx >= interpolatedMatrixCount) {
285 interpolatedMatrices[frameStartIdx + frameIdx] = Matrix4x4::interpolateLinear(*tansformationsMatrixLast, *transformationsMatrixCurrent, (timeStamp - timeStampLast) / (keyFrameTime - timeStampLast));
288 timeStampLast = timeStamp;
289 tansformationsMatrixLast = transformationsMatrixCurrent;
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(
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])
325 if (gltfNode.scale.size() == 3) {
326 nodeScaleMatrices.
scale(
Vector3(gltfNode.scale[0], gltfNode.scale[1], gltfNode.scale[2]));
328 if (gltfNode.rotation.size() == 4) {
329 Quaternion rotationQuaternion(gltfNode.rotation[0], gltfNode.rotation[1], gltfNode.rotation[2], gltfNode.rotation[3]);
332 if (gltfNode.translation.size() == 3) {
333 nodeTranslationMatrices.
translate(
Vector3(gltfNode.translation[0], gltfNode.translation[1], gltfNode.translation[2]));
336 nodeTransformationsMatrix.
set(nodeScaleMatrices);
337 nodeTransformationsMatrix.
multiply(nodeRotationMatrices);
338 nodeTransformationsMatrix.
multiply(nodeTranslationMatrices);
339 node->setTransformationsMatrix(nodeTransformationsMatrix);
341 if (gltfNode.mesh == -1)
return node;
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) {
352 if (gltfPrimitive.material != -1) {
353 auto& gltfMaterial = gltfModel.materials[gltfPrimitive.material];
354 auto& gltfMaterialName = gltfMaterial.name;
355 auto materialIt = model->
getMaterials().find(gltfMaterialName);
357 material = materialIt->second;
359 material =
new Material(gltfMaterial.name);
362 pbrMaterialProperties->setEmbedTextures(
true);
364 specularMaterialProperties->setEmbedTextures(
true);
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;
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])
379 pbrMaterialProperties->setBaseColorFactor(
381 gltfMaterialBaseColorFactor.number_array[0],
382 gltfMaterialBaseColorFactor.number_array[1],
383 gltfMaterialBaseColorFactor.number_array[2],
384 gltfMaterialBaseColorFactor.number_array[3]
387 specularMaterialProperties->setAmbientColor(
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]
395 specularMaterialProperties->setDiffuseColor(
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]
404 if (gltfMaterial.values.find(
"metallicFactor") != gltfMaterial.values.end()) {
405 auto& gltfMaterialMatallicFactor = gltfMaterial.values.find(
"metallicFactor")->second;
407 "GLTFReader::parseNode(): " +
408 node->getId() +
": " +
409 "have metallic factor with " +
410 to_string(gltfMaterialMatallicFactor.number_value)
412 pbrMaterialProperties->setMetallicFactor(gltfMaterialMatallicFactor.number_value);
414 if (gltfMaterial.values.find(
"roughnessFactor") != gltfMaterial.values.end()) {
415 auto& gltfMaterialRoughnessFactor = gltfMaterial.values.find(
"roughnessFactor")->second;
417 "GLTFReader::parseNode(): " +
418 node->getId() +
": " +
419 "have roughness factor with " +
420 to_string(gltfMaterialRoughnessFactor.number_value)
422 pbrMaterialProperties->setRoughnessFactor(gltfMaterialRoughnessFactor.number_value);
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];
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");
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);
441 image.bits * image.component,
448 pbrMaterialProperties->setBaseColorTexture(texture);
449 if (pbrMaterialProperties->hasBaseColorTextureTransparency() ==
true) pbrMaterialProperties->setBaseColorTextureMaskedTransparency(
true);
450 specularMaterialProperties->setDiffuseTexture(texture);
451 if (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(
true);
453 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": An error occurred: " + exception.what());
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];
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");
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);
472 image.bits * image.component,
479 pbrMaterialProperties->setMetallicRoughnessTexture(texture);
481 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": An error occurred: " + exception.what());
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];
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");
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);
500 image.bits * image.component,
507 pbrMaterialProperties->setNormalTexture(texture);
509 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": An error occurred: " + exception.what());
517 if (gltfPrimitive.mode != 4) {
518 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": Invalid primitive mode: " + to_string(gltfPrimitive.mode));
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));
529 switch (indicesAccessor.componentType) {
530 case TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT:
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]);
539 case TINYGLTF_COMPONENT_TYPE_UNSIGNED_INT:
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]);
549 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": Invalid indices component: " + to_string(indicesAccessor.componentType) +
", with size: " + to_string(
getComponentTypeByteSize(indicesAccessor.componentType)));
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));
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)));
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]);
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)));
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]);
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)));
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]);
603 if (gltfBufferType ==
"COLOR_0") {
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)));
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];
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];
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];
635 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": JOINTS_0: Invalid attributes component: " + to_string(attributeAccessor.componentType) +
", with size: " + to_string(
getComponentTypeByteSize(attributeAccessor.componentType)));
639 Console::println(
"GLTFReader::parseNode(): " + node->getId() +
": Invalid buffer type: " + gltfBufferType);
643 FacesEntity facesEntity(node, node->getId() +
"-" + to_string(facesEntityIdx));
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++) {
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]
659 for (
auto i = 0; i < indices.size() / 3; i++) {
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]
670 facesEntities.push_back(facesEntity);
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));
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)));
687 inverseBindMatricesBufferData = (
const float*)(inverseBindMatricesBuffer.data.data() + inverseBindMatricesAccessor.byteOffset + inverseBindMatricesBufferView.byteOffset);
689 if (inverseBindMatricesBufferData !=
nullptr) {
693 vector<Joint> skinningJoints(gltfSkin.joints.size());
694 for (
auto gltfJointNodeIdx: gltfSkin.joints) {
695 Joint joint(gltfModel.nodes[gltfJointNodeIdx].name);
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]
716 skinningJoints[i++] = joint;
718 skinning->setJoints(skinningJoints);
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++) {
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]);
732 skinning->setWeights(skinningWeights);
733 skinning->setVerticesJointsWeights(skinningJointWeights);
735 node->setSkinning(skinning);
740 node->setVertices(vertices);
741 node->setNormals(normals);
742 node->setTextureCoordinates(textureCoordinates);
743 node->setFacesEntities(facesEntities);
746 if (vertices.empty() ==
false && normals.empty() ==
false) {
747 ModelTools::createTangentsAndBitangents(node);
755 for (
auto gltfNodeIdx: gltfNodeChildrenIdx) {
756 auto& gltfNode = gltfModel.nodes[gltfNodeIdx];
757 auto node =
parseNode(pathName, gltfModel, gltfNodeIdx, parentNode->
getModel(), parentNode);
760 Console::println(
"GLTFReader::parseNodeChildren(): node already exists: " + node->getId());
763 if (gltfNode.children.empty() ==
false)
parseNodeChildren(pathName, gltfModel, gltfNode.children, node);
786 return imageName +
".png";
static string getComponentTypeString(int type)
static string getTypeString(int type)
static size_t getComponentTypeByteSize(int type)
static string determineTextureFileName(const string &imageName)
Determine texture file name.
static Node * parseNode(const string &pathName, const tinygltf::Model &gltfModel, int gltfNodeIdx, Model *model, Node *parentNode)
Parse GLTF node.
static void parseNodeChildren(const string &pathName, const tinygltf::Model &gltfModel, const vector< int > &gltfNodeChildrenIdx, Node *parentNode)
Parse GLTF node children into TDME node.
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.
Color 4 base definition class.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Node faces entity A node can have multiple entities containing faces and a applied material.
void setMaterial(Material *material)
Set up the entity's material.
void setFaces(const vector< Face > &faces)
Set up entity's faces.
void setBindMatrix(const Matrix4x4 &bindMatrix)
Bind matrix.
void setSpecularMaterialProperties(SpecularMaterialProperties *specularMaterialProperties)
Set specular material properties.
void setPBRMaterialProperties(PBRMaterialProperties *pbrMaterialProperties)
Set PBR material properties.
const string & getId() const
void setDoubleSided(bool doubleSided)
Set double sided.
Representation of a 3d model.
map< string, Node * > & getNodes()
Returns all object's nodes.
map< string, Material * > & getMaterials()
Returns all object materials.
map< string, Node * > & getSubNodes()
Represents specular material properties.
Represents rotation orders of a model.
Skinning definition for nodes.
Represents specular material properties.
Class representing texture UV coordinates data.
Matrix4x4 & identity()
Setup identity matrix.
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.
Matrix4x4 & translate(const Vector3 &v)
Sets up a translation matrix.
Matrix4x4 & scale(float s)
Scales this matrix.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Matrix4x4 computeMatrix() const
Computes a matrix from given.
Quaternion & set(float x, float y, float z, float w)
Set up this quaternion by components.
File system singleton class.
std::exception Exception
Exception base class.