TDME2 1.9.121
FBXReader.cpp
Go to the documentation of this file.
2
3#define FBXSDK_SHARED
4
5#include <fbxsdk.h>
6
7#include <map>
8#include <string>
9#include <vector>
10
11#include <tdme/tdme.h>
26#include <tdme/math/Math.h>
28#include <tdme/math/Vector2.h>
29#include <tdme/math/Vector3.h>
35
36using std::map;
37using std::string;
38using std::to_string;
39using std::vector;
40
63
64const Color4 FBXReader::BLENDER_AMBIENT_NONE(0.0f, 0.0f, 0.0f, 1.0f);
65
66Model* FBXReader::read(const string& pathName, const string& fileName) {
67 // init fbx sdk
68 FbxManager* fbxManager = NULL;
69 FbxScene* fbxScene = NULL;
70 fbxManager = FbxManager::Create();
71 if (fbxManager == nullptr) {
72 Console::println("FBXReader::read(): Unable to create FBX manager.");
73 return nullptr;
74 } else {
75 Console::println(string("FBXReader::read(): Autodesk FBX SDK version ") + string(fbxManager->GetVersion()));
76 }
77
78 Console::println("FBXReader::read(): reading FBX scene");
79
80 FbxIOSettings* ios = FbxIOSettings::Create(fbxManager, IOSROOT);
81 fbxManager->SetIOSettings(ios);
82 FbxString lPath = FbxGetApplicationDirectory();
83 fbxManager->LoadPluginsDirectory(lPath.Buffer());
84 fbxScene = FbxScene::Create(fbxManager, "My Scene");
85 if (fbxScene == nullptr) {
86 throw ModelFileIOException("FBXReader::read(): Error: Unable to create FBX scene");
87 }
88
89 // create import and import scene
90 FbxImporter* fbxImporter = FbxImporter::Create(fbxManager, "");
91 bool fbxImportStatus = fbxImporter->Initialize((pathName + "/" + fileName).c_str(), -1, fbxManager->GetIOSettings());
92 if (fbxImportStatus == false) {
93 throw ModelFileIOException("FBXReader::read(): Error: Unable to import FBX scene from '" + pathName + "/" + fileName);
94 }
95 // import the scene
96 fbxImportStatus = fbxImporter->Import(fbxScene);
97 if (fbxImportStatus == false) {
98 throw ModelFileIOException("FBXReader::read(): Error: Unable to import FBX scene from '" + pathName + "/" + fileName + " into scene");
99 }
100
101 Console::println("FBXReader::read(): Authoring program: " + string(fbxScene->GetDocumentInfo()->Original_ApplicationName.Get().Buffer()));
102
103 Console::println("FBXReader::read(): triangulating FBX");
104 // triangulate
105 FbxGeometryConverter fbxGeometryConverter(fbxManager);
106 fbxGeometryConverter.Triangulate(fbxScene, true);
107
108 Console::println("FBXReader::read(): importing FBX");
109
110 // create model
111 auto model = new Model(
112 fileName,
113 fileName,
114 getSceneUpVector(fbxScene),
115 getSceneRotationOrder(fbxScene),
116 nullptr,
117 string(fbxScene->GetDocumentInfo()->Original_ApplicationName.Get().Buffer()).find("Blender") != -1?
118 Model::AUTHORINGTOOL_BLENDER:
119 Model::AUTHORINGTOOL_UNKNOWN
120 );
121
122 // set up model import matrix
124 setupModelScaleRotationMatrix(fbxScene, model);
125
126 // process nodes
127 processScene(fbxScene, model, pathName);
128
129 //
130 Console::println("FBXReader::read(): setting up animations");
131
132 // parse animations stacks
133 FbxTime::SetGlobalTimeMode(FbxTime::eCustom, 30.0);
134 FbxArray<FbxString*> fbxAnimStackNameArray;
135 fbxScene->FillAnimStackNameArray(fbxAnimStackNameArray);
136 int framesTotal = 0;
137 for(auto i = 0; i < fbxAnimStackNameArray.GetCount(); i++) {
138 FbxTime fbxStartTime, fbxEndTime;
139 auto fbxCurrentTakeInfo = fbxScene->GetTakeInfo(*(fbxAnimStackNameArray[i]));
140 if (fbxCurrentTakeInfo != nullptr) {
141 fbxStartTime = fbxCurrentTakeInfo->mLocalTimeSpan.GetStart();
142 fbxEndTime = fbxCurrentTakeInfo->mLocalTimeSpan.GetStop();
143 } else {
144 FbxTimeSpan fbxTimeLineTimeSpan;
145 fbxScene->GetGlobalSettings().GetTimelineDefaultTimeSpan(fbxTimeLineTimeSpan);
146 fbxStartTime = fbxTimeLineTimeSpan.GetStart();
147 fbxEndTime = fbxTimeLineTimeSpan.GetStop();
148 }
149 int startFrame = (int)Math::ceil(fbxStartTime.GetMilliSeconds() / (1000.0f * 1.0f / 30.0f));
150 int endFrame = (int)Math::ceil(fbxEndTime.GetMilliSeconds() / (1000.0f * 1.0f / 30.0f)) - 1;
151 framesTotal+= endFrame - startFrame + 1;
152 }
153 model->addAnimationSetup(
154 Model::ANIMATIONSETUP_DEFAULT,
155 0,
156 framesTotal,
157 true
158 );
159 int frameOffset = 0;
160 for(auto i = 0; i < fbxAnimStackNameArray.GetCount(); i++ ) {
161 auto fbxCurrentAnimationStack = fbxScene->FindMember<FbxAnimStack>(fbxAnimStackNameArray[i]->Buffer());
162 auto fbxCurrentTakeInfo = fbxScene->GetTakeInfo(*(fbxAnimStackNameArray[i]));
163 FbxTime fbxStartTime, fbxEndTime;
164 if (fbxCurrentTakeInfo != nullptr) {
165 fbxStartTime = fbxCurrentTakeInfo->mLocalTimeSpan.GetStart();
166 fbxEndTime = fbxCurrentTakeInfo->mLocalTimeSpan.GetStop();
167 } else {
168 FbxTimeSpan fbxTimeLineTimeSpan;
169 fbxScene->GetGlobalSettings().GetTimelineDefaultTimeSpan(fbxTimeLineTimeSpan);
170 fbxStartTime = fbxTimeLineTimeSpan.GetStart();
171 fbxEndTime = fbxTimeLineTimeSpan.GetStop();
172 }
173 int startFrame = (int)Math::ceil(fbxStartTime.GetMilliSeconds() / (1000.0f * 1.0f / 30.0f));
174 int endFrame = (int)Math::ceil(fbxEndTime.GetMilliSeconds() / (1000.0f * 1.0f / 30.0f)) - 1;
175 model->addAnimationSetup(
176 string(fbxAnimStackNameArray[i]->Buffer()),
177 frameOffset + startFrame,
178 frameOffset + endFrame,
179 false
180 );
181 fbxScene->SetCurrentAnimationStack(fbxCurrentAnimationStack);
182 FbxNode* fbxNode = fbxScene->GetRootNode();
183 if (fbxNode == nullptr) continue;
184 for(auto i = 0; i < fbxNode->GetChildCount(); i++) {
185 processAnimation(fbxNode->GetChild(i), fbxStartTime, fbxEndTime, model, frameOffset);
186 }
187 frameOffset+= endFrame - startFrame + 1;
188 }
189 FbxArrayDelete(fbxAnimStackNameArray);
190
191 //
192 Console::println("FBXReader::read(): destroying FBX SDK");
193
194 // destroy the importer
195 if (fbxImporter != nullptr) fbxImporter->Destroy();
196
197 // destroy fbx manager
198 if (fbxManager != nullptr) fbxManager->Destroy();
199
200 Console::println("FBXReader::read(): prepare for indexed rendering");
201
202 //
203 ModelTools::setupJoints(model);
204 ModelTools::fixAnimationLength(model);
205 ModelTools::prepareForIndexedRendering(model);
206
207 Console::println("FBXReader::read(): done");
208
209 //
210 return model;
211}
212
214 auto upVector = getSceneUpVector(fbxScene);
215
216 // take rotation order from root node now
217 FbxNode* fbxNode = fbxScene->GetRootNode();
218 EFbxRotationOrder fbxRotationOrder;
219 fbxNode->GetRotationOrder(FbxNode::eSourcePivot, fbxRotationOrder);
220 if (fbxRotationOrder == eEulerXYZ) {
221 if (upVector == UpVector::Y_UP) {
222 return RotationOrder::ZYX;
223 } else
224 if (upVector == UpVector::Z_UP) {
225 return RotationOrder::YZX;
226 } else {
227 throw ModelFileIOException("Unknown Up vector");
228 }
229 } else {
230 throw ModelFileIOException("Not supported rotation order(" + to_string(fbxRotationOrder) + ")");
231 }
232}
233
235 int fbxUpVectorSign;
236 auto fbxUpVector = fbxScene->GetGlobalSettings().GetAxisSystem().GetUpVector(fbxUpVectorSign);
237 switch (fbxUpVector) {
238 case FbxAxisSystem::eXAxis:
239 throw ModelFileIOException("X-Up is not supported");
240 case FbxAxisSystem::eYAxis:
241 return UpVector::Y_UP;
242 case FbxAxisSystem::eZAxis:
243 return UpVector::Z_UP;
244 default:
245 throw ModelFileIOException("Unknown Up vector");
246 }
247 return UpVector::Y_UP;
248}
249
251 if (model->getUpVector() == UpVector::Y_UP) {
252 // no op
253 } else
254 if (model->getUpVector() == UpVector::Z_UP) {
255 model->setImportTransformationsMatrix(model->getImportTransformationsMatrix().clone().rotate(Vector3(1.0f, 0.0f, 0.0f), -90.0));
256 }
257}
258
259void FBXReader::setupModelScaleRotationMatrix(FbxScene* fbxScene, Model* model) {
260 FbxSystemUnit fbxSceneSystemUnit = fbxScene->GetGlobalSettings().GetSystemUnit();
261 model->setImportTransformationsMatrix(model->getImportTransformationsMatrix().clone().scale(static_cast<float>(fbxSceneSystemUnit.GetConversionFactorTo(FbxSystemUnit::m))));
262}
263
264void FBXReader::processScene(FbxScene* fbxScene, Model* model, const string& pathName) {
265 FbxNode* fbxNode = fbxScene->GetRootNode();
266 if (fbxNode == nullptr) return;
267 for(auto i = 0; i < fbxNode->GetChildCount(); i++) {
268 processNode(fbxNode->GetChild(i), model, nullptr, pathName);
269 }
270}
271
272void FBXReader::processNode(FbxNode* fbxNode, Model* model, Node* parentNode, const string& pathName) {
273 Node* node = nullptr;
274 if (fbxNode->GetNodeAttribute() != nullptr) {
275 auto fbxAttributeType = fbxNode->GetNodeAttribute()->GetAttributeType();
276 switch (fbxAttributeType) {
277 case FbxNodeAttribute::eMesh:
278 {
279 node = processMeshNode(fbxNode, model, parentNode, pathName);
280 break;
281 }
282 case FbxNodeAttribute::eSkeleton:
283 {
284 node = processSkeletonNode(fbxNode, model, parentNode, pathName);
285 break;
286 }
287 default:
288 {
289 break;
290 }
291 }
292 }
293 if (node == nullptr) {
294 auto fbxNodeName = fbxNode->GetName();
295 node = new Node(model, parentNode, fbxNodeName, fbxNodeName);
296 }
297 FbxAMatrix& fbxNodeLocalTransform = fbxNode->EvaluateLocalTransform();
299 Matrix4x4(
300 fbxNodeLocalTransform.Get(0,0),
301 fbxNodeLocalTransform.Get(0,1),
302 fbxNodeLocalTransform.Get(0,2),
303 fbxNodeLocalTransform.Get(0,3),
304 fbxNodeLocalTransform.Get(1,0),
305 fbxNodeLocalTransform.Get(1,1),
306 fbxNodeLocalTransform.Get(1,2),
307 fbxNodeLocalTransform.Get(1,3),
308 fbxNodeLocalTransform.Get(2,0),
309 fbxNodeLocalTransform.Get(2,1),
310 fbxNodeLocalTransform.Get(2,2),
311 fbxNodeLocalTransform.Get(2,3),
312 fbxNodeLocalTransform.Get(3,0),
313 fbxNodeLocalTransform.Get(3,1),
314 fbxNodeLocalTransform.Get(3,2),
315 fbxNodeLocalTransform.Get(3,3)
316 )
317 );
318 if (parentNode == nullptr) {
319 model->getSubNodes()[node->getId()] = node;
320 } else {
321 parentNode->getSubNodes()[node->getId()] = node;
322 }
323 model->getNodes()[node->getId()] = node;
324 parentNode = node;
325 for(auto i = 0; i < fbxNode->GetChildCount(); i++) {
326 processNode(fbxNode->GetChild(i), model, parentNode, pathName);
327 }
328}
329
330Node* FBXReader::processMeshNode(FbxNode* fbxNode, Model* model, Node* parentNode, const string& pathName) {
331 auto fbxNodeName = fbxNode->GetName();
332 FbxMesh* fbxMesh = (FbxMesh*)fbxNode->GetNodeAttribute();
333
334 auto node = new Node(model, parentNode, fbxNodeName, fbxNodeName);
335 vector<Vector3> vertices;
336 vector<Vector3> normals;
337 vector<TextureCoordinate> textureCoordinates;
338 vector<Vector3> tangents;
339 vector<Vector3> bitangents;
340 vector<FacesEntity> facesEntities;
341 vector<Face> faces;
342 FacesEntity* facesEntity = nullptr;
343
344 int fbxVertexId = 0;
345 int fbxPolygonCount = fbxMesh->GetPolygonCount();
346
347 FbxVector4* fbxControlPoints = fbxMesh->GetControlPoints();
348 for (auto i = 0; i < fbxMesh->GetControlPointsCount(); i++) {
349 auto fbxControlPoint = fbxControlPoints[i];
350 vertices.push_back(Vector3(fbxControlPoint[0], fbxControlPoint[1], fbxControlPoint[2]));
351 }
352 for (auto l = 0; l < fbxMesh->GetElementUVCount() && l < 1; ++l) {
353 auto fbxUV = fbxMesh->GetElementUV(l);
354 for (int i = 0; i < fbxUV->GetDirectArray().GetCount(); i++) {
355 auto fbxUVArray = fbxUV->GetDirectArray().GetAt(i);
356 textureCoordinates.push_back(TextureCoordinate(fbxUVArray[0], fbxUVArray[1]));
357 }
358 }
359 for (auto l = 0; l < fbxMesh->GetElementNormalCount() && l < 1; ++l) {
360 auto fbxNormal = fbxMesh->GetElementNormal(l);
361 for (int i = 0; i < fbxNormal->GetDirectArray().GetCount(); i++) {
362 auto fbxNormalArray = fbxNormal->GetDirectArray().GetAt(i);
363 normals.push_back(Vector3(fbxNormalArray[0], fbxNormalArray[1], fbxNormalArray[2]));
364 }
365 }
366 for (auto l = 0; l < fbxMesh->GetElementTangentCount() && l < 1; ++l) {
367 auto fbxTangent = fbxMesh->GetElementTangent(l);
368 for (int i = 0; i < fbxTangent->GetDirectArray().GetCount(); i++) {
369 auto fbxTangentArray = fbxTangent->GetDirectArray().GetAt(i);
370 tangents.push_back(Vector3(fbxTangentArray[0], fbxTangentArray[1], fbxTangentArray[2]));
371 }
372 }
373 for (auto l = 0; l < fbxMesh->GetElementBinormalCount() && l < 1; ++l) {
374 auto fbxBinormal = fbxMesh->GetElementBinormal(l);
375 for (int i = 0; i < fbxBinormal->GetDirectArray().GetCount(); i++) {
376 auto fbxBinormalArray = fbxBinormal->GetDirectArray().GetAt(i);
377 bitangents.push_back(Vector3(fbxBinormalArray[0], fbxBinormalArray[1], fbxBinormalArray[2]));
378 }
379 }
380
381 for (auto i = 0; i < fbxPolygonCount; i++) {
382 FbxSurfaceMaterial* fbxMaterial = nullptr;
383 for (auto l = 0; l < fbxMesh->GetElementMaterialCount() & l < 1; l++) {
384 FbxGeometryElementMaterial* fbxMaterialElement = fbxMesh->GetElementMaterial(l);
385 if (fbxMaterialElement->GetMappingMode() == FbxGeometryElement::eAllSame) {
386 fbxMaterial = fbxMesh->GetNode()->GetMaterial(fbxMaterialElement->GetIndexArray().GetAt(0));
387 } else {
388 fbxMaterial = fbxMesh->GetNode()->GetMaterial(fbxMaterialElement->GetIndexArray().GetAt(i));
389 }
390 }
391 Material* material = nullptr;
392 if (fbxMaterial == nullptr) {
393 material = model->getMaterials()["tdme.nomaterial"];
394 if (material == nullptr) {
395 material = new Material("tdme.nomaterial");
396 model->getMaterials()[material->getId()] = material;
397 }
398 } else {
399 string fbxMaterialName = fbxMaterial->GetName();
400 material = model->getMaterials()[fbxMaterialName];
401 if (material == nullptr) {
402 material = new Material(fbxMaterialName);
403 auto specularMaterialProperties = new SpecularMaterialProperties();
404 if (fbxMaterial->GetClassId().Is(FbxSurfacePhong::ClassId)) {
405 FbxPropertyT<FbxDouble3> fbxColor3;
406 FbxPropertyT<FbxDouble> fbxTransparency;
407 FbxPropertyT<FbxDouble> fbxFactor;
408 FbxPropertyT<FbxDouble> fbxShininess;
409 FbxPropertyT<FbxDouble> fbxShininessFactor;
410 FbxPropertyT<FbxDouble> fbxReflection;
411 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Ambient;
412 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->AmbientFactor;
413 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true) {
414 specularMaterialProperties->setAmbientColor(
415 Color4(
416 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
417 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
418 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
419 1.0f
420 )
421 );
422 }
423 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Diffuse;
424 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->DiffuseFactor;
425 fbxTransparency = ((FbxSurfacePhong*)fbxMaterial)->TransparencyFactor;
426 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true && fbxTransparency.IsValid() == true)
427 specularMaterialProperties->setDiffuseColor(
428 Color4(
429 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
430 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
431 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
432 // TODO: I am not sure about this here, but it seem to work
433 (
434 1.0f - static_cast<float>(fbxTransparency) < Math::EPSILON?
435 1.0f:
436 1.0f - static_cast<float>(fbxTransparency)
437 )
438 )
439 );
440 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Emissive;
441 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->EmissiveFactor;
442 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true) {
443 specularMaterialProperties->setEmissionColor(
444 Color4(
445 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
446 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
447 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
448 1.0f
449 )
450 );
451 }
452 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Specular;
453 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->SpecularFactor;
454 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true) {
455 specularMaterialProperties->setSpecularColor(
456 Color4(
457 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
458 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
459 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
460 1.0f
461 )
462 );
463 }
464 fbxShininess = ((FbxSurfacePhong*)fbxMaterial)->Reflection;
465 fbxShininessFactor = ((FbxSurfacePhong*)fbxMaterial)->ReflectionFactor;
466 if (fbxShininess.IsValid() == true && fbxShininessFactor.IsValid() == true) {
467 specularMaterialProperties->setShininess(static_cast<float>(fbxShininess.Get() * fbxShininessFactor.Get()));
468 }
469 fbxReflection = ((FbxSurfacePhong*)fbxMaterial)->Reflection;
470 if (fbxReflection.IsValid() == true) {
471 specularMaterialProperties->setReflection(static_cast<float>(fbxReflection.Get()));
472 }
473 } else
474 if (fbxMaterial->GetClassId().Is(FbxSurfaceLambert::ClassId)) {
475 FbxPropertyT<FbxDouble3> fbxColor3;
476 FbxPropertyT<FbxDouble> fbxTransparency;
477 FbxPropertyT<FbxDouble> fbxFactor;
478 fbxColor3 = ((FbxSurfaceLambert*)fbxMaterial)->Ambient;
479 fbxFactor = ((FbxSurfaceLambert*)fbxMaterial)->AmbientFactor;
480 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true) {
481 specularMaterialProperties->setAmbientColor(
482 Color4(
483 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
484 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
485 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
486 1.0f
487 )
488 );
489 }
490 fbxColor3 = ((FbxSurfaceLambert*)fbxMaterial)->Diffuse;
491 fbxFactor = ((FbxSurfaceLambert*)fbxMaterial)->DiffuseFactor;
492 fbxTransparency = ((FbxSurfaceLambert*)fbxMaterial)->TransparencyFactor;
493 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true && fbxTransparency.IsValid() == true)
494 specularMaterialProperties->setDiffuseColor(
495 Color4(
496 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
497 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
498 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
499 // TODO: I am not sure about this here, but it seem to work
500 (
501 1.0f - static_cast<float>(fbxTransparency) < Math::EPSILON?
502 1.0f:
503 1.0f - static_cast<float>(fbxTransparency)
504 )
505 )
506 );
507 fbxColor3 = ((FbxSurfaceLambert*)fbxMaterial)->Emissive;
508 fbxFactor = ((FbxSurfaceLambert*)fbxMaterial)->EmissiveFactor;
509 if (fbxColor3.IsValid() == true && fbxFactor.IsValid() == true) {
510 specularMaterialProperties->setEmissionColor(
511 Color4(
512 static_cast<float>(fbxColor3.Get()[0] * fbxFactor.Get()),
513 static_cast<float>(fbxColor3.Get()[1] * fbxFactor.Get()),
514 static_cast<float>(fbxColor3.Get()[2] * fbxFactor.Get()),
515 1.0f
516 )
517 );
518 }
519 } else {
520 Console::println("FBXReader::processMeshNode(): unsupported material shading class: " + fbxMaterialName);
521 }
522 FbxProperty fbxProperty;
523 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sDiffuse);
524 string diffuseTextureFileName;
525 {
526 if (fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() > 0) {
527 auto texture = FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0));
528 diffuseTextureFileName = texture->GetFileName();
529 Matrix2D3x3 textureMatrix;
530 textureMatrix.identity();
531 textureMatrix.multiply(Matrix2D3x3().identity().scale(Vector2(texture->GetScaleU(), texture->GetScaleV())));
532 // TODO: not sure about texture rotation with 2D textures here and I have no test model for now
533 textureMatrix.multiply(Matrix2D3x3::rotateAroundTextureCenter(texture->GetRotationU()));
534 textureMatrix.multiply(Matrix2D3x3().identity().translate(Vector2(texture->GetTranslationU(), texture->GetTranslationV())));
535 material->setTextureMatrix(textureMatrix);
536 } else
537 if (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0) {
538 auto texture = FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0));
539 diffuseTextureFileName = texture->GetFileName();
540 Matrix2D3x3 textureMatrix;
541 textureMatrix.identity();
542 textureMatrix.multiply(Matrix2D3x3().identity().scale(Vector2(texture->GetScaleU(), texture->GetScaleV())));
543 // TODO: not sure about texture rotation with 2D textures here and I have no test model for now
544 textureMatrix.multiply(Matrix2D3x3::rotateAroundTextureCenter(texture->GetRotationU()));
545 textureMatrix.multiply(Matrix2D3x3().identity().translate(Vector2(texture->GetTranslationU(), texture->GetTranslationV())));
546 material->setTextureMatrix(textureMatrix);
547 }
548 }
549 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sTransparentColor);
550 string diffuseTransparencyTextureFileName =
551 fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() == 0?
552 (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0?
553 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0))->GetFileName():
554 ""
555 ):
556 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0))->GetFileName();
557 if (diffuseTransparencyTextureFileName.length() == 0) {
558 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sTransparencyFactor);
559 diffuseTransparencyTextureFileName =
560 fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() == 0?
561 (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0?
562 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0))->GetFileName():
563 ""
564 ):
565 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0))->GetFileName();
566 }
567 if (diffuseTextureFileName.length() > 0) {
568 specularMaterialProperties->setDiffuseTexture(
569 FileSystem::getInstance()->fileExists(
570 FileSystem::getInstance()->getCanonicalPath(pathName, FileSystem::getInstance()->getFileName(diffuseTextureFileName))
571 )?pathName:FileSystem::getInstance()->getPathName(diffuseTextureFileName),
572 FileSystem::getInstance()->getFileName(diffuseTextureFileName),
573 FileSystem::getInstance()->fileExists(
574 FileSystem::getInstance()->getCanonicalPath(pathName, FileSystem::getInstance()->getFileName(diffuseTransparencyTextureFileName))
575 )?pathName:FileSystem::getInstance()->getPathName(diffuseTransparencyTextureFileName),
576 FileSystem::getInstance()->getFileName(diffuseTransparencyTextureFileName)
577 );
578 if (specularMaterialProperties->hasDiffuseTextureTransparency() == true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(true);
579 }
580 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sNormalMap);
581 string normalTextureFileName =
582 fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() == 0?
583 (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0?
584 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0))->GetFileName():
585 ""
586 ):
587 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0))->GetFileName();
588 if (normalTextureFileName.length() > 0) {
589 specularMaterialProperties->setNormalTexture(
590 FileSystem::getInstance()->fileExists(
591 FileSystem::getInstance()->getCanonicalPath(pathName, FileSystem::getInstance()->getFileName(normalTextureFileName))
592 )?pathName:FileSystem::getInstance()->getPathName(normalTextureFileName),
593 FileSystem::getInstance()->getFileName(normalTextureFileName)
594 );
595 }
596 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sSpecular);
597 string specularTextureFileName =
598 fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() == 0?
599 (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0?
600 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0))->GetFileName():
601 ""
602 ):
603 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0))->GetFileName();
604 if (specularTextureFileName.length() > 0) {
605 specularMaterialProperties->setSpecularTexture(
606 FileSystem::getInstance()->fileExists(
607 FileSystem::getInstance()->getCanonicalPath(pathName, FileSystem::getInstance()->getFileName(specularTextureFileName))
608 )?pathName:FileSystem::getInstance()->getPathName(specularTextureFileName),
609 FileSystem::getInstance()->getFileName(specularTextureFileName)
610 );
611 }
612 // adjust ambient light with blender
613 if (model->getAuthoringTool() == Model::AUTHORINGTOOL_BLENDER && specularMaterialProperties->getAmbientColor().equals(BLENDER_AMBIENT_NONE)) {
614 specularMaterialProperties->setAmbientColor(
615 Color4(
616 specularMaterialProperties->getDiffuseColor().getRed() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
617 specularMaterialProperties->getDiffuseColor().getGreen() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
618 specularMaterialProperties->getDiffuseColor().getBlue() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
619 1.0f
620 )
621 );
622 specularMaterialProperties->setDiffuseColor(
623 Color4(
624 specularMaterialProperties->getDiffuseColor().getRed() * BLENDER_DIFFUSE_SCALE,
625 specularMaterialProperties->getDiffuseColor().getGreen() * BLENDER_DIFFUSE_SCALE,
626 specularMaterialProperties->getDiffuseColor().getBlue() * BLENDER_DIFFUSE_SCALE,
627 specularMaterialProperties->getDiffuseColor().getAlpha()
628 )
629 );
630 }
631 material->setSpecularMaterialProperties(specularMaterialProperties);
632 model->getMaterials()[material->getId()] = material;
633 }
634 }
635 auto foundFacesEntity = false;
636 string facesEntityName = "facesentity-" + material->getId();
637 for (auto& facesEntityLookUp: facesEntities) {
638 if (facesEntityLookUp.getId() == facesEntityName) {
639 if (&facesEntityLookUp != facesEntity) {
640 if (facesEntity != nullptr) {
641 facesEntity->setFaces(faces);
642 }
643 faces = facesEntityLookUp.getFaces();
644 facesEntity = &facesEntityLookUp;
645 }
646 foundFacesEntity = true;
647 break;
648 }
649 }
650 if (foundFacesEntity == false) {
651 if (facesEntity != nullptr) {
652 facesEntity->setFaces(faces);
653 faces.clear();
654 }
655 facesEntities.push_back(FacesEntity(node, facesEntityName));
656 facesEntity = &facesEntities[facesEntities.size() - 1];
657 facesEntity->setMaterial(material);
658 }
659 auto fbxPolygonSize = fbxMesh->GetPolygonSize(i);
660 if (fbxPolygonSize != 3) throw ModelFileIOException("we only support triangles in '" + node->getName() + "'");
661 int controlPointIndicesIdx = 0;
662 array<int, 3> controlPointIndices;
663 int textureCoordinateIndicesIdx = 0;
664 array<int, 3> textureCoordinateIndices;
665 int normalIndicesIdx = 0;
666 array<int, 3> normalIndices;
667 int tangentIndicesIdx = 0;
668 array<int, 3> tangentIndices;
669 int binormalIndicesIdx = 0;
670 array<int, 3> binormalIndices;
671 for (auto j = 0; j < fbxPolygonSize; j++) {
672 int fbxControlPointIndex = fbxMesh->GetPolygonVertex(i, j);
673 controlPointIndices[controlPointIndicesIdx++] = fbxControlPointIndex;
674 for (auto l = 0; l < fbxMesh->GetElementUVCount() && l < 1; ++l) {
675 FbxGeometryElementUV* fbxUV = fbxMesh->GetElementUV(l);
676 switch (fbxUV->GetMappingMode()) {
677 case FbxGeometryElement::eByControlPoint:
678 switch (fbxUV->GetReferenceMode()) {
679 case FbxGeometryElement::eDirect:
680 {
681 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxControlPointIndex;
682 break;
683 }
684 case FbxGeometryElement::eIndexToDirect:
685 {
686 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxUV->GetIndexArray().GetAt(fbxControlPointIndex);
687 break;
688 }
689 default:
690 break;
691 }
692 break;
693 case FbxGeometryElement::eByPolygonVertex:
694 {
695 switch (fbxUV->GetReferenceMode()) {
696 case FbxGeometryElement::eDirect:
697 case FbxGeometryElement::eIndexToDirect:
698 {
699 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxMesh->GetTextureUVIndex(i, j);
700 break;
701 }
702 default:
703 break;
704 }
705 break;
706 }
707 default:
708 break;
709 }
710 }
711 for (auto l = 0; l < fbxMesh->GetElementNormalCount() && l < 1; ++l) {
712 FbxGeometryElementNormal* fbxNormal = fbxMesh->GetElementNormal(l);
713 if (fbxNormal->GetMappingMode() == FbxGeometryElement::eByPolygonVertex) {
714 switch (fbxNormal->GetReferenceMode()) {
715 case FbxGeometryElement::eDirect:
716 {
717 normalIndices[normalIndicesIdx++] = fbxVertexId;
718 break;
719 }
720 case FbxGeometryElement::eIndexToDirect:
721 {
722 normalIndices[normalIndicesIdx++] = fbxNormal->GetIndexArray().GetAt(fbxVertexId);
723 break;
724 }
725 default:
726 break;
727 }
728 } else {
729 }
730 }
731 for (auto l = 0; l < fbxMesh->GetElementTangentCount() && l < 1; ++l) {
732 FbxGeometryElementTangent* fbxTangent = fbxMesh->GetElementTangent(l);
733 if (fbxTangent->GetMappingMode() == FbxGeometryElement::eByPolygonVertex) {
734 switch (fbxTangent->GetReferenceMode()) {
735 case FbxGeometryElement::eDirect:
736 {
737 tangentIndices[tangentIndicesIdx++] = fbxVertexId;
738 break;
739 }
740 case FbxGeometryElement::eIndexToDirect:
741 {
742 tangentIndices[tangentIndicesIdx++] = fbxTangent->GetIndexArray().GetAt(fbxVertexId);
743 break;
744 }
745 default:
746 break;
747 }
748 }
749 }
750 for (auto l = 0; l < fbxMesh->GetElementBinormalCount() && l < 1; ++l) {
751 FbxGeometryElementBinormal* fbxBinormal = fbxMesh->GetElementBinormal(l);
752 if (fbxBinormal->GetMappingMode() == FbxGeometryElement::eByPolygonVertex) {
753 switch (fbxBinormal->GetReferenceMode()) {
754 case FbxGeometryElement::eDirect:
755 {
756 binormalIndices[binormalIndicesIdx++] = fbxVertexId;
757 break;
758 }
759 case FbxGeometryElement::eIndexToDirect:
760 {
761 binormalIndices[binormalIndicesIdx++] = fbxBinormal->GetIndexArray().GetAt(fbxVertexId);
762 break;
763 }
764 default:
765 break;
766 }
767 }
768 }
769 fbxVertexId++;
770 }
771 Face f(
772 node,
773 controlPointIndices[0],
774 controlPointIndices[1],
775 controlPointIndices[2],
776 normalIndices[0],
777 normalIndices[1],
778 normalIndices[2]
779 );
780 if (textureCoordinateIndicesIdx == 3) {
782 textureCoordinateIndices[0],
783 textureCoordinateIndices[1],
784 textureCoordinateIndices[2]
785 );
786 }
787 if (tangentIndicesIdx == 3 && binormalIndicesIdx == 3) {
789 tangentIndices[0],
790 tangentIndices[1],
791 tangentIndices[2]
792 );
794 binormalIndices[0],
795 binormalIndices[1],
796 binormalIndices[2]
797 );
798 }
799 faces.push_back(f);
800 }
801 if (facesEntity != nullptr) {
802 facesEntity->setFaces(faces);
803 }
804
805 node->setVertices(vertices);
806 node->setNormals(normals);
807 if (tangents.size() > 0 && bitangents.size() > 0) {
808 node->setTangents(tangents);
809 node->setBitangents(bitangents);
810 }
811 if (textureCoordinates.size() > 0) node->setTextureCoordinates(textureCoordinates);
812 node->setFacesEntities(facesEntities);
813
814 int fbxSkinCount = fbxNode->GetMesh()->GetDeformerCount(FbxDeformer::eSkin);
815 if (fbxSkinCount == 0) {
816 // no op
817 } else
818 if (fbxSkinCount == 1) {
819 FbxSkin* fbxSkinDeformer = (FbxSkin*)fbxNode->GetMesh()->GetDeformer(0, FbxDeformer::eSkin);
820 int fbxClusterCount = fbxSkinDeformer->GetClusterCount();
821 auto skinning = new Skinning();
822 vector<Joint> joints;
823 vector<float> weights;
824 map<int, vector<JointWeight>> jointWeightsByVertices;
825 for (auto fbxClusterIndex = 0; fbxClusterIndex < fbxClusterCount; fbxClusterIndex++) {
826 FbxCluster* fbxCluster = fbxSkinDeformer->GetCluster(fbxClusterIndex);
827 if (fbxCluster->GetLink() == nullptr) continue;
828 auto fbxJointName = fbxCluster->GetLink()->GetName();
829 auto jointIndex = joints.size();
830 FbxAMatrix transformMatrix;
831 FbxAMatrix transformLinkMatrix;
832 fbxCluster->GetTransformMatrix(transformMatrix);
833 fbxCluster->GetTransformLinkMatrix(transformLinkMatrix);
834 Joint joint(fbxJointName);
835 Matrix4x4 bindMatrix;
836 bindMatrix.set(
837 Matrix4x4(
838 transformMatrix.Get(0,0),
839 transformMatrix.Get(0,1),
840 transformMatrix.Get(0,2),
841 transformMatrix.Get(0,3),
842 transformMatrix.Get(1,0),
843 transformMatrix.Get(1,1),
844 transformMatrix.Get(1,2),
845 transformMatrix.Get(1,3),
846 transformMatrix.Get(2,0),
847 transformMatrix.Get(2,1),
848 transformMatrix.Get(2,2),
849 transformMatrix.Get(2,3),
850 transformMatrix.Get(3,0),
851 transformMatrix.Get(3,1),
852 transformMatrix.Get(3,2),
853 transformMatrix.Get(3,3)
854 )
855 );
856 bindMatrix.multiply(
857 Matrix4x4(
858 transformLinkMatrix.Get(0,0),
859 transformLinkMatrix.Get(0,1),
860 transformLinkMatrix.Get(0,2),
861 transformLinkMatrix.Get(0,3),
862 transformLinkMatrix.Get(1,0),
863 transformLinkMatrix.Get(1,1),
864 transformLinkMatrix.Get(1,2),
865 transformLinkMatrix.Get(1,3),
866 transformLinkMatrix.Get(2,0),
867 transformLinkMatrix.Get(2,1),
868 transformLinkMatrix.Get(2,2),
869 transformLinkMatrix.Get(2,3),
870 transformLinkMatrix.Get(3,0),
871 transformLinkMatrix.Get(3,1),
872 transformLinkMatrix.Get(3,2),
873 transformLinkMatrix.Get(3,3)
874 ).invert()
875 );
876 joint.setBindMatrix(bindMatrix);
877 joints.push_back(joint);
878 auto fbxClusterControlPointIndexCount = fbxCluster->GetControlPointIndicesCount();
879 auto fbxClusterControlPointIndices = fbxCluster->GetControlPointIndices();
880 for (auto fbxClusterControlPointIndex = 0; fbxClusterControlPointIndex < fbxClusterControlPointIndexCount; fbxClusterControlPointIndex++) {
881 int fbxControlPointIndex = fbxClusterControlPointIndices[fbxClusterControlPointIndex];
882 auto weightIndex = weights.size();
883 weights.push_back(fbxCluster->GetControlPointWeights()[fbxClusterControlPointIndex]);
884 jointWeightsByVertices[fbxControlPointIndex].push_back(JointWeight(jointIndex, weightIndex));
885 }
886 }
887 skinning->setJoints(joints);
888 skinning->setWeights(weights);
889 vector<vector<JointWeight>> verticesJointsWeights;
890 for (auto vertexIndex = 0; vertexIndex < vertices.size(); vertexIndex++) {
891 verticesJointsWeights.push_back(vector<JointWeight>());
892 auto jointWeightsByVerticesIt = jointWeightsByVertices.find(vertexIndex);
893 if (jointWeightsByVerticesIt != jointWeightsByVertices.end()) {
894 for (auto& jointWeight: jointWeightsByVerticesIt->second) {
895 verticesJointsWeights[verticesJointsWeights.size() - 1].push_back(jointWeight);
896 }
897 }
898 }
899 skinning->setVerticesJointsWeights(verticesJointsWeights);
900 node->setSkinning(skinning);
901 } else {
902 Console::println("FBXReader::processMeshNode(): " + to_string(fbxSkinCount) + " skins per mesh: Not supported");
903 }
904
905 return node;
906}
907
908Node* FBXReader::processSkeletonNode(FbxNode* fbxNode, Model* model, Node* parentNode, const string& pathName) {
909 auto fbxNodeName = fbxNode->GetName();
910 return new Node(model, parentNode, fbxNodeName, fbxNodeName);
911}
912
913void FBXReader::processAnimation(FbxNode* fbxNode, const FbxTime& fbxStartFrame, const FbxTime& fbxEndFrame, Model* model, int frameOffset) {
914 auto fbxNodeName = fbxNode->GetName();
915 auto node = model->getNodeById(fbxNodeName);
916 auto animation = node->getAnimation();
917 if (node->getAnimation() == nullptr) {
918 animation = new Animation();
919 node->setAnimation(animation);
920 }
921 auto transformationsMatrices = node->getAnimation()->getTransformationsMatrices();
922 transformationsMatrices.resize(model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT)->getFrames());
923 FbxTime fbxFrameTime;
924 fbxFrameTime.SetMilliSeconds(1000.0f * 1.0f / 30.0f);
925 for(auto i = fbxStartFrame; i < fbxEndFrame; i+= fbxFrameTime) {
926 FbxAMatrix& fbxTransformationMatrix = fbxNode->EvaluateLocalTransform(i);
927 int frameIdx = frameOffset + (int)Math::ceil((i.GetMilliSeconds() - fbxStartFrame.GetMilliSeconds()) / (1000.0f * 1.0f / 30.0f));
928 transformationsMatrices[frameIdx].set(
929 fbxTransformationMatrix.Get(0,0),
930 fbxTransformationMatrix.Get(0,1),
931 fbxTransformationMatrix.Get(0,2),
932 fbxTransformationMatrix.Get(0,3),
933 fbxTransformationMatrix.Get(1,0),
934 fbxTransformationMatrix.Get(1,1),
935 fbxTransformationMatrix.Get(1,2),
936 fbxTransformationMatrix.Get(1,3),
937 fbxTransformationMatrix.Get(2,0),
938 fbxTransformationMatrix.Get(2,1),
939 fbxTransformationMatrix.Get(2,2),
940 fbxTransformationMatrix.Get(2,3),
941 fbxTransformationMatrix.Get(3,0),
942 fbxTransformationMatrix.Get(3,1),
943 fbxTransformationMatrix.Get(3,2),
944 fbxTransformationMatrix.Get(3,3)
945 );
946 }
947 node->getAnimation()->setTransformationsMatrices(transformationsMatrices);
948 for(auto i = 0; i < fbxNode->GetChildCount(); i++) {
949 processAnimation(fbxNode->GetChild(i), fbxStartFrame, fbxEndFrame, model, frameOffset);
950 }
951}
static constexpr float BLENDER_AMBIENT_FROM_DIFFUSE_SCALE
Definition: FBXReader.h:44
static UpVector * getSceneUpVector(FbxScene *fbxScene)
Get scene up vector.
Definition: FBXReader.cpp:234
static Node * processSkeletonNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX skeleton node.
Definition: FBXReader.cpp:908
static constexpr float BLENDER_DIFFUSE_SCALE
Definition: FBXReader.h:45
static Node * processMeshNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX mesh node.
Definition: FBXReader.cpp:330
static STATIC_DLL_IMPEXT const Color4 BLENDER_AMBIENT_NONE
Definition: FBXReader.h:43
static void processNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX node.
Definition: FBXReader.cpp:272
static void setupModelScaleRotationMatrix(FbxScene *fbxScene, Model *model)
Set up model import scale maxtrix.
Definition: FBXReader.cpp:259
static void setupModelImportRotationMatrix(Model *model)
Set up model import rotation maxtrix.
Definition: FBXReader.cpp:250
static RotationOrder * getSceneRotationOrder(FbxScene *fbxScene)
Get scene rotation order.
Definition: FBXReader.cpp:213
static void processAnimation(FbxNode *fbxNode, const FbxTime &fbxStartFrame, const FbxTime &fbxEndFrame, Model *model, int frameOffset)
Process animation.
Definition: FBXReader.cpp:913
static void processScene(FbxScene *fbxScene, Model *model, const string &pathName)
Process FBX scene.
Definition: FBXReader.cpp:264
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
void setTextureCoordinateIndices(int32_t vt0, int32_t vt1, int32_t vt2)
Set up optional texture coordinate indices.
Definition: Face.cpp:45
void setTangentIndices(int32_t ti0, int32_t ti1, int32_t ti2)
Set tangent indices.
Definition: Face.cpp:52
void setBitangentIndices(int32_t bi0, int32_t bi1, int32_t bi2)
Set bitangent indices.
Definition: Face.cpp:59
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
const string & getId() const
Definition: Material.h:57
void setTextureMatrix(const Matrix2D3x3 &textureMatrix)
Set texture matrix.
Definition: Material.h:127
Representation of a 3d model.
Definition: Model.h:32
UpVector * getUpVector()
Definition: Model.h:133
const Matrix4x4 & getImportTransformationsMatrix()
Definition: Model.h:293
AuthoringTool getAuthoringTool()
Definition: Model.h:105
map< string, Node * > & getNodes()
Returns all object's nodes.
Definition: Model.h:179
AnimationSetup * getAnimationSetup(const string &id)
Definition: Model.cpp:116
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
Definition: Model.h:301
Node * getNodeById(const string &id)
Returns a node by given name or null.
Definition: Model.cpp:78
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
Definition: Model.h:194
map< string, Material * > & getMaterials()
Returns all object materials.
Definition: Model.h:171
Model node.
Definition: Node.h:31
const string & getId()
Returns id.
Definition: Node.h:85
void setTransformationsMatrix(const Matrix4x4 &transformationsMatrix)
Definition: Node.h:128
Animation * getAnimation()
Definition: Node.h:225
map< string, Node * > & getSubNodes()
Definition: Node.h:289
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
Standard math functions.
Definition: Math.h:21
3x3 2D Matrix class
Definition: Matrix2D3x3.h:22
Matrix2D3x3 & identity()
Setup identity matrix.
Definition: Matrix2D3x3.h:116
Matrix2D3x3 & multiply(const Matrix2D3x3 &m)
Multiplies this matrix with another matrix.
Definition: Matrix2D3x3.h:220
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & rotate(const Vector3 &axis, float angle)
Creates a rotation matrix.
Definition: Matrix4x4.h:473
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 clone()
Clones this matrix.
Definition: Matrix4x4.h:624
Matrix4x4 & invert()
Inverts the matrix.
Definition: Matrix4x4.h:536
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
2D vector 2 class
Definition: Vector2.h:19
3D vector 3 class
Definition: Vector3.h:22
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
Model tools functions class.
Definition: ModelTools.h:38