64const Color4 FBXReader::BLENDER_AMBIENT_NONE(0.0f, 0.0f, 0.0f, 1.0f);
66Model* FBXReader::read(
const string& pathName,
const string& fileName) {
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.");
75 Console::println(
string(
"FBXReader::read(): Autodesk FBX SDK version ") +
string(fbxManager->GetVersion()));
78 Console::println(
"FBXReader::read(): reading FBX scene");
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) {
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);
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");
101 Console::println(
"FBXReader::read(): Authoring program: " +
string(fbxScene->GetDocumentInfo()->Original_ApplicationName.Get().Buffer()));
103 Console::println(
"FBXReader::read(): triangulating FBX");
105 FbxGeometryConverter fbxGeometryConverter(fbxManager);
106 fbxGeometryConverter.Triangulate(fbxScene,
true);
108 Console::println(
"FBXReader::read(): importing FBX");
111 auto model =
new Model(
117 string(fbxScene->GetDocumentInfo()->Original_ApplicationName.Get().Buffer()).find(
"Blender") != -1?
118 Model::AUTHORINGTOOL_BLENDER:
119 Model::AUTHORINGTOOL_UNKNOWN
130 Console::println(
"FBXReader::read(): setting up animations");
133 FbxTime::SetGlobalTimeMode(FbxTime::eCustom, 30.0);
134 FbxArray<FbxString*> fbxAnimStackNameArray;
135 fbxScene->FillAnimStackNameArray(fbxAnimStackNameArray);
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();
144 FbxTimeSpan fbxTimeLineTimeSpan;
145 fbxScene->GetGlobalSettings().GetTimelineDefaultTimeSpan(fbxTimeLineTimeSpan);
146 fbxStartTime = fbxTimeLineTimeSpan.GetStart();
147 fbxEndTime = fbxTimeLineTimeSpan.GetStop();
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;
153 model->addAnimationSetup(
154 Model::ANIMATIONSETUP_DEFAULT,
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();
168 FbxTimeSpan fbxTimeLineTimeSpan;
169 fbxScene->GetGlobalSettings().GetTimelineDefaultTimeSpan(fbxTimeLineTimeSpan);
170 fbxStartTime = fbxTimeLineTimeSpan.GetStart();
171 fbxEndTime = fbxTimeLineTimeSpan.GetStop();
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,
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);
187 frameOffset+= endFrame - startFrame + 1;
189 FbxArrayDelete(fbxAnimStackNameArray);
192 Console::println(
"FBXReader::read(): destroying FBX SDK");
195 if (fbxImporter !=
nullptr) fbxImporter->Destroy();
198 if (fbxManager !=
nullptr) fbxManager->Destroy();
200 Console::println(
"FBXReader::read(): prepare for indexed rendering");
203 ModelTools::setupJoints(model);
204 ModelTools::fixAnimationLength(model);
205 ModelTools::prepareForIndexedRendering(model);
207 Console::println(
"FBXReader::read(): done");
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;
224 if (upVector == UpVector::Z_UP) {
225 return RotationOrder::YZX;
230 throw ModelFileIOException(
"Not supported rotation order(" + to_string(fbxRotationOrder) +
")");
236 auto fbxUpVector = fbxScene->GetGlobalSettings().GetAxisSystem().GetUpVector(fbxUpVectorSign);
237 switch (fbxUpVector) {
238 case FbxAxisSystem::eXAxis:
240 case FbxAxisSystem::eYAxis:
241 return UpVector::Y_UP;
242 case FbxAxisSystem::eZAxis:
243 return UpVector::Z_UP;
247 return UpVector::Y_UP;
260 FbxSystemUnit fbxSceneSystemUnit = fbxScene->GetGlobalSettings().GetSystemUnit();
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);
273 Node* node =
nullptr;
274 if (fbxNode->GetNodeAttribute() !=
nullptr) {
275 auto fbxAttributeType = fbxNode->GetNodeAttribute()->GetAttributeType();
276 switch (fbxAttributeType) {
277 case FbxNodeAttribute::eMesh:
282 case FbxNodeAttribute::eSkeleton:
293 if (node ==
nullptr) {
294 auto fbxNodeName = fbxNode->GetName();
295 node =
new Node(model, parentNode, fbxNodeName, fbxNodeName);
297 FbxAMatrix& fbxNodeLocalTransform = fbxNode->EvaluateLocalTransform();
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)
318 if (parentNode ==
nullptr) {
325 for(
auto i = 0; i < fbxNode->GetChildCount(); i++) {
326 processNode(fbxNode->GetChild(i), model, parentNode, pathName);
331 auto fbxNodeName = fbxNode->GetName();
332 FbxMesh* fbxMesh = (FbxMesh*)fbxNode->GetNodeAttribute();
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;
345 int fbxPolygonCount = fbxMesh->GetPolygonCount();
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]));
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);
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]));
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]));
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]));
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));
388 fbxMaterial = fbxMesh->GetNode()->GetMaterial(fbxMaterialElement->GetIndexArray().GetAt(i));
392 if (fbxMaterial ==
nullptr) {
394 if (material ==
nullptr) {
395 material =
new Material(
"tdme.nomaterial");
399 string fbxMaterialName = fbxMaterial->GetName();
401 if (material ==
nullptr) {
402 material =
new Material(fbxMaterialName);
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(
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()),
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(
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()),
434 1.0f -
static_cast<float>(fbxTransparency) < Math::EPSILON?
436 1.0f -
static_cast<float>(fbxTransparency)
440 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Emissive;
441 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->EmissiveFactor;
442 if (fbxColor3.IsValid() ==
true && fbxFactor.IsValid() ==
true) {
443 specularMaterialProperties->setEmissionColor(
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()),
452 fbxColor3 = ((FbxSurfacePhong*)fbxMaterial)->Specular;
453 fbxFactor = ((FbxSurfacePhong*)fbxMaterial)->SpecularFactor;
454 if (fbxColor3.IsValid() ==
true && fbxFactor.IsValid() ==
true) {
455 specularMaterialProperties->setSpecularColor(
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()),
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()));
469 fbxReflection = ((FbxSurfacePhong*)fbxMaterial)->Reflection;
470 if (fbxReflection.IsValid() ==
true) {
471 specularMaterialProperties->setReflection(
static_cast<float>(fbxReflection.Get()));
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(
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()),
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(
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()),
501 1.0f -
static_cast<float>(fbxTransparency) < Math::EPSILON?
503 1.0f -
static_cast<float>(fbxTransparency)
507 fbxColor3 = ((FbxSurfaceLambert*)fbxMaterial)->Emissive;
508 fbxFactor = ((FbxSurfaceLambert*)fbxMaterial)->EmissiveFactor;
509 if (fbxColor3.IsValid() ==
true && fbxFactor.IsValid() ==
true) {
510 specularMaterialProperties->setEmissionColor(
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()),
520 Console::println(
"FBXReader::processMeshNode(): unsupported material shading class: " + fbxMaterialName);
522 FbxProperty fbxProperty;
523 fbxProperty = fbxMaterial->FindProperty(FbxSurfaceMaterial::sDiffuse);
524 string diffuseTextureFileName;
526 if (fbxProperty.GetSrcObjectCount<FbxLayeredTexture>() > 0) {
527 auto texture = FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0));
528 diffuseTextureFileName = texture->GetFileName();
533 textureMatrix.
multiply(Matrix2D3x3::rotateAroundTextureCenter(texture->GetRotationU()));
537 if (fbxProperty.GetSrcObjectCount<FbxTexture>() > 0) {
538 auto texture = FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxTexture>(0));
539 diffuseTextureFileName = texture->GetFileName();
544 textureMatrix.
multiply(Matrix2D3x3::rotateAroundTextureCenter(texture->GetRotationU()));
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():
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():
565 FbxCast<FbxFileTexture>(fbxProperty.GetSrcObject<FbxLayeredTexture>(0))->GetFileName();
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)
578 if (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(
true);
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():
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)
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():
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)
614 specularMaterialProperties->setAmbientColor(
622 specularMaterialProperties->setDiffuseColor(
627 specularMaterialProperties->getDiffuseColor().getAlpha()
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) {
643 faces = facesEntityLookUp.getFaces();
644 facesEntity = &facesEntityLookUp;
646 foundFacesEntity =
true;
650 if (foundFacesEntity ==
false) {
651 if (facesEntity !=
nullptr) {
655 facesEntities.push_back(
FacesEntity(node, facesEntityName));
656 facesEntity = &facesEntities[facesEntities.size() - 1];
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:
681 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxControlPointIndex;
684 case FbxGeometryElement::eIndexToDirect:
686 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxUV->GetIndexArray().GetAt(fbxControlPointIndex);
693 case FbxGeometryElement::eByPolygonVertex:
695 switch (fbxUV->GetReferenceMode()) {
696 case FbxGeometryElement::eDirect:
697 case FbxGeometryElement::eIndexToDirect:
699 textureCoordinateIndices[textureCoordinateIndicesIdx++] = fbxMesh->GetTextureUVIndex(i, j);
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:
717 normalIndices[normalIndicesIdx++] = fbxVertexId;
720 case FbxGeometryElement::eIndexToDirect:
722 normalIndices[normalIndicesIdx++] = fbxNormal->GetIndexArray().GetAt(fbxVertexId);
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:
737 tangentIndices[tangentIndicesIdx++] = fbxVertexId;
740 case FbxGeometryElement::eIndexToDirect:
742 tangentIndices[tangentIndicesIdx++] = fbxTangent->GetIndexArray().GetAt(fbxVertexId);
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:
756 binormalIndices[binormalIndicesIdx++] = fbxVertexId;
759 case FbxGeometryElement::eIndexToDirect:
761 binormalIndices[binormalIndicesIdx++] = fbxBinormal->GetIndexArray().GetAt(fbxVertexId);
773 controlPointIndices[0],
774 controlPointIndices[1],
775 controlPointIndices[2],
780 if (textureCoordinateIndicesIdx == 3) {
782 textureCoordinateIndices[0],
783 textureCoordinateIndices[1],
784 textureCoordinateIndices[2]
787 if (tangentIndicesIdx == 3 && binormalIndicesIdx == 3) {
801 if (facesEntity !=
nullptr) {
805 node->setVertices(vertices);
806 node->setNormals(normals);
807 if (tangents.size() > 0 && bitangents.size() > 0) {
808 node->setTangents(tangents);
809 node->setBitangents(bitangents);
811 if (textureCoordinates.size() > 0) node->setTextureCoordinates(textureCoordinates);
812 node->setFacesEntities(facesEntities);
814 int fbxSkinCount = fbxNode->GetMesh()->GetDeformerCount(FbxDeformer::eSkin);
815 if (fbxSkinCount == 0) {
818 if (fbxSkinCount == 1) {
819 FbxSkin* fbxSkinDeformer = (FbxSkin*)fbxNode->GetMesh()->GetDeformer(0, FbxDeformer::eSkin);
820 int fbxClusterCount = fbxSkinDeformer->GetClusterCount();
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);
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)
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)
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));
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);
899 skinning->setVerticesJointsWeights(verticesJointsWeights);
900 node->setSkinning(skinning);
902 Console::println(
"FBXReader::processMeshNode(): " + to_string(fbxSkinCount) +
" skins per mesh: Not supported");
909 auto fbxNodeName = fbxNode->GetName();
910 return new Node(model, parentNode, fbxNodeName, fbxNodeName);
914 auto fbxNodeName = fbxNode->GetName();
917 if (node->getAnimation() ==
nullptr) {
919 node->setAnimation(animation);
921 auto transformationsMatrices = node->getAnimation()->getTransformationsMatrices();
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)
947 node->getAnimation()->setTransformationsMatrices(transformationsMatrices);
948 for(
auto i = 0; i < fbxNode->GetChildCount(); i++) {
949 processAnimation(fbxNode->GetChild(i), fbxStartFrame, fbxEndFrame, model, frameOffset);
static constexpr float BLENDER_AMBIENT_FROM_DIFFUSE_SCALE
static UpVector * getSceneUpVector(FbxScene *fbxScene)
Get scene up vector.
static Node * processSkeletonNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX skeleton node.
static constexpr float BLENDER_DIFFUSE_SCALE
static Node * processMeshNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX mesh node.
static STATIC_DLL_IMPEXT const Color4 BLENDER_AMBIENT_NONE
static void processNode(FbxNode *fbxNode, Model *model, Node *parentNode, const string &pathName)
Process FBX node.
static void setupModelScaleRotationMatrix(FbxScene *fbxScene, Model *model)
Set up model import scale maxtrix.
static void setupModelImportRotationMatrix(Model *model)
Set up model import rotation maxtrix.
static RotationOrder * getSceneRotationOrder(FbxScene *fbxScene)
Get scene rotation order.
static void processAnimation(FbxNode *fbxNode, const FbxTime &fbxStartFrame, const FbxTime &fbxEndFrame, Model *model, int frameOffset)
Process animation.
static void processScene(FbxScene *fbxScene, Model *model, const string &pathName)
Process FBX scene.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
void setTextureCoordinateIndices(int32_t vt0, int32_t vt1, int32_t vt2)
Set up optional texture coordinate indices.
void setTangentIndices(int32_t ti0, int32_t ti1, int32_t ti2)
Set tangent indices.
void setBitangentIndices(int32_t bi0, int32_t bi1, int32_t bi2)
Set bitangent indices.
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.
const string & getId() const
void setTextureMatrix(const Matrix2D3x3 &textureMatrix)
Set texture matrix.
Representation of a 3d model.
const Matrix4x4 & getImportTransformationsMatrix()
AuthoringTool getAuthoringTool()
map< string, Node * > & getNodes()
Returns all object's nodes.
AnimationSetup * getAnimationSetup(const string &id)
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
Node * getNodeById(const string &id)
Returns a node by given name or null.
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
map< string, Material * > & getMaterials()
Returns all object materials.
const string & getId()
Returns id.
void setTransformationsMatrix(const Matrix4x4 &transformationsMatrix)
Animation * getAnimation()
map< string, Node * > & getSubNodes()
Represents rotation orders of a model.
Skinning definition for nodes.
Represents specular material properties.
Class representing texture UV coordinates data.
Matrix2D3x3 & identity()
Setup identity matrix.
Matrix2D3x3 & multiply(const Matrix2D3x3 &m)
Multiplies this matrix with another matrix.
Matrix4x4 & rotate(const Vector3 &axis, float angle)
Creates a rotation 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 clone()
Clones this matrix.
Matrix4x4 & invert()
Inverts the matrix.
Matrix4x4 & scale(float s)
Scales this matrix.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
File system singleton class.