TDME2 1.9.121
DAEReader.cpp
Go to the documentation of this file.
2
3#include <map>
4#include <string>
5#include <unordered_set>
6#include <vector>
7
8#include <tdme/tdme.h>
30#include <tdme/math/Math.h>
31#include <tdme/math/Matrix4x4.h>
32#include <tdme/math/Vector3.h>
43
44#include <ext/tinyxml/tinyxml.h>
45
46#define AVOID_NULLPTR_STRING(arg) (arg == nullptr?"":arg)
47
48using std::array;
49using std::map;
50using std::string;
51using std::to_string;
52using std::unordered_set;
53using std::vector;
54
89
93
94const Color4 DAEReader::BLENDER_AMBIENT_NONE(0.0f, 0.0f, 0.0f, 1.0f);
95
96Model* DAEReader::read(const string& pathName, const string& fileName)
97{
98 // load dae xml document
99 auto xmlContent = FileSystem::getInstance()->getContentAsString(pathName, fileName);
100 TiXmlDocument xmlDocument;
101 xmlDocument.Parse(xmlContent.c_str());
102 if (xmlDocument.Error() == true) {
104 string("Could not parse XML. Error='") + string(xmlDocument.ErrorDesc()) + string("'")
105 );
106 }
107 TiXmlElement* xmlRoot = xmlDocument.RootElement();
108
109 // authoring tool
110 auto authoringTool = getAuthoringTool(xmlRoot);
111
112 // up vector and rotation order
113 auto upVector = getUpVector(xmlRoot);
114 RotationOrder* rotationOrder = nullptr;
115 {
116 auto v = upVector;
117 if (v == UpVector::Y_UP) {
118 rotationOrder = RotationOrder::ZYX;
119 } else
120 if (v == UpVector::Z_UP) {
121 rotationOrder = RotationOrder::YZX;
122 }
123 }
124
125 // create model
126 auto model = new Model(
127 fileName,
128 fileName,
129 upVector,
130 rotationOrder,
131 nullptr,
132 authoringTool
133 );
134
135 // import matrix
136 setupModelImportRotationMatrix(xmlRoot, model);
137 setupModelImportScaleMatrix(xmlRoot, model);
138
139 // parse scene from xml
140 string xmlSceneId;
141 auto xmlScene = getChildrenByTagName(xmlRoot, "scene").at(0);
142 for (auto xmlInstanceVisualscene: getChildrenByTagName(xmlScene, "instance_visual_scene")) {
143 xmlSceneId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceVisualscene->Attribute("url"))), 1);
144 }
145
146 // parse visual scenes
147 auto xmlLibraryVisualScenes = getChildrenByTagName(xmlRoot, "library_visual_scenes").at(0);
148 for (auto xmlLibraryVisualScene: getChildrenByTagName(xmlLibraryVisualScenes, "visual_scene")) {
149 auto xmlVisualSceneId = string(AVOID_NULLPTR_STRING(xmlLibraryVisualScene->Attribute("id")));
150 if (xmlVisualSceneId == xmlSceneId) {
151 // default FPS
152 auto fps = 30.0f;
153 // parse frames per second
154 auto xmlExtraNodes = getChildrenByTagName(xmlLibraryVisualScene, "extra");
155 if (xmlExtraNodes.empty() == false) {
156 auto xmlExtraNode = xmlExtraNodes.at(0);
157 for (auto xmlTechnique: getChildrenByTagName(xmlExtraNode, "technique")) {
158 auto xmlFrameRateNodes = getChildrenByTagName(xmlTechnique, "frame_rate");
159 if (xmlFrameRateNodes.empty() == false) {
160 fps = Float::parse(string(AVOID_NULLPTR_STRING(xmlFrameRateNodes.at(0)->GetText())));
161 break;
162 }
163 }
164 }
165 // set up frames per seconds
166 model->setFPS(fps);
167 // visual scene root nodes
168 for (auto xmlNode: getChildrenByTagName(xmlLibraryVisualScene, "node")) {
169 auto node = readVisualSceneNode(pathName, model, nullptr, xmlRoot, xmlNode, fps);
170 if (node != nullptr) {
171 model->getSubNodes()[node->getId()] = node;
172 model->getNodes()[node->getId()] = node;
173 }
174 }
175 }
176 }
177 if (ModelTools::hasDefaultAnimation(model) == false) ModelTools::createDefaultAnimation(model, 0);
178 // set up joints
179 ModelTools::setupJoints(model);
180 // fix animation length
181 ModelTools::fixAnimationLength(model);
182 // prepare for indexed rendering
183 ModelTools::prepareForIndexedRendering(model);
184 //
185 return model;
186}
187
188Model::AuthoringTool DAEReader::getAuthoringTool(TiXmlElement* xmlRoot)
189{
190 for (auto xmlAsset: getChildrenByTagName(xmlRoot, "asset")) {
191 for (auto xmlContributer: getChildrenByTagName(xmlAsset, "contributor")) {
192 for (auto xmlAuthoringTool: getChildrenByTagName(xmlContributer, "authoring_tool")) {
193 if (string(AVOID_NULLPTR_STRING(xmlAuthoringTool->GetText())).find("Blender") != -1) {
194 return Model::AUTHORINGTOOL_BLENDER;
195 }
196 }
197 }
198 }
199 return Model::AUTHORINGTOOL_UNKNOWN;
200}
201
203{
204 // determine up axis
205 for (auto xmlAsset: getChildrenByTagName(xmlRoot, "asset")) {
206 for (auto xmlAssetUpAxis: getChildrenByTagName(xmlAsset, "up_axis")) {
207 auto upAxis = string(AVOID_NULLPTR_STRING(xmlAssetUpAxis->GetText()));
208 if (StringTools::equalsIgnoreCase(upAxis, "Y_UP") == true) {
209 return UpVector::Y_UP;
210 } else
211 if (StringTools::equalsIgnoreCase(upAxis, "Z_UP") == true) {
212 return UpVector::Z_UP;
213 } else
214 if (StringTools::equalsIgnoreCase(upAxis, "X_UP") == true) {
215 throw ModelFileIOException("X-Up is not supported");
216 } else {
217 throw ModelFileIOException("Unknown Up vector");
218 }
219 }
220 }
221 throw ModelFileIOException("Unknown Up vector");
222}
223
225{
226 // determine rotation matrix
227 for (auto xmlAsset: getChildrenByTagName(xmlRoot, "asset")) {
228 for (auto xmlAssetUpAxis: getChildrenByTagName(xmlAsset, "up_axis")) {
229 auto upAxis = string(AVOID_NULLPTR_STRING(xmlAssetUpAxis->GetText()));
230 if (StringTools::equalsIgnoreCase(upAxis, "Y_UP") == true) {
231 } else
232 if (StringTools::equalsIgnoreCase(upAxis, "Z_UP") == true) {
233 model->setImportTransformationsMatrix(model->getImportTransformationsMatrix().clone().rotate(Vector3(1.0f, 0.0f, 0.0f), -90.0f));
234 } else
235 if (StringTools::equalsIgnoreCase(upAxis, "X_UP") == true) {
236 model->setImportTransformationsMatrix(model->getImportTransformationsMatrix().clone().rotate(Vector3(0.0f, 1.0f, 0.0f), -90.0f));
237 } else {
238 Console::println(string("Warning: Unknown up axis: " + upAxis));
239 }
240 }
241 }
242}
243
245{
246 // determine scale
247 for (auto xmlAsset: getChildrenByTagName(xmlRoot, "asset")) {
248 for (auto xmlAssetUnit: getChildrenByTagName(xmlAsset, "unit")) {
249 string tmp;
250 if ((tmp = string(AVOID_NULLPTR_STRING(xmlAssetUnit->Attribute("meter")))).length() > 0) {
251 float scaleFactor = Float::parse(tmp);
253 }
254 }
255 }
256}
257
258Node* DAEReader::readVisualSceneNode(const string& pathName, Model* model, Node* parentNode, TiXmlElement* xmlRoot, TiXmlElement* xmlNode, float fps)
259{
260 auto xmlInstanceControllers = getChildrenByTagName(xmlNode, "instance_controller");
261 if (xmlInstanceControllers.empty() == false) {
262 return readVisualSceneInstanceController(pathName, model, parentNode, xmlRoot, xmlNode);
263 } else {
264 return readNode(pathName, model, parentNode, xmlRoot, xmlNode, fps);
265 }
266}
267
268Node* DAEReader::readNode(const string& pathName, Model* model, Node* parentNode, TiXmlElement* xmlRoot, TiXmlElement* xmlNode, float fps)
269{
270 auto xmlNodeId = string(AVOID_NULLPTR_STRING(xmlNode->Attribute("id")));
271 auto xmlNodeName = string(AVOID_NULLPTR_STRING(xmlNode->Attribute("name")));
272 if (xmlNodeId.length() == 0) xmlNodeId = xmlNodeName;
273
274 // create node
275 auto node = new Node(model, parentNode, xmlNodeId, xmlNodeName);
276
277 // set up local transformations matrix
278 auto xmlMatrixElements = getChildrenByTagName(xmlNode, "matrix");
279 if (xmlMatrixElements.empty() == false) {
281 Matrix4x4 transformationsMatrix;
282 auto xmlMatrix = string(AVOID_NULLPTR_STRING(xmlMatrixElements.at(0)->GetText()));
283 t.tokenize(xmlMatrix, " \n\r");
284 array<float, 16> transformationsMatrixArray;
285 for (auto i = 0; i < transformationsMatrixArray.size(); i++) {
286 transformationsMatrixArray[i] = Float::parse(t.nextToken());
287 }
288 transformationsMatrix.set(transformationsMatrixArray).transpose();
289 node->setTransformationsMatrix(transformationsMatrix);
290 }
291
292 // parse animations
293 auto xmlAnimationsLibrary = getChildrenByTagName(xmlRoot, "library_animations");
294 if (xmlAnimationsLibrary.empty() == false) {
295 for (auto xmlAnimation: getChildrenByTagName(xmlAnimationsLibrary.at(0), "animation")) {
296 // older DAE has animation/animation xml nodes
297 auto _xmlAnimation = getChildrenByTagName(xmlAnimation, "animation");
298 if (_xmlAnimation.empty() == false) {
299 xmlAnimation = _xmlAnimation.at(0);
300 }
301 // find sampler source
302 string xmlSamplerSource;
303 auto xmlChannel = getChildrenByTagName(xmlAnimation, "channel").at(0);
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);
306 }
307 // check for sampler source
308 if (xmlSamplerSource.length() == 0) {
309 continue;
310 }
311 // parse animation output matrices
312 string xmlSamplerOutputSource;
313 string xmlSamplerInputSource;
314 auto xmlSampler = getChildrenByTagName(xmlAnimation, "sampler").at(0);
315 for (auto xmlSamplerInput: getChildrenByTagName(xmlSampler, "input")) {
316 if (string(AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute("semantic"))) == "OUTPUT") {
317 xmlSamplerOutputSource = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute("source"))), 1);
318 } else
319 if (string(AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute("semantic"))) == "INPUT") {
320 xmlSamplerInputSource = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlSamplerInput->Attribute("source"))), 1);
321 }
322 }
323 // check for sampler source
324 if (xmlSamplerOutputSource.length() == 0) {
326 "Could not find xml sampler output source for animation for '" +
327 (xmlNodeId) +
328 "'"
329 );
330 }
331 // load animation input matrices
332 // TODO: check accessor "time"
333 vector<float> keyFrameTimes;
334 for (auto xmlAnimationSource: getChildrenByTagName(xmlAnimation, "source")) {
335 if (string(AVOID_NULLPTR_STRING(xmlAnimationSource->Attribute("id"))) == xmlSamplerInputSource) {
336 auto xmlFloatArray = getChildrenByTagName(xmlAnimationSource, "float_array").at(0);
337 auto frames = Integer::parse(string(AVOID_NULLPTR_STRING(xmlFloatArray->Attribute("count"))));
338 auto valueString = string(AVOID_NULLPTR_STRING(xmlFloatArray->GetText()));
339 auto keyFrameIdx = 0;
340 keyFrameTimes.resize(frames);
342 t.tokenize(valueString, " \n\r");
343 while (t.hasMoreTokens()) {
344 keyFrameTimes[keyFrameIdx++] = Float::parse(t.nextToken());
345 }
346 }
347 }
348 // load animation output matrices
349 // TODO: check accessor "transform"
350 if (keyFrameTimes.size() > 0) {
351 for (auto xmlAnimationSource: getChildrenByTagName(xmlAnimation, "source")) {
352 if (string(AVOID_NULLPTR_STRING(xmlAnimationSource->Attribute("id"))) == xmlSamplerOutputSource) {
353 auto xmlFloatArray = getChildrenByTagName(xmlAnimationSource, "float_array").at(0);
354 auto keyFrames = Integer::parse(string(AVOID_NULLPTR_STRING(xmlFloatArray->Attribute("count")))) / 16;
355 // some models have animations without frames
356 if (keyFrames > 0) {
357 auto valueString = string(AVOID_NULLPTR_STRING(xmlFloatArray->GetText()));
359 t.tokenize(valueString, " \n\r");
360 // parse key frame
361 int32_t keyFrameIdx = 0;
362 vector<Matrix4x4> keyFrameMatrices;
363 keyFrameMatrices.resize(keyFrames);
364 while (t.hasMoreTokens()) {
365 // set animation transformation matrix at frame
366 array<float, 16> keyFrameMatricesArray;
367 for (auto i = 0; i < keyFrameMatricesArray.size() ;i++) {
368 keyFrameMatricesArray[i] = Float::parse(t.nextToken());
369 }
370 keyFrameMatrices[keyFrameIdx].set(keyFrameMatricesArray);
371 keyFrameMatrices[keyFrameIdx].transpose();
372 keyFrameIdx++;
373 }
374
375 auto frames = static_cast< int32_t >(Math::ceil(keyFrameTimes[keyFrameTimes.size() - 1] * fps));
376 if (frames > 0) {
377 ModelTools::createDefaultAnimation(model, frames);
378 auto animation = new Animation();
379 vector<Matrix4x4> transformationsMatrices;
380 transformationsMatrices.resize(frames);
381 auto tansformationsMatrixLast = &keyFrameMatrices[0];
382 keyFrameIdx = 0;
383 auto frameIdx = 0;
384 auto timeStampLast = 0.0f;
385 for (auto keyFrameTime : keyFrameTimes) {
386 auto transformationsMatrixCurrent = &keyFrameMatrices[(keyFrameIdx) % keyFrameMatrices.size()];
387 float timeStamp;
388 for (timeStamp = timeStampLast; timeStamp < keyFrameTime; timeStamp += 1.0f / fps) {
389 if (frameIdx >= frames) {
390 // TODO: check me again!
391 // Console::println(string("Warning: skipping frame: ") + to_string(frameIdx));
392 frameIdx++;
393 continue;
394 }
395 transformationsMatrices[frameIdx] = Matrix4x4::interpolateLinear(*tansformationsMatrixLast, *transformationsMatrixCurrent, (timeStamp - timeStampLast) / (keyFrameTime - timeStampLast));
396 frameIdx++;
397 }
398 timeStampLast = timeStamp;
399 tansformationsMatrixLast = transformationsMatrixCurrent;
400 keyFrameIdx++;
401 }
402 animation->setTransformationsMatrices(transformationsMatrices);
403 node->setAnimation(animation);
404 }
405 }
406 }
407 }
408 }
409 }
410 }
411
412 // parse sub nodes
413 for (auto _xmlNode: getChildrenByTagName(xmlNode, "node")) {
414 auto _node = readVisualSceneNode(pathName, model, node, xmlRoot, _xmlNode, fps);
415 if (_node != nullptr) {
416 node->getSubNodes()[_node->getId()] = _node;
417 model->getNodes()[_node->getId()] = _node;
418 }
419 }
420
421 // check for geometry data
422 string xmlInstanceGeometryId;
423 auto xmlInstanceGeometryElements = getChildrenByTagName(xmlNode, "instance_geometry");
424 if (xmlInstanceGeometryElements.empty() == false) {
425 auto xmlInstanceGeometryElement = xmlInstanceGeometryElements.at(0);
426 // fetch instance geometry url
427 xmlInstanceGeometryId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceGeometryElement->Attribute("url"))), 1);
428 // determine bound materials
429 map<string, string> materialSymbols;
430 for (auto xmlBindMaterial: getChildrenByTagName(xmlInstanceGeometryElement, "bind_material"))
431 for (auto xmlTechniqueCommon: getChildrenByTagName(xmlBindMaterial, "technique_common"))
432 for (auto xmlInstanceMaterial: getChildrenByTagName(xmlTechniqueCommon, "instance_material")) {
433 materialSymbols[string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("symbol")))] =
434 string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("target")));
435 }
436 // parse geometry
437 readGeometry(pathName, model, node, xmlRoot, xmlInstanceGeometryId, materialSymbols);
438 return node;
439 }
440
441 // otherwise check for "instance_node"
442 string xmlInstanceNodeId;
443 for (auto xmlInstanceNodeElement: getChildrenByTagName(xmlNode, "instance_node")) {
444 xmlInstanceNodeId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceNodeElement->Attribute("url"))), 1);
445 }
446 // do we have a instance node id?
447 if (xmlInstanceNodeId.length() > 0) {
448 for (auto xmlLibraryNodes: getChildrenByTagName(xmlRoot, "library_nodes"))
449 for (auto xmlLibraryNode: getChildrenByTagName(xmlLibraryNodes, "node"))
450 if (string(AVOID_NULLPTR_STRING(xmlLibraryNode->Attribute("id"))) == xmlInstanceNodeId) {
451 // parse sub nodes
452 for (auto _xmlNode: getChildrenByTagName(xmlLibraryNode, "node")) {
453 auto _node = readVisualSceneNode(pathName, model, parentNode, xmlRoot, _xmlNode, fps);
454 if (_node != nullptr) {
455 node->getSubNodes()[_node->getId()] = _node;
456 model->getNodes()[_node->getId()] = _node;
457 }
458 }
459 // parse geometry
460 for (auto xmlInstanceGeometry: getChildrenByTagName(xmlLibraryNode, "instance_geometry")) {
461 auto xmlGeometryId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceGeometry->Attribute("url"))), 1);
462 // parse material symbols
463 map<string, string> materialSymbols;
464 for (auto xmlBindMaterial: getChildrenByTagName(xmlInstanceGeometry, "bind_material"))
465 for (auto xmlTechniqueCommon: getChildrenByTagName(xmlBindMaterial, "technique_common"))
466 for (auto xmlInstanceMaterial: getChildrenByTagName(xmlTechniqueCommon, "instance_material")) {
467 materialSymbols[string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("symbol")))] =
468 string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("target")));
469 }
470 // parse geometry
471 readGeometry(pathName, model, node, xmlRoot, xmlGeometryId, materialSymbols);
472 }
473 }
474 }
475 return node;
476}
477
478Node* DAEReader::readVisualSceneInstanceController(const string& pathName, Model* model, Node* parentNode, TiXmlElement* xmlRoot, TiXmlElement* xmlNode)
479{
480 auto xmlNodeId = string(AVOID_NULLPTR_STRING(xmlNode->Attribute("id")));
481 auto xmlNodeName = string(AVOID_NULLPTR_STRING(xmlNode->Attribute("name")));
482 map<string, string> materialSymbols;
483 // geometry id
484 string xmlGeometryId;
485 // parse library controllers, find our controller
486 auto xmlInstanceControllers = getChildrenByTagName(xmlNode, "instance_controller");
487 TiXmlElement* xmlSkin = nullptr;
488 auto xmlInstanceController = xmlInstanceControllers.at(0);
489
490 // parse material symbols
491 for (auto xmlBindMaterial: getChildrenByTagName(xmlInstanceController, "bind_material"))
492 for (auto xmlTechniqueCommon: getChildrenByTagName(xmlBindMaterial, "technique_common"))
493 for (auto xmlInstanceMaterial: getChildrenByTagName(xmlTechniqueCommon, "instance_material")) {
494 materialSymbols[string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("symbol")))] =
495 string(AVOID_NULLPTR_STRING(xmlInstanceMaterial->Attribute("target")));
496 }
497
498 auto xmlInstanceControllerId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceController->Attribute("url"))), 1);
499 auto xmlLibraryControllers = getChildrenByTagName(xmlRoot, "library_controllers").at(0);
500 for (auto xmlLibraryController: getChildrenByTagName(xmlLibraryControllers, "controller")) {
501 // our controller ?
502 if (string(AVOID_NULLPTR_STRING(xmlLibraryController->Attribute("id"))) == xmlInstanceControllerId) {
503 // parse skin
504 auto xmlSkins = getChildrenByTagName(xmlLibraryController, "skin");
505 if (xmlSkins.empty() == false) {
506 xmlSkin = xmlSkins.at(0);
507 }
508 }
509 }
510
511 // check for xml skin
512 if (xmlSkin == nullptr) {
514 "skin not found for instance controller '" +
515 (xmlNodeId) +
516 "'"
517 );
518 }
519
520 // get geometry id
521 xmlGeometryId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlSkin->Attribute("source"))), 1);
522
523 // parse bind shape matrix
524 auto xmlMatrix = string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlSkin, "bind_shape_matrix").at(0)->GetText()));
526 t.tokenize(xmlMatrix, " \n\r");
527 array<float, 16> bindShapeMatrixArray;
528 for (auto i = 0; i < bindShapeMatrixArray.size(); i++) {
529 bindShapeMatrixArray[i] = Float::parse(t.nextToken());
530 }
531 Matrix4x4 bindShapeMatrix;
532 bindShapeMatrix.set(bindShapeMatrixArray).transpose();
533
534 // create node
535 auto node = new Node(model, parentNode, xmlNodeId, xmlNodeName);
536
537 // create skinning
538 auto skinning = new Skinning();
539
540 // parse geometry
541 readGeometry(pathName, model, node, xmlRoot, xmlGeometryId, materialSymbols);
542
543 // parse joints
544 string xmlJointsSource;
545 string xmlJointsInverseBindMatricesSource;
546 auto xmlJoints = getChildrenByTagName(xmlSkin, "joints").at(0);
547 for (auto xmlJointsInput: getChildrenByTagName(xmlJoints, "input")) {
548 if (string(AVOID_NULLPTR_STRING(xmlJointsInput->Attribute("semantic"))) == "JOINT") {
549 xmlJointsSource = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlJointsInput->Attribute("source"))), 1);
550 } else
551 if (string(AVOID_NULLPTR_STRING(xmlJointsInput->Attribute("semantic"))) == "INV_BIND_MATRIX") {
552 xmlJointsInverseBindMatricesSource = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlJointsInput->Attribute("source"))), 1);
553 }
554 }
555
556 // check for joints sources
557 if (xmlJointsSource.length() == 0) {
559 "joint source not found for instance controller '" +
560 (xmlNodeId) +
561 "'"
562 );
563 }
564
565 // parse joint ids
566 vector<Joint> joints;
567 for (auto xmlSkinSource: getChildrenByTagName(xmlSkin, "source")) {
568 if (string(AVOID_NULLPTR_STRING(xmlSkinSource->Attribute("id"))) == xmlJointsSource) {
569 t.tokenize(string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlSkinSource, "Name_array").at(0)->GetText())), " \n\r");
570 while (t.hasMoreTokens()) {
571 joints.push_back(Joint(t.nextToken()));
572 }
573 }
574 }
575
576 // check for inverse bind matrices source
577 if (xmlJointsInverseBindMatricesSource.length() == 0) {
579 "inverse bind matrices source not found for instance controller '" +
580 (xmlNodeId) +
581 "'"
582 );
583 }
584
585 // Create joints bind matrices
586 for (auto xmlSkinSource: getChildrenByTagName(xmlSkin, "source")) {
587 if (string(AVOID_NULLPTR_STRING(xmlSkinSource->Attribute("id"))) == xmlJointsInverseBindMatricesSource) {
588 t.tokenize(string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlSkinSource, "float_array").at(0)->GetText())), " \n\r");
589 auto& _joints = skinning->getJoints();
590 for (auto i = 0; i < joints.size(); i++) {
591 // The vertices are defined in model space
592 // The transformation to the local space of the joint is called the inverse bind matrix
593 array<float, 16> bindMatrixArray;
594 for (auto i = 0; i < bindMatrixArray.size(); i++) {
595 bindMatrixArray[i] = Float::parse(t.nextToken());
596 }
597 Matrix4x4 bindMatrix;
598 bindMatrix.set(bindShapeMatrix);
599 bindMatrix.multiply((Matrix4x4(bindMatrixArray)).transpose());
600 joints[i].setBindMatrix(bindMatrix);
601 }
602 }
603 }
604
605 skinning->setJoints(joints);
606
607 // read vertex influences
608 vector<float> weights;
609 auto xmlJointOffset = -1;
610 auto xmlWeightOffset = -1;
611 string xmlWeightsSource;
612 auto xmlVertexWeights = getChildrenByTagName(xmlSkin, "vertex_weights").at(0);
613 auto xmlVertexWeightInputs = getChildrenByTagName(xmlVertexWeights, "input");
614 for (auto xmlVertexWeightInput: xmlVertexWeightInputs) {
615 if (string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("semantic"))) == "JOINT") {
616 if ((StringTools::substring(string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("source"))), 1) == xmlJointsSource) == false) {
617 throw ModelFileIOException("joint inverse bind matrices source do not match");
618 }
619 xmlJointOffset = Integer::parse(string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("offset"))));
620 } else
621 if (string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("semantic"))) == "WEIGHT") {
622 xmlWeightOffset = Integer::parse(string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("offset"))));
623 xmlWeightsSource = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlVertexWeightInput->Attribute("source"))), 1);
624 }
625 }
626
627 // check for vertex weight parameter
628 if (xmlJointOffset == -1) {
630 "xml vertex weight joint offset missing for node '" +
631 (xmlNodeId) +
632 "'"
633 );
634 }
635 if (xmlWeightOffset == -1) {
637 "xml vertex weight weight offset missing for node " +
638 (xmlNodeId) +
639 "'"
640 );
641 }
642 if (xmlWeightsSource.length() == 0) {
644 "xml vertex weight weight source missing for node '" +
645 (xmlNodeId) +
646 "'"
647 );
648 }
649
650 // parse weights
651 for (auto xmlSkinSource: getChildrenByTagName(xmlSkin, "source")) {
652 if (string(AVOID_NULLPTR_STRING(xmlSkinSource->Attribute("id"))) == xmlWeightsSource) {
653 t.tokenize(string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlSkinSource, "float_array").at(0)->GetText())), " \n\r");
654 while (t.hasMoreTokens()) {
655 weights.push_back(Float::parse(t.nextToken()));
656 }
657 }
658 }
659 skinning->setWeights(weights);
660
661 // actually do parse joint influences of each vertex
662 auto xmlVertexWeightInputCount = xmlVertexWeightInputs.size();
663 auto vertexJointsInfluenceCountString = string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlVertexWeights, "vcount").at(0)->GetText()));
664 auto vertexJointsInfluencesString = string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlVertexWeights, "v").at(0)->GetText()));
665 t.tokenize(vertexJointsInfluenceCountString, " \n\r");
667 t2.tokenize(vertexJointsInfluencesString, " \n\r");
668 auto offset = 0;
669 vector<vector<JointWeight>> verticesJointsWeights;
670 while (t.hasMoreTokens()) {
671 // read joint influences for current vertex
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) {
680 vertexJoint = value;
681 } else if (offset % xmlVertexWeightInputCount == xmlWeightOffset) {
682 vertexWeight = value;
683 }
684 offset++;
685 }
686 vertexJointsWeights.push_back(JointWeight(vertexJoint, vertexWeight));
687 }
688 verticesJointsWeights.push_back(vertexJointsWeights);
689 }
690 skinning->setVerticesJointsWeights(verticesJointsWeights);
691 node->setSkinning(skinning);
692
693 //
694 return node;
695}
696
697void DAEReader::readGeometry(const string& pathName, Model* model, Node* node, TiXmlElement* xmlRoot, const string& xmlNodeId, const map<string, string>& materialSymbols)
698{
699 vector<FacesEntity> facesEntities = node->getFacesEntities();
700 auto verticesOffset = node->getVertices().size();
701 vector<Vector3> vertices = node->getVertices();
702 auto normalsOffset = node->getNormals().size();
703 vector<Vector3> normals = node->getNormals();;
704 auto textureCoordinatesOffset = node->getTextureCoordinates().size();
705 auto textureCoordinates = node->getTextureCoordinates();
706 auto xmlLibraryGeometries = getChildrenByTagName(xmlRoot, "library_geometries").at(0);
707 for (auto xmlGeometry: getChildrenByTagName(xmlLibraryGeometries, "geometry")) {
708 if (string(AVOID_NULLPTR_STRING(xmlGeometry->Attribute("id"))) == xmlNodeId) {
709 auto xmlMesh = getChildrenByTagName(xmlGeometry, "mesh").at(0);
710 vector<TiXmlElement*> xmlPolygonsList;
711 // try to read from triangles
712 for (auto xmlTriangesElement: getChildrenByTagName(xmlMesh, "triangles")) {
713 xmlPolygonsList.push_back(xmlTriangesElement);
714 }
715 // try to read from polylist
716 for (auto xmlPolyListElement: getChildrenByTagName(xmlMesh, "polylist")) {
717 xmlPolygonsList.push_back(xmlPolyListElement);
718 }
719 // try to read from polygons
720 for (auto xmlPolygonsElement: getChildrenByTagName(xmlMesh, "polygons")) {
721 xmlPolygonsList.push_back(xmlPolygonsElement);
722 }
723 // parse from xml polygons elements
724 for (auto xmlPolygons: xmlPolygonsList) {
725 vector<Face> faces;
726 FacesEntity facesEntity(node, xmlNodeId);
727 if (StringTools::toLowerCase((xmlPolygons->Value())) == "polylist") {
729 t.tokenize(string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlPolygons, "vcount").at(0)->GetText())), " \t\n\r\f");
730 while (t.hasMoreTokens()) {
731 auto vertexCount = Integer::parse(t.nextToken());
732 if (vertexCount != 3) {
734 "we only support triangles in '" +
735 (xmlNodeId) +
736 "'"
737 );
738 }
739 }
740 }
741 auto xmlInputs = -1;
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;
750 auto xmlMaterialId = string(AVOID_NULLPTR_STRING(xmlPolygons->Attribute("material")));
751 auto materialSymbolIt = materialSymbols.find(xmlMaterialId);
752 if (materialSymbolIt != materialSymbols.end()) {
753 xmlMaterialId = materialSymbolIt->second;
754 xmlMaterialId = StringTools::substring(xmlMaterialId, 1);
755 }
756
757 if (xmlMaterialId.length() > 0) {
758 Material* material = nullptr;
759 auto materialIt = model->getMaterials().find(xmlMaterialId);
760 if (materialIt != model->getMaterials().end()) {
761 material = materialIt->second;
762 } else {
763 // parse material as we do not have it yet
764 material = readMaterial(pathName, model, xmlRoot, xmlMaterialId);
765 }
766 // set it up
767 facesEntity.setMaterial(material);
768 }
769 unordered_set<int32_t> xmlInputSet;
770 for (auto xmlTrianglesInput: getChildrenByTagName(xmlPolygons, "input")) {
771 // check for vertices sources
772 if (string(AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute("semantic"))) == "VERTEX") {
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);
776 } else
777 // check for normals sources
778 if (string(AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute("semantic"))) == "NORMAL") {
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);
782 } else
783 // check for texture coordinate sources
784 if (string(AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute("semantic"))) == "TEXCOORD") {
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);
788 } else
789 // check for color coordinate sources
790 if (string(AVOID_NULLPTR_STRING(xmlTrianglesInput->Attribute("semantic"))) == "COLOR") {
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);
794 }
795 }
796 xmlInputs = xmlInputSet.size();
797 // get vertices source
798 for (auto xmlVertices: getChildrenByTagName(xmlMesh, "vertices")) {
799 if (string(AVOID_NULLPTR_STRING(xmlVertices->Attribute("id"))) == xmlVerticesSource) {
800 for (auto xmlVerticesInput: getChildrenByTagName(xmlVertices, "input")) {
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);
803 } else
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);
806 }
807 }
808 }
809 }
810 // check for triangles vertices sources
811 if (xmlVerticesSource.length() == 0) {
813 "Could not determine triangles vertices source for '" +
814 (xmlNodeId) +
815 "'"
816 );
817 }
818 // check for triangles normals sources
819 if (xmlNormalsSource.length() == 0) {
821 "Could not determine triangles normal source for '" +
822 (xmlNodeId) +
823 "'"
824 );
825 }
826 // load vertices, normals, texture coordinates
827 for (auto xmlMeshSource: getChildrenByTagName(xmlMesh, "source")) {
828 // vertices
829 if (string(AVOID_NULLPTR_STRING(xmlMeshSource->Attribute("id"))) == xmlVerticesSource) {
830 auto xmlFloatArray = getChildrenByTagName(xmlMeshSource, "float_array").at(0);
831 auto valueString = string(AVOID_NULLPTR_STRING(xmlFloatArray->GetText()));
833 t.tokenize(valueString, " \n\r");
834 while (t.hasMoreTokens()) {
835 float x = Float::parse(t.nextToken());
836 float y = Float::parse(t.nextToken());
837 float z = Float::parse(t.nextToken());
838 vertices.push_back(Vector3(x, y, z));
839 }
840 } else
841 // normals
842 if (string(AVOID_NULLPTR_STRING(xmlMeshSource->Attribute("id"))) == xmlNormalsSource) {
843 auto xmlFloatArray = getChildrenByTagName(xmlMeshSource, "float_array").at(0);
844 auto valueString = string(AVOID_NULLPTR_STRING(xmlFloatArray->GetText()));
846 t.tokenize(valueString, " \n\r");
847 while (t.hasMoreTokens()) {
848 float x = Float::parse(t.nextToken());
849 float y = Float::parse(t.nextToken());
850 float z = Float::parse(t.nextToken());
851 normals.push_back(Vector3(x, y, z));
852 }
853 } else
854 // texture coordinates
855 if (xmlTexCoordSource.length() > 0) {
856 if (string(AVOID_NULLPTR_STRING(xmlMeshSource->Attribute("id"))) == xmlTexCoordSource) {
857 auto xmlFloatArray = getChildrenByTagName(xmlMeshSource, "float_array").at(0);
858 auto valueString = string(AVOID_NULLPTR_STRING(xmlFloatArray->GetText()));
860 t.tokenize(valueString, " \n\r");
861 while (t.hasMoreTokens()) {
862 float u = Float::parse(t.nextToken());
863 float v = Float::parse(t.nextToken());
864 textureCoordinates.push_back(TextureCoordinate(u, v));
865 }
866 }
867 }
868 }
869 // load faces
870 for (auto xmlPolygon: getChildrenByTagName(xmlPolygons, "p")) {
871 auto valueString = string(AVOID_NULLPTR_STRING(xmlPolygon->GetText()));
873 t.tokenize(valueString, " \n\r");
874 array<int32_t, 3> vi;
875 auto viIdx = 0;
876 array<int32_t, 3> ni;
877 auto niIdx = 0;
878 array<int32_t, 3> ti;
879 auto tiIdx = 0;
880 auto valueIdx = 0;
881 auto valid = true;
882 while (t.hasMoreTokens()) {
883 auto value = Integer::parse(t.nextToken());
884 if (valueIdx % xmlInputs == xmlVerticesOffset) {
885 vi[viIdx++] = value;
886 // validate
887 if (value < 0 || value >= vertices.size() - verticesOffset) {
888 valid = false;
889 }
890 // fix for some strange models
891 if (xmlNormalsSource.length() > 0 && xmlNormalsOffset == -1) {
892 ni[niIdx++] = value;
893 if (value < 0 || value >= normals.size() - normalsOffset) {
894 valid = false;
895 }
896 }
897 }
898 if (xmlNormalsOffset != -1 && valueIdx % xmlInputs == xmlNormalsOffset) {
899 ni[niIdx++] = value;
900 // validate
901 if (value < 0 || value >= normals.size() - normalsOffset) {
902 valid = false;
903 }
904 }
905 if (xmlTexCoordOffset != -1 && valueIdx % xmlInputs == xmlTexCoordOffset) {
906 ti[tiIdx++] = value;
907 // validate
908 if (value < 0 || value >= textureCoordinates.size() - textureCoordinatesOffset) {
909 valid = false;
910 }
911 }
912 if (viIdx == 3 && niIdx == 3 && (xmlTexCoordSource.length() == 0 || tiIdx == 3)) {
913 // only add valid faces
914 if (valid == true) {
915 // add face
916 Face f(
917 node,
918 vi[0] + verticesOffset,
919 vi[1] + verticesOffset,
920 vi[2] + verticesOffset,
921 ni[0] + normalsOffset,
922 ni[1] + normalsOffset,
923 ni[2] + normalsOffset
924 );
925 if (xmlTexCoordSource.length() != 0) {
927 ti[0] + textureCoordinatesOffset,
928 ti[1] + textureCoordinatesOffset,
929 ti[2] + textureCoordinatesOffset
930 );
931 }
932 faces.push_back(f);
933 }
934 viIdx = 0;
935 niIdx = 0;
936 tiIdx = 0;
937 valid = true;
938 }
939 valueIdx++;
940 }
941 }
942 // add faces entities if we have any
943 if (faces.empty() == false) {
944 facesEntity.setFaces(faces);
945 facesEntities.push_back(facesEntity);
946 }
947 }
948 }
949 }
950
951 // set up node
952 node->setVertices(vertices);
953 node->setNormals(normals);
954 node->setTextureCoordinates(textureCoordinates);
955 node->setFacesEntities(facesEntities);
956}
957
958Material* DAEReader::readMaterial(const string& pathName, Model* model, TiXmlElement* xmlRoot, const string& xmlNodeId)
959{
960 // determine effect id
961 string xmlEffectId;
962 auto xmlLibraryMaterials = getChildrenByTagName(xmlRoot, "library_materials").at(0);
963 for (auto xmlMaterial: getChildrenByTagName(xmlLibraryMaterials, "material")) {
964 if (string(AVOID_NULLPTR_STRING(xmlMaterial->Attribute("id"))) == xmlNodeId) {
965 auto xmlInstanceEffect = getChildrenByTagName(xmlMaterial, "instance_effect").at(0);
966 xmlEffectId = StringTools::substring(string(AVOID_NULLPTR_STRING(xmlInstanceEffect->Attribute("url"))), 1);
967 }
968 }
969 if (xmlEffectId.length() == 0) {
970 Console::println(
971 string(
972 "Could not determine effect id for '" +
973 xmlNodeId +
974 "'"
975 )
976 );
977 return nullptr;
978 }
979 // parse effect
980 auto material = new Material(xmlNodeId);
981 auto specularMaterialProperties = new SpecularMaterialProperties();
982 string xmlDiffuseTextureId;
983 string xmlTransparencyTextureId;
984 string xmlSpecularTextureId;
985 string xmlBumpTextureId;
986 auto xmlLibraryEffects = getChildrenByTagName(xmlRoot, "library_effects").at(0);
987 for (auto xmlEffect: getChildrenByTagName(xmlLibraryEffects, "effect")) {
988 if (string(AVOID_NULLPTR_STRING(xmlEffect->Attribute("id"))) == xmlEffectId) {
989 auto xmlProfile = getChildrenByTagName(xmlEffect, "profile_COMMON").at(0);
990 // mappings
991 map<string, string> samplerSurfaceMapping;
992 map<string, string> surfaceImageMapping;
993 for (auto xmlNewParam: getChildrenByTagName(xmlProfile, "newparam")) {
994 auto xmlNewParamSID = string(AVOID_NULLPTR_STRING(xmlNewParam->Attribute("sid")));
995 for (auto xmlSurface: getChildrenByTagName(xmlNewParam, "surface"))
996 for (auto xmlSurfaceInitFrom: getChildrenByTagName(xmlSurface, "init_from")) {
997 surfaceImageMapping[xmlNewParamSID] =
998 string(AVOID_NULLPTR_STRING(xmlSurfaceInitFrom->GetText()));
999 }
1000 for (auto xmlSampler2D: getChildrenByTagName(xmlNewParam, "sampler2D"))
1001 for (auto xmlSampler2DSource: getChildrenByTagName(xmlSampler2D, "source")) {
1002 samplerSurfaceMapping[xmlNewParamSID] =
1003 string(AVOID_NULLPTR_STRING(xmlSampler2DSource->GetText()));
1004 }
1005 }
1006 for (auto xmlTechnique: getChildrenByTagName(xmlProfile, "technique")) {
1007 for (auto xmlTechniqueNode: getChildren(xmlTechnique)) {
1008 for (auto xmlDiffuse: getChildrenByTagName(xmlTechniqueNode, "transparent")) {
1009 for (auto xmlTexture: getChildrenByTagName(xmlDiffuse, "texture")) {
1010 xmlTransparencyTextureId = string(AVOID_NULLPTR_STRING(xmlTexture->Attribute("texture")));
1011
1012 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlTransparencyTextureId);
1013 string sample2Surface;
1014 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1015 sample2Surface = sample2SurfaceIt->second;
1016 }
1017 if (sample2Surface.length() == 0) continue;
1018
1019 string surface2Image;
1020 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1021 if (surface2ImageIt != surfaceImageMapping.end()) {
1022 surface2Image = surface2ImageIt->second;
1023 }
1024 if (surface2Image.length() > 0) {
1025 xmlTransparencyTextureId = surface2Image;
1026 }
1027 }
1028 }
1029 // diffuse
1030 for (auto xmlDiffuse: getChildrenByTagName(xmlTechniqueNode, "diffuse")) {
1031 // color
1032 for (auto xmlColor: getChildrenByTagName(xmlDiffuse, "color")) {
1034 t.tokenize(string(AVOID_NULLPTR_STRING(xmlColor->GetText())), " ");
1035 array<float, 4> colorArray;
1036 for (auto i = 0; i < colorArray.size(); i++) {
1037 colorArray[i] = Float::parse(t.nextToken());
1038 }
1039 specularMaterialProperties->setDiffuseColor(Color4(colorArray));
1040 }
1041 // texture
1042 for (auto xmlTexture: getChildrenByTagName(xmlDiffuse, "texture")) {
1043 xmlDiffuseTextureId = string(AVOID_NULLPTR_STRING(xmlTexture->Attribute("texture")));
1044
1045 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlDiffuseTextureId);
1046 string sample2Surface;
1047 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1048 sample2Surface = sample2SurfaceIt->second;
1049 }
1050 if (sample2Surface.length() == 0) continue;
1051
1052 string surface2Image;
1053 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1054 if (surface2ImageIt != surfaceImageMapping.end()) {
1055 surface2Image = surface2ImageIt->second;
1056 }
1057 if (surface2Image.length() > 0) {
1058 xmlDiffuseTextureId = surface2Image;
1059 }
1060 }
1061 }
1062 // ambient
1063 for (auto xmlAmbient: getChildrenByTagName(xmlTechniqueNode, "ambient")) {
1064 // color
1065 for (auto xmlColor: getChildrenByTagName(xmlAmbient, "color")) {
1067 t.tokenize(string(AVOID_NULLPTR_STRING(xmlColor->GetText())), " ");
1068 array<float, 4> colorArray;
1069 for (auto i = 0; i < colorArray.size(); i++) {
1070 colorArray[i] = Float::parse(t.nextToken());
1071 }
1072 specularMaterialProperties->setAmbientColor(Color4(colorArray));
1073 }
1074 }
1075 // emission
1076 for (auto xmlEmission: getChildrenByTagName(xmlTechniqueNode, "emission")) {
1077 // color
1078 for (auto xmlColor: getChildrenByTagName(xmlEmission, "color")) {
1080 t.tokenize(string(AVOID_NULLPTR_STRING(xmlColor->GetText())), " ");
1081 array<float, 4> colorArray;
1082 for (auto i = 0; i < colorArray.size(); i++) {
1083 colorArray[i] = Float::parse(t.nextToken());
1084 }
1085 specularMaterialProperties->setEmissionColor(Color4(colorArray));
1086 }
1087 }
1088 // specular
1089 auto hasSpecularMap = false;
1090 auto hasSpecularColor = false;
1091 for (auto xmlSpecular: getChildrenByTagName(xmlTechniqueNode, "specular")) {
1092 // texture
1093 for (auto xmlTexture: getChildrenByTagName(xmlSpecular, "texture")) {
1094 xmlSpecularTextureId = string(AVOID_NULLPTR_STRING(xmlTexture->Attribute("texture")));
1095
1096 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlSpecularTextureId);
1097 string sample2Surface;
1098 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1099 sample2Surface = sample2SurfaceIt->second;
1100 }
1101 if (sample2Surface.length() == 0) continue;
1102
1103 string surface2Image;
1104 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1105 if (surface2ImageIt != surfaceImageMapping.end()) {
1106 surface2Image = surface2ImageIt->second;
1107 }
1108
1109 if (surface2Image.length() > 0) {
1110 xmlSpecularTextureId = surface2Image;
1111 hasSpecularMap = true;
1112 }
1113 }
1114 // color
1115 for (auto xmlColor: getChildrenByTagName(xmlSpecular, "color")) {
1117 t.tokenize(string(AVOID_NULLPTR_STRING(xmlColor->GetText())), " ");
1118 array<float, 4> colorArray;
1119 for (auto i = 0; i < colorArray.size(); i++) {
1120 colorArray[i] = Float::parse(t.nextToken());
1121 }
1122 specularMaterialProperties->setSpecularColor(Color4(colorArray));
1123 hasSpecularColor = true;
1124 }
1125 }
1126 if (hasSpecularMap == true && hasSpecularColor == false) {
1127 specularMaterialProperties->setSpecularColor(Color4(1.0f, 1.0f, 1.0f, 1.0f));
1128 }
1129 // shininess
1130 for (auto xmlShininess: getChildrenByTagName(xmlTechniqueNode, "shininess"))
1131 for (auto xmlFloat: getChildrenByTagName(xmlShininess, "float")) {
1132 specularMaterialProperties->setShininess(Float::parse(string(AVOID_NULLPTR_STRING(xmlFloat->GetText()))));
1133 }
1134 }
1135 // normal/bump texture
1136 for (auto xmlBumpExtra: getChildrenByTagName(xmlTechnique, "extra"))
1137 for (auto xmlBumpTechnique: getChildrenByTagName(xmlBumpExtra, "technique"))
1138 for (auto xmlBumpTechniqueBump: getChildrenByTagName(xmlBumpTechnique, "bump"))
1139 for (auto xmlBumpTexture: getChildrenByTagName(xmlBumpTechniqueBump, "texture")) {
1140 xmlBumpTextureId = string(AVOID_NULLPTR_STRING(xmlBumpTexture->Attribute("texture")));
1141
1142 auto sample2SurfaceIt = samplerSurfaceMapping.find(xmlBumpTextureId);
1143 string sample2Surface;
1144 if (sample2SurfaceIt != samplerSurfaceMapping.end()) {
1145 sample2Surface = sample2SurfaceIt->second;
1146 }
1147 if (sample2Surface.length() == 0) continue;
1148
1149 string surface2Image;
1150 auto surface2ImageIt = surfaceImageMapping.find(sample2Surface);
1151 if (surface2ImageIt != surfaceImageMapping.end()) {
1152 surface2Image = surface2ImageIt->second;
1153 }
1154
1155 if (surface2Image.length() > 0) xmlBumpTextureId = surface2Image;
1156 }
1157 }
1158 }
1159 }
1160
1161 // diffuse transparency texture
1162 string xmlTransparencyTextureFilename;
1163 if (xmlDiffuseTextureId.length() > 0) {
1164 xmlTransparencyTextureFilename = getTextureFileNameById(xmlRoot, xmlTransparencyTextureId);
1165 // do we have a file name
1166 if (xmlTransparencyTextureFilename.length() > 0) {
1167 // add texture
1168 xmlTransparencyTextureFilename = makeFileNameRelative(xmlTransparencyTextureFilename);
1169 }
1170 }
1171
1172 // diffuse texture
1173 string xmlDiffuseTextureFilename;
1174 if (xmlDiffuseTextureId.length() > 0) {
1175 xmlDiffuseTextureFilename = getTextureFileNameById(xmlRoot, xmlDiffuseTextureId);
1176 // do we have a file name
1177 if (xmlDiffuseTextureFilename.length() > 0) {
1178 xmlDiffuseTextureFilename = makeFileNameRelative(xmlDiffuseTextureFilename);
1179 // add texture
1180 specularMaterialProperties->setDiffuseTexture(pathName, xmlDiffuseTextureFilename, pathName, xmlTransparencyTextureFilename);
1181 if (specularMaterialProperties->hasDiffuseTextureTransparency() == true) specularMaterialProperties->setDiffuseTextureMaskedTransparency(true);
1182 }
1183 }
1184
1185 // specular texture
1186 string xmlSpecularTextureFilename;
1187 if (xmlSpecularTextureId.length() > 0) {
1188 xmlSpecularTextureFilename = getTextureFileNameById(xmlRoot, xmlSpecularTextureId);
1189 // do we have a file name
1190 if (xmlSpecularTextureFilename.length() > 0) {
1191 xmlSpecularTextureFilename = makeFileNameRelative(xmlSpecularTextureFilename);
1192 // add texture
1193 specularMaterialProperties->setSpecularTexture(pathName, xmlSpecularTextureFilename);
1194 }
1195 }
1196
1197 // normal/bump texture
1198 string xmlBumpTextureFilename;
1199 if (xmlBumpTextureId.length() > 0) {
1200 xmlBumpTextureFilename = getTextureFileNameById(xmlRoot, xmlBumpTextureId);
1201 // do we have a file name
1202 if (xmlBumpTextureFilename.length() > 0) {
1203 xmlBumpTextureFilename = makeFileNameRelative(xmlBumpTextureFilename);
1204 // add texture
1205 specularMaterialProperties->setNormalTexture(pathName, xmlBumpTextureFilename);
1206 }
1207 }
1208
1209 /*
1210 // determine displacement map file name
1211 string xmlDisplacementFilename;
1212 // add texture
1213 if (xmlDisplacementFilename.length() > 0) {
1214 specularMaterialProperties->setDisplacementTexture(pathName, xmlDisplacementFilename);
1215 }
1216 */
1217
1218 // adjust ambient light with blender
1219 if (model->getAuthoringTool() == Model::AUTHORINGTOOL_BLENDER && specularMaterialProperties->getAmbientColor().equals(BLENDER_AMBIENT_NONE)) {
1220 specularMaterialProperties->setAmbientColor(
1221 Color4(
1222 specularMaterialProperties->getDiffuseColor().getRed() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
1223 specularMaterialProperties->getDiffuseColor().getGreen() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
1224 specularMaterialProperties->getDiffuseColor().getBlue() * BLENDER_AMBIENT_FROM_DIFFUSE_SCALE,
1225 1.0f
1226 )
1227 );
1228 specularMaterialProperties->setDiffuseColor(
1229 Color4(
1230 specularMaterialProperties->getDiffuseColor().getRed() * BLENDER_DIFFUSE_SCALE,
1231 specularMaterialProperties->getDiffuseColor().getGreen() * BLENDER_DIFFUSE_SCALE,
1232 specularMaterialProperties->getDiffuseColor().getBlue() * BLENDER_DIFFUSE_SCALE,
1233 specularMaterialProperties->getDiffuseColor().getAlpha()
1234 )
1235 );
1236 }
1237
1238 // add specular material properties
1239 material->setSpecularMaterialProperties(specularMaterialProperties);
1240
1241 // add material to library
1242 model->getMaterials()[material->getId()] = material;
1243
1244 //
1245 return material;
1246}
1247
1248const string DAEReader::makeFileNameRelative(const string& fileName)
1249{
1250 // check if absolute path
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);
1258 } else {
1259 return StringTools::substring(fileName, indexBackslash + 1);
1260 }
1261 }
1262 }
1263 return fileName;
1264}
1265
1266const string DAEReader::getTextureFileNameById(TiXmlElement* xmlRoot, const string& xmlTextureId)
1267{
1268 string xmlTextureFilename;
1269 auto xmlLibraryImages = getChildrenByTagName(xmlRoot, "library_images").at(0);
1270 for (auto xmlImage: getChildrenByTagName(xmlLibraryImages, "image")) {
1271 if (string(AVOID_NULLPTR_STRING(xmlImage->Attribute("id"))) == xmlTextureId) {
1272 xmlTextureFilename = string(AVOID_NULLPTR_STRING(getChildrenByTagName(xmlImage, "init_from").at(0)->GetText()));
1273 if (StringTools::startsWith(xmlTextureFilename, "file://") == true) {
1274 xmlTextureFilename = StringTools::substring(xmlTextureFilename, 7);
1275 }
1276 break;
1277 }
1278 }
1279 return xmlTextureFilename;
1280}
1281
1282const vector<TiXmlElement*> DAEReader::getChildrenByTagName(TiXmlElement* parent, const char* name)
1283{
1284 vector<TiXmlElement*> elementList;
1285 for (auto *child = parent->FirstChildElement(name); child != nullptr; child = child->NextSiblingElement(name)) {
1286 elementList.push_back(child);
1287 }
1288 return elementList;
1289}
1290
1291const vector<TiXmlElement*> DAEReader::getChildren(TiXmlElement* parent)
1292{
1293 vector<TiXmlElement*> elementList;
1294 for (auto *child = parent->FirstChildElement(); child != nullptr; child = child->NextSiblingElement()) {
1295 elementList.push_back(child);
1296 }
1297 return elementList;
1298}
#define AVOID_NULLPTR_STRING(arg)
Definition: DAEReader.cpp:46
Transformations which contain scale, rotations and translation.
Collada DAE model reader.
Definition: DAEReader.h:40
static void readGeometry(const string &pathName, Model *model, Node *node, TiXmlElement *xmlRoot, const string &xmlNodeId, const map< string, string > &materialSymbols)
Reads a geometry.
Definition: DAEReader.cpp:697
static const vector< TiXmlElement * > getChildren(TiXmlElement *parent)
Returns immediate children tags.
Definition: DAEReader.cpp:1291
static constexpr float BLENDER_AMBIENT_FROM_DIFFUSE_SCALE
Definition: DAEReader.h:44
static Material * readMaterial(const string &pathName, Model *model, TiXmlElement *xmlRoot, const string &xmlNodeId)
Reads a material.
Definition: DAEReader.cpp:958
static constexpr float BLENDER_DIFFUSE_SCALE
Definition: DAEReader.h:45
static UpVector * getUpVector(TiXmlElement *xmlRoot)
Get Up vector.
Definition: DAEReader.cpp:202
static Node * readVisualSceneInstanceController(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode)
Reads a instance controller.
Definition: DAEReader.cpp:478
static Node * readNode(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode, float fps)
Reads a DAE visual scene node node.
Definition: DAEReader.cpp:268
static void setupModelImportRotationMatrix(TiXmlElement *xmlRoot, Model *model)
Set up model import rotation matrix.
Definition: DAEReader.cpp:224
static const string makeFileNameRelative(const string &fileName)
Make file name relative.
Definition: DAEReader.cpp:1248
static const string getTextureFileNameById(TiXmlElement *xmlRoot, const string &xmlTextureId)
Get texture file name by id.
Definition: DAEReader.cpp:1266
static STATIC_DLL_IMPEXT const Color4 BLENDER_AMBIENT_NONE
Definition: DAEReader.h:43
static void setupModelImportScaleMatrix(TiXmlElement *xmlRoot, Model *model)
Set up model import scale matrix.
Definition: DAEReader.cpp:244
static const vector< TiXmlElement * > getChildrenByTagName(TiXmlElement *parent, const char *name)
Returns immediate children tags by tag name.
Definition: DAEReader.cpp:1282
static Model::AuthoringTool getAuthoringTool(TiXmlElement *xmlRoot)
Get authoring tool.
Definition: DAEReader.cpp:188
static Node * readVisualSceneNode(const string &pathName, Model *model, Node *parentNode, TiXmlElement *xmlRoot, TiXmlElement *xmlNode, float fps)
Read a DAE visual scene node.
Definition: DAEReader.cpp:258
Color 4 base definition class.
Definition: Color4Base.h:19
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
void setTextureCoordinateIndices(int32_t vt0, int32_t vt1, int32_t vt2)
Set up optional texture coordinate indices.
Definition: Face.cpp:45
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
Represents a material.
Definition: Material.h:21
Representation of a 3d model.
Definition: Model.h:32
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
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
Definition: Model.h:301
map< string, Material * > & getMaterials()
Returns all object materials.
Definition: Model.h:171
Model node.
Definition: Node.h:31
void setTextureCoordinates(const vector< TextureCoordinate > &textureCoordinates)
Set texture coordinates.
Definition: Node.cpp:70
void setVertices(const vector< Vector3 > &vertices)
Set vertices.
Definition: Node.cpp:50
void setFacesEntities(const vector< FacesEntity > &facesEntities)
Set up faces entities.
Definition: Node.cpp:125
const vector< TextureCoordinate > & getTextureCoordinates() const
Definition: Node.h:186
const vector< Vector3 > & getNormals() const
Definition: Node.h:173
const vector< FacesEntity > & getFacesEntities() const
Definition: Node.h:256
void setNormals(const vector< Vector3 > &normals)
Set normals.
Definition: Node.cpp:60
const vector< Vector3 > & getVertices() const
Definition: Node.h:151
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
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 & transpose()
Transposes this matrix.
Definition: Matrix4x4.h:510
Matrix4x4 clone()
Clones this matrix.
Definition: Matrix4x4.h:624
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
3D vector 3 class
Definition: Vector3.h:22
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
Float class.
Definition: Float.h:23
Integer class.
Definition: Integer.h:26
Model tools functions class.
Definition: ModelTools.h:38
String tokenizer class.
void tokenize(const string &str, const string &delimiters)
Tokenize.
String tools class.
Definition: StringTools.h:20
An attribute is a name-value pair.
Definition: tinyxml.h:734
Always the top level node.
Definition: tinyxml.h:1317
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.
Definition: tinyxml.h:1371
const char * ErrorDesc() const
Contains a textual (english) description of the error if one occurs.
Definition: tinyxml.h:1382
bool Error() const
If an error occurs, Error will be set to true.
Definition: tinyxml.h:1379
The element is a container class.
Definition: tinyxml.h:886
const char * Attribute(const char *name) const
Given an attribute name, Attribute() returns the value for the attribute of that name,...
Definition: tinyxml.cpp:564
const TiXmlElement * NextSiblingElement() const
Convenience function to get through elements.
Definition: tinyxml.cpp:471
const TiXmlElement * FirstChildElement() const
Convenience function to get through elements.
Definition: tinyxml.cpp:441
std::exception Exception
Exception base class.
Definition: Exception.h:19