5#include <unordered_set>
46#define AVOID_NULLPTR_STRING(arg) (arg == nullptr?"":arg)
52using std::unordered_set;
94const Color4 DAEReader::BLENDER_AMBIENT_NONE(0.0f, 0.0f, 0.0f, 1.0f);
96Model* DAEReader::read(
const string& pathName,
const string& fileName)
99 auto xmlContent = FileSystem::getInstance()->getContentAsString(pathName, fileName);
101 xmlDocument.
Parse(xmlContent.c_str());
102 if (xmlDocument.
Error() ==
true) {
104 string(
"Could not parse XML. Error='") +
string(xmlDocument.
ErrorDesc()) +
string(
"'")
117 if (v == UpVector::Y_UP) {
118 rotationOrder = RotationOrder::ZYX;
120 if (v == UpVector::Z_UP) {
121 rotationOrder = RotationOrder::YZX;
126 auto model =
new Model(
143 xmlSceneId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceVisualscene->Attribute(
"url"))), 1);
148 for (
auto xmlLibraryVisualScene:
getChildrenByTagName(xmlLibraryVisualScenes,
"visual_scene")) {
150 if (xmlVisualSceneId == xmlSceneId) {
155 if (xmlExtraNodes.empty() ==
false) {
156 auto xmlExtraNode = xmlExtraNodes.at(0);
159 if (xmlFrameRateNodes.empty() ==
false) {
170 if (node !=
nullptr) {
171 model->getSubNodes()[node->getId()] = node;
172 model->getNodes()[node->getId()] = node;
177 if (ModelTools::hasDefaultAnimation(model) ==
false) ModelTools::createDefaultAnimation(model, 0);
179 ModelTools::setupJoints(model);
181 ModelTools::fixAnimationLength(model);
183 ModelTools::prepareForIndexedRendering(model);
194 return Model::AUTHORINGTOOL_BLENDER;
199 return Model::AUTHORINGTOOL_UNKNOWN;
208 if (StringTools::equalsIgnoreCase(upAxis,
"Y_UP") ==
true) {
209 return UpVector::Y_UP;
211 if (StringTools::equalsIgnoreCase(upAxis,
"Z_UP") ==
true) {
212 return UpVector::Z_UP;
214 if (StringTools::equalsIgnoreCase(upAxis,
"X_UP") ==
true) {
230 if (StringTools::equalsIgnoreCase(upAxis,
"Y_UP") ==
true) {
232 if (StringTools::equalsIgnoreCase(upAxis,
"Z_UP") ==
true) {
235 if (StringTools::equalsIgnoreCase(upAxis,
"X_UP") ==
true) {
238 Console::println(
string(
"Warning: Unknown up axis: " + upAxis));
251 float scaleFactor = Float::parse(tmp);
261 if (xmlInstanceControllers.empty() ==
false) {
264 return readNode(pathName, model, parentNode, xmlRoot, xmlNode, fps);
272 if (xmlNodeId.length() == 0) xmlNodeId = xmlNodeName;
275 auto node =
new Node(model, parentNode, xmlNodeId, xmlNodeName);
279 if (xmlMatrixElements.empty() ==
false) {
284 array<float, 16> transformationsMatrixArray;
285 for (
auto i = 0; i < transformationsMatrixArray.size(); i++) {
286 transformationsMatrixArray[i] = Float::parse(t.
nextToken());
288 transformationsMatrix.
set(transformationsMatrixArray).
transpose();
289 node->setTransformationsMatrix(transformationsMatrix);
294 if (xmlAnimationsLibrary.empty() ==
false) {
298 if (_xmlAnimation.empty() ==
false) {
299 xmlAnimation = _xmlAnimation.at(0);
302 string xmlSamplerSource;
304 if (StringTools::startsWith(
string(
AVOID_NULLPTR_STRING(xmlChannel->Attribute(
"target"))), xmlNodeId +
"/") ==
true) {
305 xmlSamplerSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlChannel->Attribute(
"source"))), 1);
308 if (xmlSamplerSource.length() == 0) {
312 string xmlSamplerOutputSource;
313 string xmlSamplerInputSource;
317 xmlSamplerOutputSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute(
"source"))), 1);
320 xmlSamplerInputSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute(
"source"))), 1);
324 if (xmlSamplerOutputSource.length() == 0) {
326 "Could not find xml sampler output source for animation for '" +
333 vector<float> keyFrameTimes;
335 if (
string(
AVOID_NULLPTR_STRING(xmlAnimationSource->Attribute(
"id"))) == xmlSamplerInputSource) {
339 auto keyFrameIdx = 0;
340 keyFrameTimes.resize(frames);
344 keyFrameTimes[keyFrameIdx++] = Float::parse(t.
nextToken());
350 if (keyFrameTimes.size() > 0) {
352 if (
string(
AVOID_NULLPTR_STRING(xmlAnimationSource->Attribute(
"id"))) == xmlSamplerOutputSource) {
354 auto keyFrames = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlFloatArray->Attribute(
"count")))) / 16;
361 int32_t keyFrameIdx = 0;
362 vector<Matrix4x4> keyFrameMatrices;
363 keyFrameMatrices.resize(keyFrames);
366 array<float, 16> keyFrameMatricesArray;
367 for (
auto i = 0; i < keyFrameMatricesArray.size() ;i++) {
368 keyFrameMatricesArray[i] = Float::parse(t.
nextToken());
370 keyFrameMatrices[keyFrameIdx].set(keyFrameMatricesArray);
371 keyFrameMatrices[keyFrameIdx].transpose();
375 auto frames =
static_cast< int32_t
>(Math::ceil(keyFrameTimes[keyFrameTimes.size() - 1] * fps));
377 ModelTools::createDefaultAnimation(model, frames);
379 vector<Matrix4x4> transformationsMatrices;
380 transformationsMatrices.resize(frames);
381 auto tansformationsMatrixLast = &keyFrameMatrices[0];
384 auto timeStampLast = 0.0f;
385 for (
auto keyFrameTime : keyFrameTimes) {
386 auto transformationsMatrixCurrent = &keyFrameMatrices[(keyFrameIdx) % keyFrameMatrices.size()];
388 for (timeStamp = timeStampLast; timeStamp < keyFrameTime; timeStamp += 1.0f / fps) {
389 if (frameIdx >= frames) {
395 transformationsMatrices[frameIdx] = Matrix4x4::interpolateLinear(*tansformationsMatrixLast, *transformationsMatrixCurrent, (timeStamp - timeStampLast) / (keyFrameTime - timeStampLast));
398 timeStampLast = timeStamp;
399 tansformationsMatrixLast = transformationsMatrixCurrent;
402 animation->setTransformationsMatrices(transformationsMatrices);
403 node->setAnimation(animation);
415 if (_node !=
nullptr) {
416 node->getSubNodes()[_node->getId()] = _node;
417 model->
getNodes()[_node->getId()] = _node;
422 string xmlInstanceGeometryId;
424 if (xmlInstanceGeometryElements.empty() ==
false) {
425 auto xmlInstanceGeometryElement = xmlInstanceGeometryElements.at(0);
427 xmlInstanceGeometryId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceGeometryElement->Attribute(
"url"))), 1);
429 map<string, string> materialSymbols;
437 readGeometry(pathName, model, node, xmlRoot, xmlInstanceGeometryId, materialSymbols);
442 string xmlInstanceNodeId;
444 xmlInstanceNodeId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceNodeElement->Attribute(
"url"))), 1);
447 if (xmlInstanceNodeId.length() > 0) {
454 if (_node !=
nullptr) {
455 node->getSubNodes()[_node->getId()] = _node;
456 model->
getNodes()[_node->getId()] = _node;
461 auto xmlGeometryId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceGeometry->Attribute(
"url"))), 1);
463 map<string, string> materialSymbols;
471 readGeometry(pathName, model, node, xmlRoot, xmlGeometryId, materialSymbols);
482 map<string, string> materialSymbols;
484 string xmlGeometryId;
488 auto xmlInstanceController = xmlInstanceControllers.at(0);
498 auto xmlInstanceControllerId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceController->Attribute(
"url"))), 1);
502 if (
string(
AVOID_NULLPTR_STRING(xmlLibraryController->Attribute(
"id"))) == xmlInstanceControllerId) {
505 if (xmlSkins.empty() ==
false) {
506 xmlSkin = xmlSkins.at(0);
512 if (xmlSkin ==
nullptr) {
514 "skin not found for instance controller '" +
527 array<float, 16> bindShapeMatrixArray;
528 for (
auto i = 0; i < bindShapeMatrixArray.size(); i++) {
529 bindShapeMatrixArray[i] = Float::parse(t.
nextToken());
535 auto node =
new Node(model, parentNode, xmlNodeId, xmlNodeName);
541 readGeometry(pathName, model, node, xmlRoot, xmlGeometryId, materialSymbols);
544 string xmlJointsSource;
545 string xmlJointsInverseBindMatricesSource;
549 xmlJointsSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlJointsInput->Attribute(
"source"))), 1);
552 xmlJointsInverseBindMatricesSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlJointsInput->Attribute(
"source"))), 1);
557 if (xmlJointsSource.length() == 0) {
559 "joint source not found for instance controller '" +
566 vector<Joint> joints;
577 if (xmlJointsInverseBindMatricesSource.length() == 0) {
579 "inverse bind matrices source not found for instance controller '" +
587 if (
string(
AVOID_NULLPTR_STRING(xmlSkinSource->Attribute(
"id"))) == xmlJointsInverseBindMatricesSource) {
589 auto& _joints = skinning->getJoints();
590 for (
auto i = 0; i < joints.size(); i++) {
593 array<float, 16> bindMatrixArray;
594 for (
auto i = 0; i < bindMatrixArray.size(); i++) {
595 bindMatrixArray[i] = Float::parse(t.
nextToken());
598 bindMatrix.
set(bindShapeMatrix);
600 joints[i].setBindMatrix(bindMatrix);
605 skinning->setJoints(joints);
608 vector<float> weights;
609 auto xmlJointOffset = -1;
610 auto xmlWeightOffset = -1;
611 string xmlWeightsSource;
614 for (
auto xmlVertexWeightInput: xmlVertexWeightInputs) {
616 if ((StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute(
"source"))), 1) == xmlJointsSource) ==
false) {
619 xmlJointOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute(
"offset"))));
622 xmlWeightOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute(
"offset"))));
623 xmlWeightsSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute(
"source"))), 1);
628 if (xmlJointOffset == -1) {
630 "xml vertex weight joint offset missing for node '" +
635 if (xmlWeightOffset == -1) {
637 "xml vertex weight weight offset missing for node " +
642 if (xmlWeightsSource.length() == 0) {
644 "xml vertex weight weight source missing for node '" +
655 weights.push_back(Float::parse(t.
nextToken()));
659 skinning->setWeights(weights);
662 auto xmlVertexWeightInputCount = xmlVertexWeightInputs.size();
665 t.
tokenize(vertexJointsInfluenceCountString,
" \n\r");
667 t2.
tokenize(vertexJointsInfluencesString,
" \n\r");
669 vector<vector<JointWeight>> verticesJointsWeights;
672 auto vertexJointsInfluencesCount = Integer::parse(t.
nextToken());
673 vector<JointWeight>vertexJointsWeights;
674 for (
auto i = 0; i < vertexJointsInfluencesCount; i++) {
675 auto vertexJoint = -1;
676 auto vertexWeight = -1;
677 while (vertexJoint == -1 || vertexWeight == -1) {
678 auto value = Integer::parse(t2.
nextToken());
679 if (offset % xmlVertexWeightInputCount == xmlJointOffset) {
681 }
else if (offset % xmlVertexWeightInputCount == xmlWeightOffset) {
682 vertexWeight = value;
686 vertexJointsWeights.push_back(
JointWeight(vertexJoint, vertexWeight));
688 verticesJointsWeights.push_back(vertexJointsWeights);
690 skinning->setVerticesJointsWeights(verticesJointsWeights);
691 node->setSkinning(skinning);
702 auto normalsOffset = node->
getNormals().size();
703 vector<Vector3> normals = node->
getNormals();;
710 vector<TiXmlElement*> xmlPolygonsList;
713 xmlPolygonsList.push_back(xmlTriangesElement);
717 xmlPolygonsList.push_back(xmlPolyListElement);
721 xmlPolygonsList.push_back(xmlPolygonsElement);
724 for (
auto xmlPolygons: xmlPolygonsList) {
727 if (StringTools::toLowerCase((xmlPolygons->Value())) ==
"polylist") {
731 auto vertexCount = Integer::parse(t.
nextToken());
732 if (vertexCount != 3) {
734 "we only support triangles in '" +
742 auto xmlVerticesOffset = -1;
743 string xmlVerticesSource;
744 auto xmlNormalsOffset = -1;
745 string xmlNormalsSource;
746 auto xmlTexCoordOffset = -1;
747 string xmlTexCoordSource;
748 auto xmlColorOffset = -1;
749 string xmlColorSource;
751 auto materialSymbolIt = materialSymbols.find(xmlMaterialId);
752 if (materialSymbolIt != materialSymbols.end()) {
753 xmlMaterialId = materialSymbolIt->second;
754 xmlMaterialId = StringTools::substring(xmlMaterialId, 1);
757 if (xmlMaterialId.length() > 0) {
759 auto materialIt = model->
getMaterials().find(xmlMaterialId);
761 material = materialIt->second;
764 material =
readMaterial(pathName, model, xmlRoot, xmlMaterialId);
769 unordered_set<int32_t> xmlInputSet;
773 xmlVerticesOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"offset"))));
774 xmlVerticesSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"source"))), 1);
775 xmlInputSet.insert(xmlVerticesOffset);
779 xmlNormalsOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"offset"))));
780 xmlNormalsSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"source"))), 1);
781 xmlInputSet.insert(xmlNormalsOffset);
785 xmlTexCoordOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"offset"))));
786 xmlTexCoordSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"source"))), 1);
787 xmlInputSet.insert(xmlTexCoordOffset);
791 xmlColorOffset = Integer::parse(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"offset"))));
792 xmlColorSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute(
"source"))), 1);
793 xmlInputSet.insert(xmlColorOffset);
796 xmlInputs = xmlInputSet.size();
801 if (StringTools::equalsIgnoreCase(
string(
AVOID_NULLPTR_STRING(xmlVerticesInput->Attribute(
"semantic"))),
"position") ==
true) {
802 xmlVerticesSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlVerticesInput->Attribute(
"source"))), 1);
804 if (StringTools::equalsIgnoreCase(
string(
AVOID_NULLPTR_STRING(xmlVerticesInput->Attribute(
"semantic"))),
"normal") ==
true) {
805 xmlNormalsSource = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlVerticesInput->Attribute(
"source"))), 1);
811 if (xmlVerticesSource.length() == 0) {
813 "Could not determine triangles vertices source for '" +
819 if (xmlNormalsSource.length() == 0) {
821 "Could not determine triangles normal source for '" +
838 vertices.push_back(
Vector3(x, y, z));
851 normals.push_back(
Vector3(x, y, z));
855 if (xmlTexCoordSource.length() > 0) {
874 array<int32_t, 3> vi;
876 array<int32_t, 3> ni;
878 array<int32_t, 3> ti;
883 auto value = Integer::parse(t.
nextToken());
884 if (valueIdx % xmlInputs == xmlVerticesOffset) {
887 if (value < 0 || value >= vertices.size() - verticesOffset) {
891 if (xmlNormalsSource.length() > 0 && xmlNormalsOffset == -1) {
893 if (value < 0 || value >= normals.size() - normalsOffset) {
898 if (xmlNormalsOffset != -1 && valueIdx % xmlInputs == xmlNormalsOffset) {
901 if (value < 0 || value >= normals.size() - normalsOffset) {
905 if (xmlTexCoordOffset != -1 && valueIdx % xmlInputs == xmlTexCoordOffset) {
908 if (value < 0 || value >= textureCoordinates.size() - textureCoordinatesOffset) {
912 if (viIdx == 3 && niIdx == 3 && (xmlTexCoordSource.length() == 0 || tiIdx == 3)) {
918 vi[0] + verticesOffset,
919 vi[1] + verticesOffset,
920 vi[2] + verticesOffset,
921 ni[0] + normalsOffset,
922 ni[1] + normalsOffset,
923 ni[2] + normalsOffset
925 if (xmlTexCoordSource.length() != 0) {
927 ti[0] + textureCoordinatesOffset,
928 ti[1] + textureCoordinatesOffset,
929 ti[2] + textureCoordinatesOffset
943 if (faces.empty() ==
false) {
945 facesEntities.push_back(facesEntity);
966 xmlEffectId = StringTools::substring(
string(
AVOID_NULLPTR_STRING(xmlInstanceEffect->Attribute(
"url"))), 1);
969 if (xmlEffectId.length() == 0) {
972 "Could not determine effect id for '" +
980 auto material =
new Material(xmlNodeId);
982 string xmlDiffuseTextureId;
983 string xmlTransparencyTextureId;
984 string xmlSpecularTextureId;
985 string xmlBumpTextureId;
991 map<string, string> samplerSurfaceMapping;
992 map<string, string> surfaceImageMapping;
997 surfaceImageMapping[xmlNewParamSID] =
1002 samplerSurfaceMapping[xmlNewParamSID] =
1007 for (
auto xmlTechniqueNode:
getChildren(xmlTechnique)) {
1012 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlTransparencyTextureId);
1013 string sample2Surface;
1014 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1015 sample2Surface = sample2SurfaceIt->second;
1017 if (sample2Surface.length() == 0)
continue;
1019 string surface2Image;
1020 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1021 if (surface2ImageIt != surfaceImageMapping.end()) {
1022 surface2Image = surface2ImageIt->second;
1024 if (surface2Image.length() > 0) {
1025 xmlTransparencyTextureId = surface2Image;
1035 array<float, 4> colorArray;
1036 for (
auto i = 0; i < colorArray.size(); i++) {
1037 colorArray[i] = Float::parse(t.
nextToken());
1039 specularMaterialProperties->setDiffuseColor(
Color4(colorArray));
1045 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlDiffuseTextureId);
1046 string sample2Surface;
1047 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1048 sample2Surface = sample2SurfaceIt->second;
1050 if (sample2Surface.length() == 0)
continue;
1052 string surface2Image;
1053 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1054 if (surface2ImageIt != surfaceImageMapping.end()) {
1055 surface2Image = surface2ImageIt->second;
1057 if (surface2Image.length() > 0) {
1058 xmlDiffuseTextureId = surface2Image;
1068 array<float, 4> colorArray;
1069 for (
auto i = 0; i < colorArray.size(); i++) {
1070 colorArray[i] = Float::parse(t.
nextToken());
1072 specularMaterialProperties->setAmbientColor(
Color4(colorArray));
1081 array<float, 4> colorArray;
1082 for (
auto i = 0; i < colorArray.size(); i++) {
1083 colorArray[i] = Float::parse(t.
nextToken());
1085 specularMaterialProperties->setEmissionColor(
Color4(colorArray));
1089 auto hasSpecularMap =
false;
1090 auto hasSpecularColor =
false;
1096 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlSpecularTextureId);
1097 string sample2Surface;
1098 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1099 sample2Surface = sample2SurfaceIt->second;
1101 if (sample2Surface.length() == 0)
continue;
1103 string surface2Image;
1104 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1105 if (surface2ImageIt != surfaceImageMapping.end()) {
1106 surface2Image = surface2ImageIt->second;
1109 if (surface2Image.length() > 0) {
1110 xmlSpecularTextureId = surface2Image;
1111 hasSpecularMap =
true;
1118 array<float, 4> colorArray;
1119 for (
auto i = 0; i < colorArray.size(); i++) {
1120 colorArray[i] = Float::parse(t.
nextToken());
1122 specularMaterialProperties->setSpecularColor(
Color4(colorArray));
1123 hasSpecularColor =
true;
1126 if (hasSpecularMap ==
true && hasSpecularColor ==
false) {
1127 specularMaterialProperties->setSpecularColor(
Color4(1.0f, 1.0f, 1.0f, 1.0f));
1132 specularMaterialProperties->setShininess(Float::parse(
string(
AVOID_NULLPTR_STRING(xmlFloat->GetText()))));
1142 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlBumpTextureId);
1143 string sample2Surface;
1144 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1145 sample2Surface = sample2SurfaceIt->second;
1147 if (sample2Surface.length() == 0)
continue;
1149 string surface2Image;
1150 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1151 if (surface2ImageIt != surfaceImageMapping.end()) {
1152 surface2Image = surface2ImageIt->second;
1155 if (surface2Image.length() > 0) xmlBumpTextureId = surface2Image;
1162 string xmlTransparencyTextureFilename;
1163 if (xmlDiffuseTextureId.length() > 0) {
1166 if (xmlTransparencyTextureFilename.length() > 0) {
1173 string xmlDiffuseTextureFilename;
1174 if (xmlDiffuseTextureId.length() > 0) {
1177 if (xmlDiffuseTextureFilename.length() > 0) {
1180 specularMaterialProperties->setDiffuseTexture(pathName, xmlDiffuseTextureFilename, pathName, xmlTransparencyTextureFilename);
1181 if (specularMaterialProperties->hasDiffuseTextureTransparency() ==
true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(
true);
1186 string xmlSpecularTextureFilename;
1187 if (xmlSpecularTextureId.length() > 0) {
1190 if (xmlSpecularTextureFilename.length() > 0) {
1193 specularMaterialProperties->setSpecularTexture(pathName, xmlSpecularTextureFilename);
1198 string xmlBumpTextureFilename;
1199 if (xmlBumpTextureId.length() > 0) {
1202 if (xmlBumpTextureFilename.length() > 0) {
1205 specularMaterialProperties->setNormalTexture(pathName, xmlBumpTextureFilename);
1220 specularMaterialProperties->setAmbientColor(
1228 specularMaterialProperties->setDiffuseColor(
1233 specularMaterialProperties->getDiffuseColor().getAlpha()
1239 material->setSpecularMaterialProperties(specularMaterialProperties);
1251 if (StringTools::startsWith(fileName,
"/") ==
true ||
1252 StringTools::regexMatch(fileName,
"^[A-Z]\\:\\\\.*$") ==
true) {
1253 int indexSlash = fileName.find_last_of(L
'/');
1254 int indexBackslash = fileName.find_last_of(L
'\\');
1255 if (indexSlash != -1 || indexBackslash != -1) {
1256 if (indexSlash > indexBackslash) {
1257 return StringTools::substring(fileName, indexSlash + 1);
1259 return StringTools::substring(fileName, indexBackslash + 1);
1268 string xmlTextureFilename;
1273 if (StringTools::startsWith(xmlTextureFilename,
"file://") ==
true) {
1274 xmlTextureFilename = StringTools::substring(xmlTextureFilename, 7);
1279 return xmlTextureFilename;
1284 vector<TiXmlElement*> elementList;
1286 elementList.push_back(child);
1293 vector<TiXmlElement*> elementList;
1295 elementList.push_back(child);
#define AVOID_NULLPTR_STRING(arg)
Collada DAE model reader.
static void readGeometry(const string &pathName, Model *model, Node *node, TiXmlElement *xmlRoot, const string &xmlNodeId, const map< string, string > &materialSymbols)
Reads a geometry.
static const vector< TiXmlElement * > getChildren(TiXmlElement *parent)
Returns immediate children tags.
static constexpr float BLENDER_AMBIENT_FROM_DIFFUSE_SCALE
static Material * readMaterial(const string &pathName, Model *model, TiXmlElement *xmlRoot, const string &xmlNodeId)
Reads a material.
static constexpr float BLENDER_DIFFUSE_SCALE
static UpVector * getUpVector(TiXmlElement *xmlRoot)
Get Up vector.
static Node * readVisualSceneInstanceController(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode)
Reads a instance controller.
static Node * readNode(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode, float fps)
Reads a DAE visual scene node node.
static void setupModelImportRotationMatrix(TiXmlElement *xmlRoot, Model *model)
Set up model import rotation matrix.
static const string makeFileNameRelative(const string &fileName)
Make file name relative.
static const string getTextureFileNameById(TiXmlElement *xmlRoot, const string &xmlTextureId)
Get texture file name by id.
static STATIC_DLL_IMPEXT const Color4 BLENDER_AMBIENT_NONE
static void setupModelImportScaleMatrix(TiXmlElement *xmlRoot, Model *model)
Set up model import scale matrix.
static const vector< TiXmlElement * > getChildrenByTagName(TiXmlElement *parent, const char *name)
Returns immediate children tags by tag name.
static Model::AuthoringTool getAuthoringTool(TiXmlElement *xmlRoot)
Get authoring tool.
static Node * readVisualSceneNode(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode, float fps)
Read a DAE visual scene node.
Color 4 base definition class.
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.
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.
Representation of a 3d model.
const Matrix4x4 & getImportTransformationsMatrix()
AuthoringTool getAuthoringTool()
map< string, Node * > & getNodes()
Returns all object's nodes.
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
map< string, Material * > & getMaterials()
Returns all object materials.
void setTextureCoordinates(const vector< TextureCoordinate > &textureCoordinates)
Set texture coordinates.
void setVertices(const vector< Vector3 > &vertices)
Set vertices.
void setFacesEntities(const vector< FacesEntity > &facesEntities)
Set up faces entities.
const vector< TextureCoordinate > & getTextureCoordinates() const
const vector< Vector3 > & getNormals() const
const vector< FacesEntity > & getFacesEntities() const
void setNormals(const vector< Vector3 > &normals)
Set normals.
const vector< Vector3 > & getVertices() const
Represents rotation orders of a model.
Skinning definition for nodes.
Represents specular material properties.
Class representing texture UV coordinates data.
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 & transpose()
Transposes this matrix.
Matrix4x4 clone()
Clones this 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.
const string & nextToken()
void tokenize(const string &str, const string &delimiters)
Tokenize.
An attribute is a name-value pair.
Always the top level node.
virtual const char * Parse(const char *p, TiXmlParsingData *data=0, TiXmlEncoding encoding=TIXML_DEFAULT_ENCODING)
Parse the given null terminated block of xml data.
const TiXmlElement * RootElement() const
Get the root element – the only top level element – of the document.
const char * ErrorDesc() const
Contains a textual (english) description of the error if one occurs.
bool Error() const
If an error occurs, Error will be set to true.
The element is a container class.
const char * Attribute(const char *name) const
Given an attribute name, Attribute() returns the value for the attribute of that name,...
const TiXmlElement * NextSiblingElement() const
Convenience function to get through elements.
const TiXmlElement * FirstChildElement() const
Convenience function to get through elements.
std::exception Exception
Exception base class.