TDME2 1.9.121
ModelTools.cpp
Go to the documentation of this file.
2
3#include <array>
4#include <map>
5#include <string>
6#include <unordered_set>
7#include <vector>
8
9#include <tdme/tdme.h>
28#include <tdme/math/Matrix4x4.h>
29#include <tdme/math/Vector2.h>
30#include <tdme/math/Vector3.h>
34
35using std::array;
36using std::map;
37using std::string;
38using std::to_string;
39using std::unordered_set;
40using std::vector;
41
67
68ModelTools::VertexOrder ModelTools::determineVertexOrder(const vector<Vector3>& vertices)
69{
70 auto edgeSum = 0;
71 for (auto i = 0; i < vertices.size(); i++) {
72 auto& currentVertexXYZ = vertices[i].getArray();
73 auto& nextVertexXYZ = vertices[(i + 1) % vertices.size()].getArray();
74 edgeSum +=
75 (nextVertexXYZ[0] - currentVertexXYZ[0]) * (nextVertexXYZ[1] - currentVertexXYZ[1]) * (nextVertexXYZ[2] - currentVertexXYZ[2]);
76 }
77 if (edgeSum >= 0) {
79 } else {
81 }
82}
83
85{
87}
88
89void ModelTools::prepareForIndexedRendering(const map<string, Node*>& nodes)
90{
91 // we need to prepare the node for indexed rendering
92 for (auto it: nodes) {
93 Node* node = it.second;
94 auto& nodeVertices = node->getVertices();
95 auto& nodeNormals = node->getNormals();
96 auto& nodeTextureCoordinates = node->getTextureCoordinates();
97 auto& nodeTangents = node->getTangents();
98 auto& nodeBitangents = node->getBitangents();
99 auto& nodeOrigins = node->getOrigins();
100 vector<int32_t> vertexMapping;
101 vector<Vector3> indexedVertices;
102 vector<Vector3> indexedNormals;
103 vector<TextureCoordinate> indexedTextureCoordinates;
104 vector<Vector3> indexedTangents;
105 vector<Vector3> indexedBitangents;
106 vector<Vector3> indexedOrigins;
107 // construct indexed vertex data suitable for indexed rendering
108 auto preparedIndices = 0;
109 auto newFacesEntities = node->getFacesEntities();
110 for (auto& newFacesEntity: newFacesEntities) {
111 auto newFaces = newFacesEntity.getFaces();
112 for (auto& face: newFaces) {
113 auto faceVertexIndices = face.getVertexIndices();
114 auto faceNormalIndices = face.getNormalIndices();
115 auto faceTextureIndices = face.getTextureCoordinateIndices();
116 auto faceTangentIndices = face.getTangentIndices();
117 auto faceBitangentIndices = face.getBitangentIndices();
118 array<int32_t, 3> indexedFaceVertexIndices;
119 for (int16_t idx = 0; idx < 3; idx++) {
120 auto nodeVertexIndex = faceVertexIndices[idx];
121 auto nodeNormalIndex = faceNormalIndices[idx];
122 auto nodeTextureCoordinateIndex = faceTextureIndices[idx];
123 auto nodeTangentIndex = faceTangentIndices[idx];
124 auto nodeBitangentIndex = faceBitangentIndices[idx];
125 auto vertex = &nodeVertices[nodeVertexIndex];
126 auto normal = &nodeNormals[nodeNormalIndex];
127 auto textureCoordinate = nodeTextureCoordinates.size() > 0 ? &nodeTextureCoordinates[nodeTextureCoordinateIndex] : static_cast< TextureCoordinate* >(nullptr);
128 auto tangent = nodeTangents.size() > 0 ? &nodeTangents[nodeTangentIndex] : static_cast< Vector3* >(nullptr);
129 auto bitangent = nodeBitangents.size() > 0 ? &nodeBitangents[nodeBitangentIndex] : static_cast< Vector3* >(nullptr);
130 auto origin = nodeOrigins.size() > 0 ? &nodeOrigins[nodeVertexIndex] : static_cast< Vector3* >(nullptr);
131 auto newIndex = preparedIndices;
132 for (auto i = 0; i < preparedIndices; i++)
133 if (indexedVertices[i].equals(*vertex) &&
134 indexedNormals[i].equals(*normal) &&
135 (textureCoordinate == nullptr || indexedTextureCoordinates[i].equals(*textureCoordinate)) &&
136 (tangent == nullptr || indexedTangents[i].equals(*tangent)) &&
137 (bitangent == nullptr || indexedBitangents[i].equals(*bitangent))) {
138 newIndex = i;
139 break;
140 }
141 if (newIndex == preparedIndices) {
142 vertexMapping.push_back(nodeVertexIndex);
143 indexedVertices.push_back(*vertex);
144 indexedNormals.push_back(*normal);
145 if (textureCoordinate != nullptr) indexedTextureCoordinates.push_back(*textureCoordinate);
146 if (tangent != nullptr) indexedTangents.push_back(*tangent);
147 if (bitangent != nullptr) indexedBitangents.push_back(*bitangent);
148 if (origin != nullptr) indexedOrigins.push_back(*origin);
149 preparedIndices++;
150 }
151 indexedFaceVertexIndices[idx] = newIndex;
152 }
153 face.setIndexedRenderingIndices(indexedFaceVertexIndices);
154 }
155 newFacesEntity.setFaces(newFaces);
156 }
157 node->setFacesEntities(newFacesEntities);
158 // remap skinning
159 auto skinning = node->getSkinning();
160 if (skinning != nullptr) {
161 prepareForIndexedRendering(skinning, vertexMapping, preparedIndices);
162 }
163 node->setVertices(indexedVertices);
164 node->setNormals(indexedNormals);
165 if (nodeTextureCoordinates.size() > 0) {
166 node->setTextureCoordinates(indexedTextureCoordinates);
167 }
168 node->setTangents(indexedTangents);
169 node->setBitangents(indexedBitangents);
170 if (nodeOrigins.size() > 0) {
171 node->setOrigins(indexedOrigins);
172 }
173 // process sub nodes
175 }
176}
177
178void ModelTools::prepareForIndexedRendering(Skinning* skinning, const vector<int32_t>& vertexMapping, int32_t vertices)
179{
180 auto& originalVerticesJointsWeights = skinning->getVerticesJointsWeights();
181 vector<vector<JointWeight>> verticesJointsWeights;
182 verticesJointsWeights.resize(vertices);
183 for (auto i = 0; i < vertices; i++) {
184 auto vertexOriginalMappedToIdx = vertexMapping[i];
185 verticesJointsWeights[i].resize(originalVerticesJointsWeights[vertexOriginalMappedToIdx].size());
186 for (auto j = 0; j < verticesJointsWeights[i].size(); j++) {
187 verticesJointsWeights[i][j] = originalVerticesJointsWeights[vertexOriginalMappedToIdx][j];
188 }
189 }
190 skinning->setVerticesJointsWeights(verticesJointsWeights);
191}
192
194{
195 // determine joints and mark them as joints
196 auto nodes = model->getNodes();
197 for (auto it: model->getSubNodes()) {
198 Node* node = it.second;
199 auto skinning = node->getSkinning();
200 // do we have a skinning
201 if (skinning != nullptr) {
202 // yep
203 for (auto& joint : skinning->getJoints()) {
204 auto jointNodeIt = nodes.find(joint.getNodeId());
205 if (jointNodeIt != nodes.end()) {
206 setJoint(jointNodeIt->second);
207 }
208 }
209 }
210 }
211}
212
214{
215 root->setJoint(true);
216 for (auto it: root->getSubNodes()) {
217 Node* node = it.second;
218 setJoint(node);
219 }
220}
221
223{
224 // fix animation length
225 auto defaultAnimation = model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT);
226 if (defaultAnimation != nullptr) {
227 for (auto it: model->getSubNodes()) {
228 Node* node = it.second;
229 fixAnimationLength(node, defaultAnimation->getFrames());
230 }
231 }
232}
233
234void ModelTools::fixAnimationLength(Node* root, int32_t frames)
235{
236 auto animation = root->getAnimation();
237 if (animation != nullptr) {
238 vector<Matrix4x4> newTransformationsMatrices;
239 auto oldTransformationsMatrices = root->getAnimation()->getTransformationsMatrices();
240 auto animation = new Animation();
241 newTransformationsMatrices.resize(frames);
242 for (auto i = 0; i < frames; i++) {
243 if (i < oldTransformationsMatrices.size()) {
244 newTransformationsMatrices[i] = oldTransformationsMatrices[i];
245 } else {
246 newTransformationsMatrices[i].identity();
247 }
248 }
249 animation->setTransformationsMatrices(newTransformationsMatrices);
250 root->setAnimation(animation);
251 }
252 for (auto it: root->getSubNodes()) {
253 Node* node = it.second;
254 fixAnimationLength(node, frames);
255 }
256}
257
259 return model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT) != nullptr;
260}
261
262void ModelTools::createDefaultAnimation(Model* model, int32_t frames)
263{
264 // add default model animation setup if not yet done
265 auto defaultAnimation = model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT);
266 if (defaultAnimation == nullptr) {
267 model->addAnimationSetup(Model::ANIMATIONSETUP_DEFAULT, 0, frames - 1, true);
268 } else {
269 // check default animation setup
270 if (defaultAnimation->getStartFrame() != 0 || defaultAnimation->getEndFrame() != frames - 1) {
271 Console::println(string("Warning: default animation mismatch"));
272 }
273 if (frames - 1 > defaultAnimation->getEndFrame()) {
274 Console::println(string("Warning: default animation mismatch, will be fixed"));
275 model->addAnimationSetup(Model::ANIMATIONSETUP_DEFAULT, 0, frames - 1, true);
276 }
277 }
278}
279
280Material* ModelTools::cloneMaterial(const Material* material, const string& id) {
281 auto clonedMaterial = new Material(id.empty()?material->getId():id);
282 auto specularMaterialProperties = material->getSpecularMaterialProperties();
283 if (specularMaterialProperties != nullptr) {
284 auto clonedSpecularMaterialProperties = new SpecularMaterialProperties();
285 clonedSpecularMaterialProperties->setEmbedTextures(specularMaterialProperties->hasEmbeddedTextures());
286 clonedSpecularMaterialProperties->setAmbientColor(specularMaterialProperties->getAmbientColor());
287 clonedSpecularMaterialProperties->setDiffuseColor(specularMaterialProperties->getDiffuseColor());
288 clonedSpecularMaterialProperties->setEmissionColor(specularMaterialProperties->getEmissionColor());
289 clonedSpecularMaterialProperties->setSpecularColor(specularMaterialProperties->getSpecularColor());
290 clonedSpecularMaterialProperties->setShininess(specularMaterialProperties->getShininess());
291 clonedSpecularMaterialProperties->setTextureAtlasSize(specularMaterialProperties->getTextureAtlasSize());
292 if (specularMaterialProperties->getDiffuseTexture() != nullptr) {
293 clonedSpecularMaterialProperties->setDiffuseTexture(specularMaterialProperties->getDiffuseTexture());
294 if (specularMaterialProperties->hasEmbeddedTextures() == false) {
295 clonedSpecularMaterialProperties->setDiffuseTexturePathName(specularMaterialProperties->getDiffuseTexturePathName());
296 clonedSpecularMaterialProperties->setDiffuseTextureFileName(specularMaterialProperties->getDiffuseTextureFileName());
297 }
298 }
299 clonedSpecularMaterialProperties->setDiffuseTextureTransparency(specularMaterialProperties->hasDiffuseTextureTransparency());
300 clonedSpecularMaterialProperties->setDiffuseTextureMaskedTransparency(specularMaterialProperties->hasDiffuseTextureMaskedTransparency());
301 clonedSpecularMaterialProperties->setDiffuseTextureMaskedTransparencyThreshold(specularMaterialProperties->getDiffuseTextureMaskedTransparencyThreshold());
302 if (specularMaterialProperties->getSpecularTexture() != nullptr) {
303 clonedSpecularMaterialProperties->setSpecularTexture(specularMaterialProperties->getSpecularTexture());
304 if (specularMaterialProperties->hasEmbeddedTextures() == false) {
305 clonedSpecularMaterialProperties->setSpecularTexturePathName(specularMaterialProperties->getSpecularTexturePathName());
306 clonedSpecularMaterialProperties->setSpecularTextureFileName(specularMaterialProperties->getSpecularTextureFileName());
307 }
308 }
309 if (specularMaterialProperties->getNormalTexture() != nullptr) {
310 clonedSpecularMaterialProperties->setNormalTexture(specularMaterialProperties->getNormalTexture());
311 if (specularMaterialProperties->hasEmbeddedTextures() == false) {
312 clonedSpecularMaterialProperties->setNormalTexturePathName(specularMaterialProperties->getNormalTexturePathName());
313 clonedSpecularMaterialProperties->setNormalTextureFileName(specularMaterialProperties->getNormalTextureFileName());
314 }
315 }
316 clonedMaterial->setSpecularMaterialProperties(clonedSpecularMaterialProperties);
317 }
318 return clonedMaterial;
319}
320
321void ModelTools::cloneNode(Node* sourceNode, Model* targetModel, Node* targetParentNode, bool cloneMesh) {
322 auto clonedNode = new Node(targetModel, targetParentNode, sourceNode->getId(), sourceNode->getName());
323 clonedNode->setTransformationsMatrix(sourceNode->getTransformationsMatrix());
324 clonedNode->setJoint(sourceNode->isJoint());
325 if (cloneMesh == true) {
326 clonedNode->setVertices(sourceNode->getVertices());
327 clonedNode->setNormals(sourceNode->getNormals());
328 clonedNode->setTextureCoordinates(sourceNode->getTextureCoordinates());
329 clonedNode->setTangents(sourceNode->getTangents());
330 clonedNode->setBitangents(sourceNode->getBitangents());
331 clonedNode->setOrigins(sourceNode->getOrigins());
332 auto facesEntities = clonedNode->getFacesEntities();
333 for (auto& facesEntity: sourceNode->getFacesEntities()) {
334 if (facesEntity.getMaterial() == nullptr) continue;
335 Material* material = nullptr;
336 auto materialIt = targetModel->getMaterials().find(facesEntity.getMaterial()->getId());
337 if (materialIt == targetModel->getMaterials().end()) {
338 material = cloneMaterial(facesEntity.getMaterial());
339 targetModel->getMaterials()[material->getId()] = material;
340 } else {
341 material = materialIt->second;
342 }
343 auto clonedFacesEntity = FacesEntity(clonedNode, facesEntity.getId());
344 clonedFacesEntity.setMaterial(material);
345 clonedFacesEntity.setFaces(facesEntity.getFaces());
346 facesEntities.push_back(clonedFacesEntity);
347 }
348 clonedNode->setFacesEntities(facesEntities);
349 }
350 if (sourceNode->getAnimation() != nullptr) {
351 auto clonedAnimation = new Animation();
352 clonedAnimation->setTransformationsMatrices(sourceNode->getAnimation()->getTransformationsMatrices());
353 clonedNode->setAnimation(clonedAnimation);
354 }
355 targetModel->getNodes()[clonedNode->getId()] = clonedNode;
356 if (targetParentNode == nullptr) {
357 targetModel->getSubNodes()[clonedNode->getId()] = clonedNode;
358 } else {
359 targetParentNode->getSubNodes()[clonedNode->getId()] = clonedNode;
360 }
361 for (auto sourceSubNodeIt: sourceNode->getSubNodes()) {
362 auto subNode = sourceSubNodeIt.second;
363 cloneNode(subNode, targetModel, clonedNode, cloneMesh);
364 }
365}
366
367void ModelTools::partitionNode(Node* sourceNode, map<string, Model*>& modelsByPartition, map<string, Vector3>& modelsPosition, const Matrix4x4& parentTransformationsMatrix) {
368 // TODO: performance: faces handling is very suboptimal currently, however this is only executed in SceneEditor if doing partitioning
369 Vector3 faceCenter;
370
371 Matrix4x4 transformationsMatrix;
372 transformationsMatrix.set(sourceNode->getTransformationsMatrix());
373 transformationsMatrix.multiply(parentTransformationsMatrix);
374
375 Vector3 vertex0;
376 Vector3 vertex1;
377 Vector3 vertex2;
378 Vector3 normal0;
379 Vector3 normal1;
380 Vector3 normal2;
381 TextureCoordinate textureCoordinate0;
382 TextureCoordinate textureCoordinate1;
383 TextureCoordinate textureCoordinate2;
384 Vector3 tangent0;
385 Vector3 tangent1;
386 Vector3 tangent2;
387 Vector3 bitangent0;
388 Vector3 bitangent1;
389 Vector3 bitangent2;
390
391 Vector3 vertex0Transformed;
392 Vector3 vertex1Transformed;
393 Vector3 vertex2Transformed;
394
395 auto sourceNodeId = sourceNode->getModel()->getId();
396 auto sourceNodeName = sourceNode->getModel()->getName();
397
398 // TODO: maybe check if id and node do have real file endings like .tm, .dae, .fbx or something
399 if (StringTools::lastIndexOf(sourceNodeId, '.') != -1) {
400 sourceNodeId = StringTools::substring(sourceNodeId, 0, StringTools::lastIndexOf(sourceNodeId, '.') - 1);
401 }
402 if (StringTools::lastIndexOf(sourceNodeName, '.') != -1) {
403 sourceNodeName = StringTools::substring(sourceNodeName, 0, StringTools::lastIndexOf(sourceNodeName, '.') - 1);
404 }
405
406 //
407 map<string, Node*> partitionModelNodes;
408
409 // partition model node vertices and such
410 map<string, vector<Vector3>> partitionModelNodesVertices;
411 map<string, vector<Vector3>> partitionModelNodesNormals;
412 map<string, vector<TextureCoordinate>> partitionModelNodesTextureCoordinates;
413 map<string, vector<Vector3>> partitionModelNodesTangents;
414 map<string, vector<Vector3>> partitionModelNodesBitangents;
415 map<string, vector<FacesEntity>> partitionModelNodesFacesEntities;
416
417 for (auto& facesEntity: sourceNode->getFacesEntities()) {
418 bool haveTextureCoordinates = facesEntity.isTextureCoordinatesAvailable();
419 bool haveTangentsBitangents = facesEntity.isTangentBitangentAvailable();
420 for (auto& face: facesEntity.getFaces()) {
421 // get face vertices and such
422 auto& vertexIndices = face.getVertexIndices();
423 auto& normalIndices = face.getNormalIndices();
424 auto& textureCoordinatesIndices = face.getTextureCoordinateIndices();
425 auto& tangentIndices = face.getTangentIndices();
426 auto& bitangentIndices = face.getBitangentIndices();
427 vertex0.set(sourceNode->getVertices()[vertexIndices[0]]);
428 vertex1.set(sourceNode->getVertices()[vertexIndices[1]]);
429 vertex2.set(sourceNode->getVertices()[vertexIndices[2]]);
430 normal0.set(sourceNode->getNormals()[normalIndices[0]]);
431 normal1.set(sourceNode->getNormals()[normalIndices[1]]);
432 normal2.set(sourceNode->getNormals()[normalIndices[2]]);
433 if (haveTextureCoordinates == true) {
434 textureCoordinate0.set(sourceNode->getTextureCoordinates()[textureCoordinatesIndices[0]]);
435 textureCoordinate1.set(sourceNode->getTextureCoordinates()[textureCoordinatesIndices[1]]);
436 textureCoordinate2.set(sourceNode->getTextureCoordinates()[textureCoordinatesIndices[2]]);
437 }
438 if (haveTangentsBitangents == true) {
439 tangent0.set(sourceNode->getTangents()[tangentIndices[0]]);
440 tangent1.set(sourceNode->getTangents()[tangentIndices[1]]);
441 tangent2.set(sourceNode->getTangents()[tangentIndices[2]]);
442 bitangent0.set(sourceNode->getBitangents()[bitangentIndices[0]]);
443 bitangent1.set(sourceNode->getBitangents()[bitangentIndices[1]]);
444 bitangent2.set(sourceNode->getBitangents()[bitangentIndices[2]]);
445 }
446
447 // find out partition by transforming vertices into world coordinates
448 vertex0Transformed = transformationsMatrix.multiply(vertex0);
449 vertex1Transformed = transformationsMatrix.multiply(vertex1);
450 vertex2Transformed = transformationsMatrix.multiply(vertex2);
451 faceCenter.set(vertex0Transformed);
452 faceCenter.add(vertex1Transformed);
453 faceCenter.add(vertex2Transformed);
454 faceCenter.scale(1.0f / 3.0f);
455 auto minX = Math::min(Math::min(vertex0Transformed.getX(), vertex1Transformed.getX()), vertex2Transformed.getX());
456 auto minY = Math::min(Math::min(vertex0Transformed.getY(), vertex1Transformed.getY()), vertex2Transformed.getY());
457 auto minZ = Math::min(Math::min(vertex0Transformed.getZ(), vertex1Transformed.getZ()), vertex2Transformed.getZ());
458 auto partitionX = (int)(minX / 64.0f);
459 auto partitionY = (int)(minY / 64.0f);
460 auto partitionZ = (int)(minZ / 64.0f);
461
462 // key
463 string partitionModelKey =
464 to_string(partitionX) + "," +
465 to_string(partitionY) + "," +
466 to_string(partitionZ);
467
468 // get model
469 auto partitionModel = modelsByPartition[partitionModelKey];
470 if (partitionModel == nullptr) {
471 partitionModel = new Model(
472 sourceNodeId + "." + partitionModelKey,
473 sourceNodeName + "." + partitionModelKey,
474 sourceNode->getModel()->getUpVector(),
475 sourceNode->getModel()->getRotationOrder(),
476 nullptr
477 );
478 modelsByPartition[partitionModelKey] = partitionModel;
479 modelsPosition[partitionModelKey].set(partitionX * 64.0f, partitionY * 64.0f, partitionZ * 64.0f);
480 }
481
482 // get node
483 auto partitionModelNode = partitionModelNodes[partitionModelKey];
484 partitionModelNode = partitionModel->getNodeById(sourceNode->getId());
485 if (partitionModelNode == nullptr) {
486 // TODO: create sub nodes if they do not yet exist
487 partitionModelNode = new Node(
488 partitionModel,
489 sourceNode->getParentNode() == nullptr?nullptr:partitionModel->getNodeById(sourceNode->getParentNode()->getId()),
490 sourceNode->getId(),
491 sourceNode->getName()
492 );
493 partitionModelNode->setTransformationsMatrix(sourceNode->getTransformationsMatrix());
494 if (sourceNode->getParentNode() == nullptr) {
495 partitionModel->getSubNodes()[partitionModelNode->getId()] = partitionModelNode;
496 } else {
497 partitionModelNode->getParentNode()->getSubNodes()[partitionModelNode->getId()] = partitionModelNode;
498 }
499 partitionModel->getNodes()[partitionModelNode->getId()] = partitionModelNode;
500 partitionModelNodes[partitionModelKey] = partitionModelNode;
501 }
502
503 // get faces entity
504 FacesEntity* partitionModelNodeFacesEntity = nullptr;
505 for (auto& partitionModelNodeFacesEntityExisting: partitionModelNodesFacesEntities[partitionModelKey]) {
506 if (partitionModelNodeFacesEntityExisting.getId() == facesEntity.getId()) {
507 partitionModelNodeFacesEntity = &partitionModelNodeFacesEntityExisting;
508 }
509 }
510 if (partitionModelNodeFacesEntity == nullptr) {
511 auto newFacesEntity = FacesEntity(
512 partitionModelNode,
513 facesEntity.getId()
514 );
515 partitionModelNodesFacesEntities[partitionModelKey].push_back(newFacesEntity);
516 partitionModelNodeFacesEntity = &newFacesEntity;
517 auto partitionModelNodeFacesEntityMaterial = partitionModel->getMaterials()[facesEntity.getMaterial()->getId()];
518 if (partitionModelNodeFacesEntityMaterial == nullptr) {
519 partitionModelNodeFacesEntityMaterial = cloneMaterial(facesEntity.getMaterial());
520 partitionModel->getMaterials()[facesEntity.getMaterial()->getId()] = partitionModelNodeFacesEntityMaterial;
521 }
522 partitionModelNodeFacesEntity->setMaterial(partitionModelNodeFacesEntityMaterial);
523 }
524
525 auto faces = partitionModelNodeFacesEntity->getFaces();
526
527 // add vertices and such
528 auto verticesIdx = partitionModelNodesVertices[partitionModelKey].size();
529 partitionModelNodesVertices[partitionModelKey].push_back(vertex0);
530 partitionModelNodesVertices[partitionModelKey].push_back(vertex1);
531 partitionModelNodesVertices[partitionModelKey].push_back(vertex2);
532 partitionModelNodesNormals[partitionModelKey].push_back(normal0);
533 partitionModelNodesNormals[partitionModelKey].push_back(normal1);
534 partitionModelNodesNormals[partitionModelKey].push_back(normal2);
535 if (haveTextureCoordinates == true) {
536 partitionModelNodesTextureCoordinates[partitionModelKey].push_back(textureCoordinate0);
537 partitionModelNodesTextureCoordinates[partitionModelKey].push_back(textureCoordinate1);
538 partitionModelNodesTextureCoordinates[partitionModelKey].push_back(textureCoordinate2);
539 }
540 if (haveTangentsBitangents == true) {
541 partitionModelNodesTangents[partitionModelKey].push_back(tangent0);
542 partitionModelNodesTangents[partitionModelKey].push_back(tangent1);
543 partitionModelNodesTangents[partitionModelKey].push_back(tangent2);
544 partitionModelNodesBitangents[partitionModelKey].push_back(bitangent0);
545 partitionModelNodesBitangents[partitionModelKey].push_back(bitangent1);
546 partitionModelNodesBitangents[partitionModelKey].push_back(bitangent2);
547 }
548 Face newFace =
549 Face(
550 partitionModelNode,
551 verticesIdx + 0,
552 verticesIdx + 1,
553 verticesIdx + 2,
554 verticesIdx + 0,
555 verticesIdx + 1,
556 verticesIdx + 2
557 );
558 if (haveTextureCoordinates == true) {
560 verticesIdx + 0,
561 verticesIdx + 1,
562 verticesIdx + 2
563 );
564 }
565 if (haveTangentsBitangents == true) {
566 newFace.setTangentIndices(
567 verticesIdx + 0,
568 verticesIdx + 1,
569 verticesIdx + 2
570 );
571 newFace.setBitangentIndices(
572 verticesIdx + 0,
573 verticesIdx + 1,
574 verticesIdx + 2
575 );
576 }
577 faces.push_back(newFace);
578 partitionModelNodeFacesEntity->setFaces(faces);
579 }
580 }
581
582 // set vertices and such
583 for (auto it: modelsByPartition) {
584 auto partitionModelKey = it.first;
585 if (partitionModelNodes[partitionModelKey] == nullptr) continue;
586 partitionModelNodes[partitionModelKey]->setVertices(partitionModelNodesVertices[partitionModelKey]);
587 partitionModelNodes[partitionModelKey]->setNormals(partitionModelNodesNormals[partitionModelKey]);
588 partitionModelNodes[partitionModelKey]->setTextureCoordinates(partitionModelNodesTextureCoordinates[partitionModelKey]);
589 partitionModelNodes[partitionModelKey]->setTangents(partitionModelNodesTangents[partitionModelKey]);
590 partitionModelNodes[partitionModelKey]->setBitangents(partitionModelNodesBitangents[partitionModelKey]);
591 partitionModelNodes[partitionModelKey]->setFacesEntities(partitionModelNodesFacesEntities[partitionModelKey]);
592 }
593
594 for (auto nodeIt: sourceNode->getSubNodes()) {
595 partitionNode(nodeIt.second, modelsByPartition, modelsPosition, transformationsMatrix);
596 }
597}
598
599void ModelTools::partition(Model* model, const Transformations& transformations, map<string, Model*>& modelsByPartition, map<string, Vector3>& modelsPosition) {
600 Matrix4x4 transformationsMatrix;
601 transformationsMatrix.set(model->getImportTransformationsMatrix());
602 transformationsMatrix.multiply(transformations.getTransformationsMatrix());
603 for (auto nodeIt: model->getSubNodes()) {
604 partitionNode(nodeIt.second, modelsByPartition, modelsPosition, transformationsMatrix);
605 }
606 for (auto modelsByPartitionIt: modelsByPartition) {
607 auto partitionKey = modelsByPartitionIt.first;
608 auto partitionModel = modelsByPartitionIt.second;
609 partitionModel->setImportTransformationsMatrix(model->getImportTransformationsMatrix());
610 ModelTools::createDefaultAnimation(partitionModel, 0);
611 ModelTools::setupJoints(partitionModel);
612 ModelTools::fixAnimationLength(partitionModel);
614 }
615}
616
618 // TODO: a.drewke
619 /*
620 for (auto& facesEntity: node->getFacesEntities()) {
621 facesEntity.getFaces().shrink_to_fit();
622 }
623
624 node->getFacesEntities().shrink_to_fit();
625 node->getVertices().shrink_to_fit();
626 node->getNormals().shrink_to_fit();
627 node->getTextureCoordinates().shrink_to_fit();
628 node->getTangents().shrink_to_fit();
629 node->getBitangents().shrink_to_fit();
630
631 // do child nodes
632 for (auto nodeIt: node->getSubNodes()) {
633 shrinkToFit(nodeIt.second);
634 }
635 */
636}
637
639 for (auto nodeIt: model->getSubNodes()) {
640 shrinkToFit(nodeIt.second);
641 }
642}
643
644float ModelTools::computeNormals(Node* node, ProgressCallback* progressCallback, float incrementPerFace, float progress) {
645 node->setNormals(vector<Vector3>());
646 array<Vector3, 3> vertices;
647 Vector3 normal;
648 auto facesEntityProcessed = 0;
649 vector<Vector3> normals;
650 auto facesEntities = node->getFacesEntities();
651 for (auto& facesEntity: facesEntities) {
652 auto faces = facesEntity.getFaces();
653 for (auto& face: faces) {
654 for (auto i = 0; i < vertices.size(); i++) {
655 vertices[i] = node->getVertices()[face.getVertexIndices()[i]];
656 }
657 normal = computeNormal(vertices);
658 face.setNormalIndices(normals.size(), normals.size() + 1, normals.size() + 2);
659 normals.push_back(normal);
660 normals.push_back(normal);
661 normals.push_back(normal);
662 if (progressCallback != nullptr) {
663 progress+= incrementPerFace / 2.0f;
664 if (facesEntityProcessed == 0 || facesEntityProcessed % 1000 == 0) progressCallback->progress(progress);
665 facesEntityProcessed++;
666 }
667 }
668 facesEntity.setFaces(faces);
669 }
670 node->setFacesEntities(facesEntities);
671 facesEntityProcessed = 0;
672 for (auto& facesEntity: node->getFacesEntities()) {
673 for (auto& face: facesEntity.getFaces()) {
674 for (auto i = 0; i < vertices.size(); i++) {
675 if (interpolateNormal(node, node->getVertices()[face.getVertexIndices()[i]], normals, normal) == true) {
676 normals[face.getNormalIndices()[i]].set(normal);
677 }
678 }
679 if (progressCallback != nullptr) {
680 progress+= incrementPerFace / 2.0f;
681 if (facesEntityProcessed == 0 || facesEntityProcessed % 1000 == 0) progressCallback->progress(progress);
682 facesEntityProcessed++;
683 }
684 }
685 }
686 node->setNormals(normals);
687 for (auto subNodeIt: node->getSubNodes()) {
688 progress = computeNormals(subNodeIt.second, progressCallback, incrementPerFace, progress);
689 }
690 return progress;
691}
692
693void ModelTools::computeNormals(Model* model, ProgressCallback* progressCallback) {
694 auto faceCount = 0;
695 for (auto nodeIt: model->getSubNodes()) {
696 faceCount+= determineFaceCount(nodeIt.second);
697 }
698 for (auto nodeIt: model->getSubNodes()) {
699 computeNormals(nodeIt.second, progressCallback, 1.0f / static_cast<float>(faceCount), 0.0f);
700 }
702 if (progressCallback != nullptr) {
703 progressCallback->progress(1.0f);
704 delete progressCallback;
705 }
706}
707
709 auto faceCount = 0;
710 faceCount+= node->getFaceCount();
711 for (auto subNodeIt: node->getSubNodes()) {
712 faceCount+= determineFaceCount(subNodeIt.second);
713 }
714 return faceCount;
715}
716
717void ModelTools::prepareForShader(Model* model, const string& shader) {
718 if (shader == "foliage" || shader == "pbr-foliage" || shader == "tree" || shader == "pbr-tree") {
719 for (auto nodeIt: model->getSubNodes()) prepareForFoliageTreeShader(nodeIt.second, model->getImportTransformationsMatrix(), shader);
720 model->setImportTransformationsMatrix(Matrix4x4().identity());
721 model->setUpVector(UpVector::Y_UP);
722 } else
723 if (shader == "water") {
724 for (auto nodeIt: model->getSubNodes()) prepareForWaterShader(nodeIt.second, model->getImportTransformationsMatrix());
725 model->setImportTransformationsMatrix(Matrix4x4().identity());
726 model->setUpVector(UpVector::Y_UP);
727 } else {
728 for (auto nodeIt: model->getSubNodes()) prepareForDefaultShader(nodeIt.second);
729 }
730}
731
733 vector<Vector3> objectOrigins;
734 node->setOrigins(objectOrigins);
735 for (auto nodeIt: node->getSubNodes()) {
736 prepareForDefaultShader(nodeIt.second);
737 }
738}
739
740void ModelTools::prepareForFoliageTreeShader(Node* node, const Matrix4x4& parentTransformationsMatrix, const string& shader) {
741 vector<Vector3> objectOrigins;
742 objectOrigins.resize(node->getVertices().size());
743 auto transformationsMatrix = node->getTransformationsMatrix().clone().multiply(parentTransformationsMatrix);
744 {
745 auto vertices = node->getVertices();
746 auto vertexIdx = 0;
747 for (auto& vertex: vertices) {
748 vertex = transformationsMatrix.multiply(vertex);
749 if (shader == "tree" || shader == "pbr-tree") objectOrigins[vertexIdx].set(0.0f, vertex.getY(), 0.0f);
750 vertexIdx++;
751 }
752 node->setVertices(vertices);
753 }
754 {
755 auto normals = node->getNormals();
756 for (auto& normal: normals) {
757 normal = transformationsMatrix.multiplyNoTranslation(normal);
758 normal.normalize();
759 }
760 node->setNormals(normals);
761 }
762 {
763 auto tangents = node->getTangents();
764 for (auto& tangent: tangents) {
765 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
766 tangent.normalize();
767 }
768 node->setTangents(tangents);
769 }
770 {
771 auto bitangents = node->getBitangents();
772 for (auto& bitangent: bitangents) {
773 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
774 bitangent.normalize();
775 }
776 node->setBitangents(bitangents);
777 }
778 node->setTransformationsMatrix(Matrix4x4().identity());
779 node->setOrigins(objectOrigins);
780 for (auto nodeIt: node->getSubNodes()) {
781 prepareForFoliageTreeShader(nodeIt.second, transformationsMatrix, shader);
782 }
783}
784
785void ModelTools::prepareForWaterShader(Node* node, const Matrix4x4& parentTransformationsMatrix) {
786 auto transformationsMatrix = node->getTransformationsMatrix().clone().multiply(parentTransformationsMatrix);
787 {
788 auto vertices = node->getVertices();
789 auto vertexIdx = 0;
790 for (auto& vertex: vertices) {
791 vertex = transformationsMatrix.multiply(vertex);
792 }
793 node->setVertices(vertices);
794 }
795 {
796 auto normals = node->getNormals();
797 for (auto& normal: normals) {
798 normal = transformationsMatrix.multiplyNoTranslation(normal);
799 normal.normalize();
800 }
801 node->setNormals(normals);
802 }
803 {
804 auto tangents = node->getTangents();
805 for (auto& tangent: tangents) {
806 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
807 tangent.normalize();
808 }
809 node->setTangents(tangents);
810 }
811 {
812 auto bitangents = node->getBitangents();
813 for (auto& bitangent: bitangents) {
814 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
815 bitangent.normalize();
816 }
817 node->setBitangents(bitangents);
818 }
819 node->setTransformationsMatrix(Matrix4x4().identity());
820 for (auto nodeIt: node->getSubNodes()) {
821 prepareForWaterShader(nodeIt.second, transformationsMatrix);
822 }
823}
824
825void ModelTools::checkForOptimization(Node* node, map<string, int>& materialUseCount, const vector<string>& excludeDiffuseTextureFileNamePatterns) {
826 // skip on joints as they do not have textures to display and no vertex data
827 if (node->isJoint() == true) return;
828
829 // track material usage
830 for (auto& facesEntity: node->getFacesEntities()) {
831 if (facesEntity.getMaterial() == nullptr) continue;
832 bool excludeDiffuseTexture = false;
833 for (auto& excludeDiffuseTextureFileNamePattern: excludeDiffuseTextureFileNamePatterns) {
834 if (StringTools::startsWith(facesEntity.getMaterial()->getSpecularMaterialProperties()->getDiffuseTextureFileName(), excludeDiffuseTextureFileNamePattern) == true) {
835 excludeDiffuseTexture = true;
836 break;
837 }
838 }
839 if (excludeDiffuseTexture == true) {
840 continue;
841 }
842 materialUseCount[facesEntity.getMaterial()->getId()]++;
843 }
844
845 // do not transform skinning vertices and such
846 if (node->getSkinning() != nullptr) return;
847
848 //
849 for (auto nodeIt: node->getSubNodes()) {
850 checkForOptimization(nodeIt.second, materialUseCount, excludeDiffuseTextureFileNamePatterns);
851 }
852}
853
854void ModelTools::prepareForOptimization(Node* node, const Matrix4x4& parentTransformationsMatrix) {
855 // skip on joints as they do not have textures to display and no vertex data
856 if (node->isJoint() == true) return;
857
858 // do not transform skinning vertices and such
859 if (node->getSkinning() != nullptr) return;
860
861 // static node, apply node transformations matrix
862 auto transformationsMatrix = node->getTransformationsMatrix().clone().multiply(parentTransformationsMatrix);
863 {
864 auto vertices = node->getVertices();
865 for (auto& vertex: vertices) {
866 vertex = transformationsMatrix.multiply(vertex);
867 }
868 node->setVertices(vertices);
869 }
870 {
871 auto normals = node->getNormals();
872 for (auto& normal: normals) {
873 normal = transformationsMatrix.multiplyNoTranslation(normal);
874 normal.normalize();
875 }
876 node->setNormals(normals);
877 }
878 {
879 auto tangents = node->getTangents();
880 for (auto& tangent: tangents) {
881 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
882 tangent.normalize();
883 }
884 node->setTangents(tangents);
885 }
886 {
887 auto bitangents = node->getBitangents();
888 for (auto& bitangent: bitangents) {
889 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
890 bitangent.normalize();
891 }
892 node->setBitangents(bitangents);
893 }
894
895 // we do not set node transformations matrix as we need it later
896 // node->setTransformationsMatrix(Matrix4x4().identity());
897
898 //
899 for (auto nodeIt: node->getSubNodes()) {
900 prepareForOptimization(nodeIt.second, transformationsMatrix);
901 }
902}
903
904void ModelTools::optimizeNode(Node* sourceNode, Model* targetModel, int diffuseTextureAtlasSize, const map<string, int>& diffuseTextureAtlasIndices, const vector<string>& excludeDiffuseTextureFileNamePatterns) {
905 if (sourceNode->getFacesEntities().size() > 0) {
906 unordered_set<int> processedTextureCoordinates;
907 auto& sourceVertices = sourceNode->getVertices();
908 auto& sourceNormals = sourceNode->getNormals();
909 auto& sourceTangents = sourceNode->getTangents();
910 auto& sourceBitangents = sourceNode->getBitangents();
911 auto& sourceTextureCoordinates = sourceNode->getTextureCoordinates();
912 auto& sourceOrigins = sourceNode->getOrigins();
913 auto targetNode = targetModel->getNodes()["tdme.node.optimized"];
914 auto targetVertices = targetNode->getVertices();
915 auto targetNormals = targetNode->getNormals();
916 auto targetTangents = targetNode->getTangents();
917 auto targetBitangents = targetNode->getBitangents();
918 auto targetTextureCoordinates = targetNode->getTextureCoordinates();
919 auto targetOrigins = targetNode->getOrigins();
920 auto targetOffset = targetVertices.size();
921 for (auto& v: sourceVertices) targetVertices.push_back(v);
922 for (auto& v: sourceNormals) targetNormals.push_back(v);
923 for (auto& v: sourceTangents) targetTangents.push_back(v);
924 for (auto& v: sourceBitangents) targetBitangents.push_back(v);
925 for (auto& v: sourceTextureCoordinates) targetTextureCoordinates.push_back(v);
926 for (auto& v: sourceOrigins) targetOrigins.push_back(v);
927 targetNode->setVertices(targetVertices);
928 targetNode->setNormals(targetNormals);
929 targetNode->setTangents(targetTangents);
930 targetNode->setBitangents(targetBitangents);
931 targetNode->setOrigins(targetOrigins);
932 vector<FacesEntity> targetFacesEntitiesKeptMaterials;
933 for (auto& targetFacesEntity: targetNode->getFacesEntities()) {
934 if (StringTools::startsWith(targetFacesEntity.getId(), "tdme.facesentity.kept.") == false) continue;
935 targetFacesEntitiesKeptMaterials.push_back(targetFacesEntity);
936 }
937 FacesEntity* tmpFacesEntity = nullptr;
938 auto targetFaces = (tmpFacesEntity = targetNode->getFacesEntity("tdme.facesentity.optimized")) != nullptr?tmpFacesEntity->getFaces():vector<Face>();
939 auto targetFacesMaskedTransparency = (tmpFacesEntity = targetNode->getFacesEntity("tdme.facesentity.optimized.maskedtransparency")) != nullptr?tmpFacesEntity->getFaces():vector<Face>();
940 auto targetFacesTransparency = (tmpFacesEntity = targetNode->getFacesEntity("tdme.facesentity.optimized.transparency")) != nullptr?tmpFacesEntity->getFaces():vector<Face>();
941 for (auto& sourceFacesEntity: sourceNode->getFacesEntities()) {
942 auto material = sourceFacesEntity.getMaterial();
943
944 //
945 string keptMaterialId;
946 for (auto& excludeDiffuseTextureFileNamePattern: excludeDiffuseTextureFileNamePatterns) {
947 if (StringTools::startsWith(sourceFacesEntity.getMaterial()->getSpecularMaterialProperties()->getDiffuseTextureFileName(), excludeDiffuseTextureFileNamePattern) == true) {
948 keptMaterialId = sourceFacesEntity.getMaterial()->getId();
949 break;
950 }
951 }
952 auto targetFacesKeptMaterial = (tmpFacesEntity = targetNode->getFacesEntity("tdme.facesentity.kept." + keptMaterialId)) != nullptr?tmpFacesEntity->getFaces():vector<Face>();
953
954 auto diffuseTextureAtlasIndexIt = diffuseTextureAtlasIndices.find(material->getId());
955 auto diffuseTextureAtlasIndex = -1;
956 if (diffuseTextureAtlasIndexIt != diffuseTextureAtlasIndices.end()) {
957 diffuseTextureAtlasIndex = diffuseTextureAtlasIndices.find(material->getId())->second;
958 }
959 auto textureXOffset = 0.0f;
960 auto textureYOffset = 0.0f;
961 auto textureXScale = 1.0f;
962 auto textureYScale = 1.0f;
963 if (diffuseTextureAtlasIndex != -1) {
964 textureXOffset = diffuseTextureAtlasSize == 0?0.0f:static_cast<float>(diffuseTextureAtlasIndex % diffuseTextureAtlasSize) * 1000.0f + 500.0f;
965 textureYOffset = diffuseTextureAtlasSize == 0?0.0f:static_cast<float>(diffuseTextureAtlasIndex / diffuseTextureAtlasSize) * 1000.0f + 500.0f;
966 textureXScale = diffuseTextureAtlasSize == 0?1.0f:1.0f;
967 textureYScale = diffuseTextureAtlasSize == 0?1.0f:1.0f;
968 }
969 for (auto& face: sourceFacesEntity.getFaces()) {
970 auto sourceVertexIndices = face.getVertexIndices();
971 auto sourceNormalIndices = face.getNormalIndices();
972 auto sourceTangentIndices = face.getTangentIndices();
973 auto sourceBitangentIndices = face.getBitangentIndices();
974 auto sourceTextureCoordinateIndices = face.getTextureCoordinateIndices();
975 // TODO: could happen that in one node are two faces entities with different textures that reference the same texture coordinate, in this case the following breaks the texture coordinates
976 for (auto i = 0; i < sourceTextureCoordinateIndices.size(); i++) {
977 if (sourceTextureCoordinateIndices[i] != -1 &&
978 sourceTextureCoordinates.size() > 0 &&
979 processedTextureCoordinates.find(sourceTextureCoordinateIndices[i]) == processedTextureCoordinates.end()) {
980 auto textureCoordinateArray = sourceTextureCoordinates[sourceTextureCoordinateIndices[i]].getArray();
981 textureCoordinateArray[0]*= textureXScale;
982 textureCoordinateArray[0]+= textureXOffset;
983 textureCoordinateArray[1]*= textureYScale;
984 textureCoordinateArray[1]+= textureYOffset;
985 targetTextureCoordinates[sourceTextureCoordinateIndices[i] + targetOffset] = TextureCoordinate(textureCoordinateArray);
986 processedTextureCoordinates.insert(sourceTextureCoordinateIndices[i]);
987 }
988 }
989 if (keptMaterialId.empty() == false) {
990 targetFacesKeptMaterial.push_back(
991 Face(
992 targetNode,
993 sourceVertexIndices[0] + targetOffset,
994 sourceVertexIndices[1] + targetOffset,
995 sourceVertexIndices[2] + targetOffset,
996 sourceNormalIndices[0] + targetOffset,
997 sourceNormalIndices[1] + targetOffset,
998 sourceNormalIndices[2] + targetOffset,
999 sourceTextureCoordinateIndices[0] + targetOffset,
1000 sourceTextureCoordinateIndices[1] + targetOffset,
1001 sourceTextureCoordinateIndices[2] + targetOffset
1002 )
1003 );
1004 } else
1005 if (material->getSpecularMaterialProperties()->hasDiffuseTextureTransparency() == true) {
1006 if (material->getSpecularMaterialProperties()->hasDiffuseTextureMaskedTransparency() == true) {
1007 targetFacesMaskedTransparency.push_back(
1008 Face(
1009 targetNode,
1010 sourceVertexIndices[0] + targetOffset,
1011 sourceVertexIndices[1] + targetOffset,
1012 sourceVertexIndices[2] + targetOffset,
1013 sourceNormalIndices[0] + targetOffset,
1014 sourceNormalIndices[1] + targetOffset,
1015 sourceNormalIndices[2] + targetOffset,
1016 sourceTextureCoordinateIndices[0] + targetOffset,
1017 sourceTextureCoordinateIndices[1] + targetOffset,
1018 sourceTextureCoordinateIndices[2] + targetOffset
1019 )
1020 );
1021 } else {
1022 targetFacesTransparency.push_back(
1023 Face(
1024 targetNode,
1025 sourceVertexIndices[0] + targetOffset,
1026 sourceVertexIndices[1] + targetOffset,
1027 sourceVertexIndices[2] + targetOffset,
1028 sourceNormalIndices[0] + targetOffset,
1029 sourceNormalIndices[1] + targetOffset,
1030 sourceNormalIndices[2] + targetOffset,
1031 sourceTextureCoordinateIndices[0] + targetOffset,
1032 sourceTextureCoordinateIndices[1] + targetOffset,
1033 sourceTextureCoordinateIndices[2] + targetOffset
1034 )
1035 );
1036 }
1037 } else {
1038 targetFaces.push_back(
1039 Face(
1040 targetNode,
1041 sourceVertexIndices[0] + targetOffset,
1042 sourceVertexIndices[1] + targetOffset,
1043 sourceVertexIndices[2] + targetOffset,
1044 sourceNormalIndices[0] + targetOffset,
1045 sourceNormalIndices[1] + targetOffset,
1046 sourceNormalIndices[2] + targetOffset,
1047 sourceTextureCoordinateIndices[0] + targetOffset,
1048 sourceTextureCoordinateIndices[1] + targetOffset,
1049 sourceTextureCoordinateIndices[2] + targetOffset
1050 )
1051 );
1052 }
1053 }
1054 if (targetFacesKeptMaterial.size() > 0) {
1055 FacesEntity* facesEntity = nullptr;
1056 for (auto& targetFacesEntityKeptMaterial: targetFacesEntitiesKeptMaterials) {
1057 if (targetFacesEntityKeptMaterial.getId() == "tdme.facesentity.kept." + keptMaterialId) {
1058 facesEntity = &targetFacesEntityKeptMaterial;
1059 break;
1060 }
1061 }
1062 if (facesEntity == nullptr) {
1063 targetFacesEntitiesKeptMaterials.push_back(FacesEntity(targetNode, "tdme.facesentity.kept." + keptMaterialId));
1064 facesEntity = &targetFacesEntitiesKeptMaterials[targetFacesEntitiesKeptMaterials.size() - 1];
1065 }
1066 facesEntity->setFaces(targetFacesKeptMaterial);
1067 facesEntity->setMaterial(targetModel->getMaterials()[keptMaterialId]);
1068 }
1069 }
1070 targetNode->setTextureCoordinates(targetTextureCoordinates);
1071 vector<FacesEntity> targetFacesEntities;
1072 if (targetFaces.size() > 0) {
1073 targetFacesEntities.push_back(FacesEntity(targetNode, "tdme.facesentity.optimized"));
1074 targetFacesEntities[targetFacesEntities.size() - 1].setFaces(targetFaces);
1075 }
1076 if (targetFacesMaskedTransparency.size() > 0) {
1077 targetFacesEntities.push_back(FacesEntity(targetNode, "tdme.facesentity.optimized.maskedtransparency"));
1078 targetFacesEntities[targetFacesEntities.size() - 1].setFaces(targetFacesMaskedTransparency);
1079 }
1080 if (targetFacesTransparency.size() > 0) {
1081 targetFacesEntities.push_back(FacesEntity(targetNode, "tdme.facesentity.optimized.transparency"));
1082 targetFacesEntities[targetFacesEntities.size() - 1].setFaces(targetFacesTransparency);
1083 }
1084 for (auto& targetFacesEntityKeptMaterial: targetFacesEntitiesKeptMaterials) {
1085 targetFacesEntities.push_back(targetFacesEntityKeptMaterial);
1086 }
1087 targetNode->setFacesEntities(targetFacesEntities);
1088 }
1089 for (auto& subNodeIt: sourceNode->getSubNodes()) {
1090 optimizeNode(subNodeIt.second, targetModel, diffuseTextureAtlasSize, diffuseTextureAtlasIndices, excludeDiffuseTextureFileNamePatterns);
1091 }
1092}
1093
1094Texture* ModelTools::createAtlasTexture(const string& id, map<int, Texture*>& textureAtlasTextures) {
1095 auto textureAtlasSize = static_cast<int>(Math::ceil(sqrt(textureAtlasTextures.size())));
1096 #define ATLAS_TEXTURE_SIZE 512
1097 #define ATLAS_TEXTURE_BORDER 32
1098 auto textureWidth = textureAtlasSize * ATLAS_TEXTURE_SIZE;
1099 auto textureHeight = textureAtlasSize * ATLAS_TEXTURE_SIZE;
1100 auto textureByteBuffer = ByteBuffer::allocate(textureWidth * textureHeight * 4);
1101 for (auto y = 0; y < textureHeight; y++)
1102 for (auto x = 0; x < textureWidth; x++) {
1103 auto atlasTextureIdxX = x / ATLAS_TEXTURE_SIZE;
1104 auto atlasTextureIdxY = y / ATLAS_TEXTURE_SIZE;
1105 auto materialTextureX = x - (atlasTextureIdxX * ATLAS_TEXTURE_SIZE);
1106 auto materialTextureY = y - (atlasTextureIdxY * ATLAS_TEXTURE_SIZE);
1107 auto materialTextureXFloat = static_cast<float>(materialTextureX) / static_cast<float>(ATLAS_TEXTURE_SIZE);
1108 auto materialTextureYFloat = static_cast<float>(materialTextureY) / static_cast<float>(ATLAS_TEXTURE_SIZE);
1109 auto atlasTextureIdx = atlasTextureIdxY * textureAtlasSize + atlasTextureIdxX;
1110 auto materialTexture = textureAtlasTextures[atlasTextureIdx];
1111 if (materialTexture != nullptr) {
1112 auto materialTextureWidth = materialTexture->getTextureWidth();
1113 auto materialTextureHeight = materialTexture->getTextureHeight();
1114 auto materialTextureBytesPerPixel = materialTexture->getDepth() / 8;
1115 auto materialTextureXInt = static_cast<int>(materialTextureXFloat * static_cast<float>(materialTextureWidth));
1116 auto materialTextureYInt = static_cast<int>(materialTextureYFloat * static_cast<float>(materialTextureHeight));
1117 if (materialTextureXInt < ATLAS_TEXTURE_BORDER) materialTextureXInt = 0; else
1118 if (materialTextureXInt > materialTextureWidth - ATLAS_TEXTURE_BORDER) materialTextureXInt = materialTextureWidth - 1; else
1119 materialTextureXInt = static_cast<int>((static_cast<float>(materialTextureXInt) - static_cast<float>(ATLAS_TEXTURE_BORDER)) * (static_cast<float>(materialTextureWidth) + static_cast<float>(ATLAS_TEXTURE_BORDER) * 2.0f) / static_cast<float>(materialTextureWidth));
1120 if (materialTextureYInt < ATLAS_TEXTURE_BORDER) materialTextureYInt = 0; else
1121 if (materialTextureYInt > materialTextureHeight - ATLAS_TEXTURE_BORDER) materialTextureYInt = materialTextureHeight - 1; else
1122 materialTextureYInt = static_cast<int>((static_cast<float>(materialTextureYInt) - static_cast<float>(ATLAS_TEXTURE_BORDER)) * (static_cast<float>(materialTextureHeight) + static_cast<float>(ATLAS_TEXTURE_BORDER) * 2.0f) / static_cast<float>(materialTextureHeight));
1123 auto materialTexturePixelOffset =
1124 materialTextureYInt * materialTextureWidth * materialTextureBytesPerPixel +
1125 materialTextureXInt * materialTextureBytesPerPixel;
1126 auto materialPixelR = materialTexture->getTextureData()->get(materialTexturePixelOffset + 0);
1127 auto materialPixelG = materialTexture->getTextureData()->get(materialTexturePixelOffset + 1);
1128 auto materialPixelB = materialTexture->getTextureData()->get(materialTexturePixelOffset + 2);
1129 auto materialPixelA = materialTextureBytesPerPixel == 4?materialTexture->getTextureData()->get(materialTexturePixelOffset + 3):0xff;
1130 textureByteBuffer->put(materialPixelR);
1131 textureByteBuffer->put(materialPixelG);
1132 textureByteBuffer->put(materialPixelB);
1133 textureByteBuffer->put(materialPixelA);
1134 } else {
1135 auto materialPixelR = 0xff;
1136 auto materialPixelG = 0x00;
1137 auto materialPixelB = 0x00;
1138 auto materialPixelA = 0xff;
1139 textureByteBuffer->put(materialPixelR);
1140 textureByteBuffer->put(materialPixelG);
1141 textureByteBuffer->put(materialPixelB);
1142 textureByteBuffer->put(materialPixelA);
1143 }
1144 }
1145 auto atlasTexture = new Texture(
1146 id,
1147 32,
1148 textureWidth, textureHeight,
1149 textureWidth, textureHeight,
1150 textureByteBuffer
1151 );
1152 atlasTexture->setAtlasSize(textureAtlasSize);
1153 return atlasTexture;
1154}
1155
1157 return model->getNodes()["tdme.node.optimized"] != nullptr;
1158}
1159
1160Model* ModelTools::optimizeModel(Model* model, const string& texturePathName, const vector<string>& excludeDiffuseTextureFileNamePatterns) {
1161 // exit early if model has been optimized already
1162 if (isOptimizedModel(model) == true) return model;
1163
1164 // TODO: 2 mats could have the same texture
1165 // prepare for optimizations
1166 map<string, int> materialUseCount;
1167 for (auto& nodeIt: model->getSubNodes()) {
1169 nodeIt.second,
1170 materialUseCount,
1171 excludeDiffuseTextureFileNamePatterns
1172 );
1173 }
1174
1175 // check materials and diffuse textures
1176 auto diffuseTextureCount = 0;
1177 map<string, int> diffuseTextureAtlasIndices;
1178 map<int, Texture*> diffuseTextureAtlasTextures;
1179 map<string, Material*> atlasMaterials;
1180 for (auto& materialUseCountIt: materialUseCount) {
1181 auto material = model->getMaterials().find(materialUseCountIt.first)->second;
1182 auto diffuseTexture = material->getSpecularMaterialProperties()->getDiffuseTexture();
1183 if (diffuseTexture != nullptr) {
1184 diffuseTextureAtlasIndices[material->getId()] = diffuseTextureCount;
1185 diffuseTextureAtlasTextures[diffuseTextureCount] = diffuseTexture;
1186 diffuseTextureCount++;
1187 atlasMaterials[material->getId()] = material;
1188 }
1189 }
1190
1191 // do we need to optimize?
1192 if (diffuseTextureCount < 2) return model;
1193
1194 // prepare for optimizations
1195 for (auto& nodeIt: model->getSubNodes()) {
1197 nodeIt.second,
1198 Matrix4x4().identity()
1199 );
1200 }
1201
1202 // create diffuse atlas texture
1203 auto diffuseAtlasTexture = createAtlasTexture(model->getName() + ".diffuse.atlas", diffuseTextureAtlasTextures);
1204
1205 // create model with optimizations applied
1206 auto optimizedModel = new Model(model->getId() + ".optimized", model->getName() + ".optimized", model->getUpVector(), model->getRotationOrder(), new BoundingBox(model->getBoundingBox()), model->getAuthoringTool());
1207 optimizedModel->setImportTransformationsMatrix(model->getImportTransformationsMatrix());
1208 auto optimizedNode = new Node(optimizedModel, nullptr, "tdme.node.optimized", "tdme.node.optimized");
1209 optimizedModel->getNodes()["tdme.node.optimized"] = optimizedNode;
1210 optimizedModel->getSubNodes()["tdme.node.optimized"] = optimizedNode;
1211
1212 // clone materials with diffuse textures that we like to keep
1213 for (auto& materialIt: model->getMaterials()) {
1214 auto material = materialIt.second;
1215 bool keepDiffuseTexture = false;
1216 for (auto& excludeDiffuseTextureFileNamePattern: excludeDiffuseTextureFileNamePatterns) {
1217 if (StringTools::startsWith(material->getSpecularMaterialProperties()->getDiffuseTextureFileName(), excludeDiffuseTextureFileNamePattern) == true) {
1218 keepDiffuseTexture = true;
1219 break;
1220 }
1221 }
1222 if (keepDiffuseTexture == false) continue;
1223 optimizedModel->getMaterials()[material->getId()] = cloneMaterial(material);
1224 }
1225
1226 // create optimized material
1227 auto optimizedMaterial = new Material("tdme.material.optimized");
1228 {
1229 optimizedMaterial->getSpecularMaterialProperties()->setEmbedTextures(true);
1230 optimizedMaterial->getSpecularMaterialProperties()->setDiffuseTexture(diffuseAtlasTexture);
1231 optimizedMaterial->getSpecularMaterialProperties()->setTextureAtlasSize(diffuseAtlasTexture->getAtlasSize());
1232 optimizedMaterial->getSpecularMaterialProperties()->setDiffuseTextureTransparency(false);
1233 optimizedMaterial->getSpecularMaterialProperties()->setDiffuseTextureMaskedTransparency(false);
1234 Vector4 optimizedMaterialEmission(0.0f, 0.0f, 0.0f, 0.0f);
1235 Vector4 optimizedMaterialAmbient(0.0f, 0.0f, 0.0f, 0.0f);
1236 Vector4 optimizedMaterialDiffuse(0.0f, 0.0f, 0.0f, 0.0f);
1237 Vector4 optimizedMaterialSpecular(0.0f, 0.0f, 0.0f, 0.0f);
1238 float optimizedMaterialShininess = 0.0f;
1239 for (auto& atlasMaterialsIt: atlasMaterials) {
1240 auto material = atlasMaterialsIt.second;
1241 optimizedMaterialEmission+= Vector4(material->getSpecularMaterialProperties()->getEmissionColor().getArray());
1242 optimizedMaterialAmbient+= Vector4(material->getSpecularMaterialProperties()->getAmbientColor().getArray());
1243 optimizedMaterialDiffuse+= Vector4(material->getSpecularMaterialProperties()->getDiffuseColor().getArray());
1244 optimizedMaterialSpecular+= Vector4(material->getSpecularMaterialProperties()->getSpecularColor().getArray());
1245 optimizedMaterialShininess+= material->getSpecularMaterialProperties()->getShininess();
1246 }
1247 optimizedMaterialEmission/= static_cast<float>(atlasMaterials.size());
1248 optimizedMaterialAmbient/= static_cast<float>(atlasMaterials.size());
1249 optimizedMaterialDiffuse/= static_cast<float>(atlasMaterials.size());
1250 optimizedMaterialSpecular/= static_cast<float>(atlasMaterials.size());
1251 optimizedMaterialShininess/= static_cast<float>(atlasMaterials.size());
1252 optimizedMaterial->getSpecularMaterialProperties()->setEmissionColor(Color4(optimizedMaterialEmission.getArray()));
1253 optimizedMaterial->getSpecularMaterialProperties()->setAmbientColor(Color4(optimizedMaterialAmbient.getArray()));
1254 optimizedMaterial->getSpecularMaterialProperties()->setDiffuseColor(Color4(optimizedMaterialDiffuse.getArray()));
1255 optimizedMaterial->getSpecularMaterialProperties()->setSpecularColor(Color4(optimizedMaterialSpecular.getArray()));
1256 optimizedMaterial->getSpecularMaterialProperties()->setShininess(optimizedMaterialShininess);
1257 }
1258
1259 // also have a material with masked transparency
1260 auto optimizedMaterialMaskedTransparency = cloneMaterial(optimizedMaterial, "tdme.material.optimized.maskedtransparency");
1261 optimizedMaterialMaskedTransparency->getSpecularMaterialProperties()->setDiffuseTextureTransparency(true);
1262 optimizedMaterialMaskedTransparency->getSpecularMaterialProperties()->setDiffuseTextureMaskedTransparency(true);
1263
1264 // also have a material with transparency
1265 auto optimizedMaterialTransparency = cloneMaterial(optimizedMaterial, "tdme.material.optimized.transparency");
1266 optimizedMaterialTransparency->getSpecularMaterialProperties()->setDiffuseTextureTransparency(true);
1267
1268 // now optimize into our optimized model
1269 for (auto& subNodeIt: model->getSubNodes()) {
1270 auto node = subNodeIt.second;
1271 if ((model->hasSkinning() == true && node->getSkinning() != nullptr) ||
1272 (model->hasSkinning() == false && node->isJoint() == false)) {
1273 optimizeNode(node, optimizedModel, diffuseAtlasTexture->getAtlasSize(), diffuseTextureAtlasIndices, excludeDiffuseTextureFileNamePatterns);
1274 if (model->hasSkinning() == true) {
1275 auto skinning = node->getSkinning();
1276 auto optimizedSkinning = new Skinning();
1277 optimizedSkinning->setWeights(skinning->getWeights());
1278 optimizedSkinning->setJoints(skinning->getJoints());
1279 optimizedSkinning->setVerticesJointsWeights(skinning->getVerticesJointsWeights());
1280 optimizedModel->getNodes()["tdme.node.optimized"]->setSkinning(optimizedSkinning);
1281 }
1282 }
1283 cloneNode(node, optimizedModel, nullptr, false);
1284 }
1285
1286 // set up materials
1287 {
1288 auto optimizedFacesEntity = optimizedModel->getNodes()["tdme.node.optimized"]->getFacesEntity("tdme.facesentity.optimized");
1289 if (optimizedFacesEntity != nullptr) {
1290 optimizedModel->getMaterials()[optimizedMaterial->getId()] = optimizedMaterial;
1291 optimizedFacesEntity->setMaterial(optimizedMaterial);
1292 } else {
1293 delete optimizedMaterial;
1294 }
1295 }
1296 {
1297 auto optimizedFacesEntityMaskedTransparency = optimizedModel->getNodes()["tdme.node.optimized"]->getFacesEntity("tdme.facesentity.optimized.maskedtransparency");
1298 if (optimizedFacesEntityMaskedTransparency != nullptr) {
1299 optimizedModel->getMaterials()[optimizedMaterialMaskedTransparency->getId()] = optimizedMaterialMaskedTransparency;
1300 optimizedFacesEntityMaskedTransparency->setMaterial(optimizedMaterialMaskedTransparency);
1301 } else {
1302 delete optimizedMaterialMaskedTransparency;
1303 }
1304 }
1305 {
1306 auto optimizedFacesEntityTransparency = optimizedModel->getNodes()["tdme.node.optimized"]->getFacesEntity("tdme.facesentity.optimized.transparency");
1307 if (optimizedFacesEntityTransparency != nullptr) {
1308 optimizedModel->getMaterials()[optimizedMaterialTransparency->getId()] = optimizedMaterialTransparency;
1309 optimizedFacesEntityTransparency->setMaterial(optimizedMaterialTransparency);
1310 } else {
1311 delete optimizedMaterialTransparency;
1312 }
1313 }
1314
1315 // copy animation set up
1316 for (auto animationSetupIt: model->getAnimationSetups()) {
1317 auto animationSetup = animationSetupIt.second;
1318 if (animationSetup->getOverlayFromNodeId().empty() == false) {
1319 optimizedModel->addOverlayAnimationSetup(
1320 animationSetup->getId(),
1321 animationSetup->getOverlayFromNodeId(),
1322 animationSetup->getStartFrame(),
1323 animationSetup->getEndFrame(),
1324 animationSetup->isLoop(),
1325 animationSetup->getSpeed()
1326 );
1327 } else {
1328 optimizedModel->addAnimationSetup(
1329 animationSetup->getId(),
1330 animationSetup->getStartFrame(),
1331 animationSetup->getEndFrame(),
1332 animationSetup->isLoop(),
1333 animationSetup->getSpeed()
1334 );
1335 }
1336 }
1337
1338 //
1339 delete model;
1340
1341 // done
1342 return optimizedModel;
1343}
1344
1346{
1347 // see: https://www.opengl-tutorial.org/intermediate-tutorials/tutorial-13-normal-mapping/
1348 // what we need
1349 vector<Vector3> tangents;
1350 vector<Vector3> bitangents;
1351 // temporary variables
1352 Vector2 uv0;
1353 Vector2 uv1;
1354 Vector2 uv2;
1355 Vector3 deltaPos1;
1356 Vector3 deltaPos2;
1357 Vector2 deltaUV1;
1358 Vector2 deltaUV2;
1359 // create it
1360 auto facesEntities = node->getFacesEntities();
1361 auto& vertices = node->getVertices();
1362 auto& normals = node->getNormals();
1363 auto textureCoordinates = node->getTextureCoordinates();
1364 for (auto& faceEntity: facesEntities) {
1365 auto faces = faceEntity.getFaces();
1366 for (auto& face: faces) {
1367 // Shortcuts for vertices
1368 auto verticesIndexes = face.getVertexIndices();
1369 auto v0 = vertices[verticesIndexes[0]];
1370 auto v1 = vertices[verticesIndexes[1]];
1371 auto v2 = vertices[verticesIndexes[2]];
1372 // shortcuts for UVs
1373 auto textureCoordinatesIndexes = face.getTextureCoordinateIndices();
1374 uv0.set(textureCoordinates[textureCoordinatesIndexes[0]].getArray());
1375 uv0.setY(1.0f - uv0.getY());
1376 uv1.set(textureCoordinates[textureCoordinatesIndexes[1]].getArray());
1377 uv1.setY(1.0f - uv1.getY());
1378 uv2.set(textureCoordinates[textureCoordinatesIndexes[2]].getArray());
1379 uv2.setY(1.0f - uv2.getY());
1380 // edges of the triangle : position delta
1381 deltaPos1.set(v1).sub(v0);
1382 deltaPos2.set(v2).sub(v0);
1383 // UV delta
1384 deltaUV1.set(uv1).sub(uv0);
1385 deltaUV2.set(uv2).sub(uv0);
1386 // compute tangent and bitangent
1387 auto r = 1.0f / (deltaUV1.getX() * deltaUV2.getY() - deltaUV1.getY() * deltaUV2.getX());
1388 auto tangent = deltaPos1.clone().scale(deltaUV2.getY()).sub(deltaPos2.clone().scale(deltaUV1.getY())).scale(r);
1389 auto bitangent = deltaPos2.clone().scale(deltaUV1.getX()).sub(deltaPos1.clone().scale(deltaUV2.getX())).scale(r);
1390 // set up tangent face indices
1391 face.setTangentIndices(tangents.size() + 0, tangents.size() + 1, tangents.size() + 2);
1392 // set up bitangent face indices
1393 face.setBitangentIndices(bitangents.size() + 0, bitangents.size() + 1, bitangents.size() + 2);
1394 // add to group tangents, bitangents
1395 tangents.push_back(tangent);
1396 tangents.push_back(tangent);
1397 tangents.push_back(tangent);
1398 bitangents.push_back(bitangent);
1399 bitangents.push_back(bitangent);
1400 bitangents.push_back(bitangent);
1401 }
1402 faceEntity.setFaces(faces);
1403 }
1404 node->setFacesEntities(facesEntities);
1405
1406 // set up tangents and bitangents if we have any
1407 if (tangents.size() > 0 && bitangents.size() > 0) {
1408 // going further
1409 auto facesEntities = node->getFacesEntities();
1410 for (auto& faceEntity: facesEntities) {
1411 auto faces = faceEntity.getFaces();
1412 for (auto& face: faces) {
1413 for (auto i = 0; i < 3; i++) {
1414 auto normal = normals[face.getNormalIndices()[i]];
1415 auto& tangent = tangents[face.getTangentIndices()[i]];
1416 auto& bitangent = bitangents[face.getBitangentIndices()[i]];
1417 tangent.sub(normal.clone().scale(Vector3::computeDotProduct(normal, tangent))).normalize();
1418 if (Vector3::computeDotProduct(Vector3::computeCrossProduct(normal, tangent), bitangent) < 0.0f) {
1419 tangent.scale(-1.0f);
1420 }
1421 bitangent.normalize();
1422 }
1423 }
1424 faceEntity.setFaces(faces);
1425 }
1426 node->setFacesEntities(facesEntities);
1427 //
1428 node->setTangents(tangents);
1429 node->setBitangents(bitangents);
1430 }
1431}
#define ATLAS_TEXTURE_BORDER
#define ATLAS_TEXTURE_SIZE
Transformations which contain scale, rotations and translation.
const Matrix4x4 & getTransformationsMatrix() const
const vector< Matrix4x4 > & getTransformationsMatrices() const
Returns transformation matrices.
Definition: Animation.h:41
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
void setTextureCoordinateIndices(int32_t vt0, int32_t vt1, int32_t vt2)
Set up optional texture coordinate indices.
Definition: Face.cpp:45
void setTangentIndices(int32_t ti0, int32_t ti1, int32_t ti2)
Set tangent indices.
Definition: Face.cpp:52
void setBitangentIndices(int32_t bi0, int32_t bi1, int32_t bi2)
Set bitangent indices.
Definition: Face.cpp:59
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
void setMaterial(Material *material)
Set up the entity's material.
Definition: FacesEntity.h:72
const vector< Face > & getFaces() const
Definition: FacesEntity.h:86
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
const string & getId() const
Definition: Material.h:57
const SpecularMaterialProperties * getSpecularMaterialProperties() const
Definition: Material.h:64
Representation of a 3d model.
Definition: Model.h:32
map< string, AnimationSetup * > & getAnimationSetups()
TODO: return const map.
Definition: Model.h:274
UpVector * getUpVector()
Definition: Model.h:133
RotationOrder * getRotationOrder()
Definition: Model.h:148
const Matrix4x4 & getImportTransformationsMatrix()
Definition: Model.h:293
AuthoringTool getAuthoringTool()
Definition: Model.h:105
void setUpVector(UpVector *upVector)
Set up vector.
Definition: Model.h:141
map< string, Node * > & getNodes()
Returns all object's nodes.
Definition: Model.h:179
AnimationSetup * addAnimationSetup(const string &id, int32_t startFrame, int32_t endFrame, bool loop, float speed=1.0f)
Adds an base animation setup.
Definition: Model.cpp:97
const string & getId()
Definition: Model.h:119
AnimationSetup * getAnimationSetup(const string &id)
Definition: Model.cpp:116
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
Definition: Model.h:301
const string & getName()
Definition: Model.h:126
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
Definition: Model.h:194
BoundingBox * getBoundingBox()
Definition: Model.cpp:142
map< string, Material * > & getMaterials()
Returns all object materials.
Definition: Model.h:171
Model node.
Definition: Node.h:31
int32_t getFaceCount() const
Definition: Node.cpp:109
Model * getModel()
Definition: Node.h:70
const vector< Vector3 > & getTangents() const
Definition: Node.h:199
const vector< Vector3 > & getOrigins() const
Definition: Node.h:276
const Matrix4x4 & getTransformationsMatrix() const
Definition: Node.h:121
Node * getParentNode()
Definition: Node.h:77
void setAnimation(Animation *animation)
Sets animation object.
Definition: Node.cpp:97
void setTextureCoordinates(const vector< TextureCoordinate > &textureCoordinates)
Set texture coordinates.
Definition: Node.cpp:70
void setOrigins(const vector< Vector3 > &origins)
Set origins.
Definition: Node.cpp:134
void setVertices(const vector< Vector3 > &vertices)
Set vertices.
Definition: Node.cpp:50
const vector< Vector3 > & getBitangents() const
Definition: Node.h:212
Skinning * getSkinning()
Definition: Node.h:238
void setFacesEntities(const vector< FacesEntity > &facesEntities)
Set up faces entities.
Definition: Node.cpp:125
bool isJoint() const
Definition: Node.h:106
const vector< TextureCoordinate > & getTextureCoordinates() const
Definition: Node.h:186
void setJoint(bool isJoint)
Sets up if this node is a joint or not.
Definition: Node.h:114
const string & getId()
Returns id.
Definition: Node.h:85
const vector< Vector3 > & getNormals() const
Definition: Node.h:173
void setTransformationsMatrix(const Matrix4x4 &transformationsMatrix)
Definition: Node.h:128
const string & getName()
Definition: Node.h:92
Animation * getAnimation()
Definition: Node.h:225
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
map< string, Node * > & getSubNodes()
Definition: Node.h:289
void setBitangents(const vector< Vector3 > &bitangents)
Set bitangents.
Definition: Node.cpp:88
void setTangents(const vector< Vector3 > &tangents)
Set tangents.
Definition: Node.cpp:79
Skinning definition for nodes.
Definition: Skinning.h:27
void setVerticesJointsWeights(const vector< vector< JointWeight > > &verticesJointsWeights)
Sets up vertices joints weights.
Definition: Skinning.cpp:42
const vector< vector< JointWeight > > & getVerticesJointsWeights()
Definition: Skinning.h:68
Represents specular material properties.
Class representing texture UV coordinates data.
TextureCoordinate & set(const TextureCoordinate &textureCoordinate)
Set texture coordinate.
Model up vector.
Definition: UpVector.h:20
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
Scene entity definition.
Definition: SceneEntity.h:24
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Definition: Matrix4x4.h:95
Matrix4x4 clone()
Clones this matrix.
Definition: Matrix4x4.h:624
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
2D vector 2 class
Definition: Vector2.h:19
float getY() const
Definition: Vector2.h:111
Vector2 & set(float x, float y)
Set up vector.
Definition: Vector2.h:65
float getX() const
Definition: Vector2.h:94
Vector2 & sub(const Vector2 &v)
Subtracts a vector.
Definition: Vector2.h:140
Vector2 & setY(float y)
set Y
Definition: Vector2.h:119
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Definition: Vector3.h:325
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
3D vector 4 class
Definition: Vector4.h:19
array< float, 4 > & getArray() const
Definition: Vector4.h:407
Byte buffer class.
Definition: ByteBuffer.h:24
static ByteBuffer * allocate(int32_t capacity)
Allocate byte buffer.
Definition: ByteBuffer.h:29
Console class.
Definition: Console.h:26
static void println()
Print new line to console.
Definition: Console.cpp:78
Model tools functions class.
Definition: ModelTools.h:38
static bool interpolateNormal(Node *node, const Vector3 &vertex, const vector< Vector3 > &normals, Vector3 &normal)
Find all faces that include vertex and compute the avarage normal.
Definition: ModelTools.h:204
static void shrinkToFit(Node *node)
Shrink to fit node.
Definition: ModelTools.cpp:617
static void createTangentsAndBitangents(Node *node)
Create tangents and bitangents for given group.
static void optimizeNode(Node *sourceNode, Model *targetModel, int diffuseTextureAtlasSize, const map< string, int > &diffuseTextureAtlasIndices, const vector< string > &excludeDiffuseTextureFileNamePatterns)
Prepare for optimization.
Definition: ModelTools.cpp:904
static void prepareForOptimization(Node *node, const Matrix4x4 &parentTransformationsMatrix)
Prepare for optimization.
Definition: ModelTools.cpp:854
static void createDefaultAnimation(Model *model, int32_t frames)
Create default animation.
Definition: ModelTools.cpp:262
static void prepareForDefaultShader(Node *node)
Prepare node for default shader.
Definition: ModelTools.cpp:732
static bool hasDefaultAnimation(Model *model)
Check default animation.
Definition: ModelTools.cpp:258
static void prepareForIndexedRendering(Model *model)
Prepare for indexed rendering.
Definition: ModelTools.cpp:84
static void checkForOptimization(Node *node, map< string, int > &materialUseCount, const vector< string > &excludeDiffuseTextureFileNamePatterns)
Check for optimization.
Definition: ModelTools.cpp:825
static void setupJoints(Model *model)
Set up joints for skinning nodes.
Definition: ModelTools.cpp:193
static void prepareForShader(Model *model, const string &shader)
Prepare model for foliage shader.
Definition: ModelTools.cpp:717
static Model * optimizeModel(Model *model, const string &texturePathName=string(), const vector< string > &excludeDiffuseTextureFileNamePatterns=vector< string >())
Optimizes model in terms of material / node reduction.
static int determineFaceCount(Node *node)
Compute face count.
Definition: ModelTools.cpp:708
static void prepareForFoliageTreeShader(Node *node, const Matrix4x4 &parentTransformationsMatrix, const string &shader)
Prepare node for foliage shader.
Definition: ModelTools.cpp:740
static Material * cloneMaterial(const Material *material, const string &id=string())
Clone material.
Definition: ModelTools.cpp:280
static array< Vector3, 3 > computeNormals(const array< Vector3, 3 > &vertices)
Computes face normals for given face vertices these normals will not be smooth.
Definition: ModelTools.h:78
static bool isOptimizedModel(Model *model)
static void fixAnimationLength(Model *model)
Fix animation length.
Definition: ModelTools.cpp:222
static Texture * createAtlasTexture(const string &id, map< int, Texture * > &textureAtlasTextures)
Create atlas texture.
static void prepareForWaterShader(Node *node, const Matrix4x4 &parentTransformationsMatrix)
Prepare node for water shader.
Definition: ModelTools.cpp:785
static void setJoint(Node *root)
Sets up a node as joint taking all subnodes into account.
Definition: ModelTools.cpp:213
static void partitionNode(Node *sourceNode, map< string, Model * > &modelsByPartition, map< string, Vector3 > &modelsPosition, const Matrix4x4 &parentTransformationsMatrix)
Partition sub nodes.
Definition: ModelTools.cpp:367
static void cloneNode(Node *sourceNode, Model *targetModel, Node *targetParentNode=nullptr, bool cloneMesh=true)
Create model from source sub nodes into target sub nodes.
Definition: ModelTools.cpp:321
static void partition(Model *model, const Transformations &transformations, map< string, Model * > &modelsByPartition, map< string, Vector3 > &modelsPosition)
Partition model.
Definition: ModelTools.cpp:599
static Vector3 computeNormal(const array< Vector3, 3 > &vertices)
Computes face normal for given face vertices.
Definition: ModelTools.h:55
String tools class.
Definition: StringTools.h:20
static int32_t lastIndexOf(const string &src, char what)
Finds last index of given character.
Definition: StringTools.h:113
static const bool startsWith(const string &src, const string &prefix)
Checks if string starts with prefix.
Definition: StringTools.h:29
static const string substring(const string &src, int32_t beginIndex)
Returns substring of given string from begin index.
Definition: StringTools.h:133
virtual void progress(float value)=0
Perform action.