6#include <unordered_set>
39using std::unordered_set;
71 for (
auto i = 0; i < vertices.size(); i++) {
72 auto& currentVertexXYZ = vertices[i].getArray();
73 auto& nextVertexXYZ = vertices[(i + 1) % vertices.size()].getArray();
75 (nextVertexXYZ[0] - currentVertexXYZ[0]) * (nextVertexXYZ[1] - currentVertexXYZ[1]) * (nextVertexXYZ[2] - currentVertexXYZ[2]);
92 for (
auto it: nodes) {
93 Node* node = it.second;
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;
108 auto preparedIndices = 0;
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))) {
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);
151 indexedFaceVertexIndices[idx] = newIndex;
153 face.setIndexedRenderingIndices(indexedFaceVertexIndices);
155 newFacesEntity.setFaces(newFaces);
160 if (skinning !=
nullptr) {
165 if (nodeTextureCoordinates.size() > 0) {
170 if (nodeOrigins.size() > 0) {
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];
198 Node* node = it.second;
201 if (skinning !=
nullptr) {
203 for (
auto& joint : skinning->getJoints()) {
204 auto jointNodeIt = nodes.find(joint.getNodeId());
205 if (jointNodeIt != nodes.end()) {
217 Node* node = it.second;
226 if (defaultAnimation !=
nullptr) {
228 Node* node = it.second;
237 if (animation !=
nullptr) {
238 vector<Matrix4x4> newTransformationsMatrices;
241 newTransformationsMatrices.resize(frames);
242 for (
auto i = 0; i < frames; i++) {
243 if (i < oldTransformationsMatrices.size()) {
244 newTransformationsMatrices[i] = oldTransformationsMatrices[i];
246 newTransformationsMatrices[i].identity();
249 animation->setTransformationsMatrices(newTransformationsMatrices);
253 Node* node = it.second;
266 if (defaultAnimation ==
nullptr) {
270 if (defaultAnimation->getStartFrame() != 0 || defaultAnimation->getEndFrame() != frames - 1) {
273 if (frames - 1 > defaultAnimation->getEndFrame()) {
274 Console::println(
string(
"Warning: default animation mismatch, will be fixed"));
281 auto clonedMaterial =
new Material(
id.empty()?material->
getId():
id);
283 if (specularMaterialProperties !=
nullptr) {
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());
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());
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());
316 clonedMaterial->setSpecularMaterialProperties(clonedSpecularMaterialProperties);
318 return clonedMaterial;
322 auto clonedNode =
new Node(targetModel, targetParentNode, sourceNode->
getId(), sourceNode->
getName());
324 clonedNode->setJoint(sourceNode->
isJoint());
325 if (cloneMesh ==
true) {
326 clonedNode->setVertices(sourceNode->
getVertices());
327 clonedNode->setNormals(sourceNode->
getNormals());
329 clonedNode->setTangents(sourceNode->
getTangents());
331 clonedNode->setOrigins(sourceNode->
getOrigins());
332 auto facesEntities = clonedNode->getFacesEntities();
334 if (facesEntity.getMaterial() ==
nullptr)
continue;
336 auto materialIt = targetModel->
getMaterials().find(facesEntity.getMaterial()->getId());
341 material = materialIt->second;
343 auto clonedFacesEntity =
FacesEntity(clonedNode, facesEntity.getId());
344 clonedFacesEntity.setMaterial(material);
345 clonedFacesEntity.setFaces(facesEntity.getFaces());
346 facesEntities.push_back(clonedFacesEntity);
348 clonedNode->setFacesEntities(facesEntities);
353 clonedNode->setAnimation(clonedAnimation);
355 targetModel->
getNodes()[clonedNode->getId()] = clonedNode;
356 if (targetParentNode ==
nullptr) {
357 targetModel->
getSubNodes()[clonedNode->getId()] = clonedNode;
359 targetParentNode->
getSubNodes()[clonedNode->getId()] = clonedNode;
361 for (
auto sourceSubNodeIt: sourceNode->
getSubNodes()) {
362 auto subNode = sourceSubNodeIt.second;
363 cloneNode(subNode, targetModel, clonedNode, cloneMesh);
373 transformationsMatrix.
multiply(parentTransformationsMatrix);
407 map<string, Node*> partitionModelNodes;
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;
418 bool haveTextureCoordinates = facesEntity.isTextureCoordinatesAvailable();
419 bool haveTangentsBitangents = facesEntity.isTangentBitangentAvailable();
420 for (
auto& face: facesEntity.getFaces()) {
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();
433 if (haveTextureCoordinates ==
true) {
438 if (haveTangentsBitangents ==
true) {
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);
463 string partitionModelKey =
464 to_string(partitionX) +
"," +
465 to_string(partitionY) +
"," +
466 to_string(partitionZ);
469 auto partitionModel = modelsByPartition[partitionModelKey];
470 if (partitionModel ==
nullptr) {
471 partitionModel =
new Model(
472 sourceNodeId +
"." + partitionModelKey,
473 sourceNodeName +
"." + partitionModelKey,
478 modelsByPartition[partitionModelKey] = partitionModel;
479 modelsPosition[partitionModelKey].set(partitionX * 64.0f, partitionY * 64.0f, partitionZ * 64.0f);
483 auto partitionModelNode = partitionModelNodes[partitionModelKey];
484 partitionModelNode = partitionModel->getNodeById(sourceNode->
getId());
485 if (partitionModelNode ==
nullptr) {
487 partitionModelNode =
new Node(
495 partitionModel->getSubNodes()[partitionModelNode->getId()] = partitionModelNode;
497 partitionModelNode->getParentNode()->getSubNodes()[partitionModelNode->getId()] = partitionModelNode;
499 partitionModel->getNodes()[partitionModelNode->getId()] = partitionModelNode;
500 partitionModelNodes[partitionModelKey] = partitionModelNode;
504 FacesEntity* partitionModelNodeFacesEntity =
nullptr;
505 for (
auto& partitionModelNodeFacesEntityExisting: partitionModelNodesFacesEntities[partitionModelKey]) {
506 if (partitionModelNodeFacesEntityExisting.getId() == facesEntity.getId()) {
507 partitionModelNodeFacesEntity = &partitionModelNodeFacesEntityExisting;
510 if (partitionModelNodeFacesEntity ==
nullptr) {
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;
522 partitionModelNodeFacesEntity->
setMaterial(partitionModelNodeFacesEntityMaterial);
525 auto faces = partitionModelNodeFacesEntity->
getFaces();
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);
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);
558 if (haveTextureCoordinates ==
true) {
565 if (haveTangentsBitangents ==
true) {
577 faces.push_back(newFace);
578 partitionModelNodeFacesEntity->
setFaces(faces);
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]);
595 partitionNode(nodeIt.second, modelsByPartition, modelsPosition, transformationsMatrix);
604 partitionNode(nodeIt.second, modelsByPartition, modelsPosition, transformationsMatrix);
606 for (
auto modelsByPartitionIt: modelsByPartition) {
607 auto partitionKey = modelsByPartitionIt.first;
608 auto partitionModel = modelsByPartitionIt.second;
646 array<Vector3, 3> vertices;
648 auto facesEntityProcessed = 0;
649 vector<Vector3> normals;
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]];
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++;
668 facesEntity.setFaces(faces);
671 facesEntityProcessed = 0;
673 for (
auto& face: facesEntity.getFaces()) {
674 for (
auto i = 0; i < vertices.size(); i++) {
676 normals[face.getNormalIndices()[i]].set(normal);
679 if (progressCallback !=
nullptr) {
680 progress+= incrementPerFace / 2.0f;
681 if (facesEntityProcessed == 0 || facesEntityProcessed % 1000 == 0) progressCallback->
progress(progress);
682 facesEntityProcessed++;
688 progress =
computeNormals(subNodeIt.second, progressCallback, incrementPerFace, progress);
699 computeNormals(nodeIt.second, progressCallback, 1.0f /
static_cast<float>(faceCount), 0.0f);
702 if (progressCallback !=
nullptr) {
704 delete progressCallback;
718 if (shader ==
"foliage" || shader ==
"pbr-foliage" || shader ==
"tree" || shader ==
"pbr-tree") {
723 if (shader ==
"water") {
733 vector<Vector3> objectOrigins;
741 vector<Vector3> objectOrigins;
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);
756 for (
auto& normal: normals) {
757 normal = transformationsMatrix.multiplyNoTranslation(normal);
764 for (
auto& tangent: tangents) {
765 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
772 for (
auto& bitangent: bitangents) {
773 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
774 bitangent.normalize();
790 for (
auto& vertex: vertices) {
791 vertex = transformationsMatrix.multiply(vertex);
797 for (
auto& normal: normals) {
798 normal = transformationsMatrix.multiplyNoTranslation(normal);
805 for (
auto& tangent: tangents) {
806 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
813 for (
auto& bitangent: bitangents) {
814 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
815 bitangent.normalize();
827 if (node->
isJoint() ==
true)
return;
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;
839 if (excludeDiffuseTexture ==
true) {
842 materialUseCount[facesEntity.getMaterial()->getId()]++;
856 if (node->
isJoint() ==
true)
return;
865 for (
auto& vertex: vertices) {
866 vertex = transformationsMatrix.multiply(vertex);
872 for (
auto& normal: normals) {
873 normal = transformationsMatrix.multiplyNoTranslation(normal);
880 for (
auto& tangent: tangents) {
881 tangent = transformationsMatrix.multiplyNoTranslation(tangent);
888 for (
auto& bitangent: bitangents) {
889 bitangent = transformationsMatrix.multiplyNoTranslation(bitangent);
890 bitangent.normalize();
904void ModelTools::optimizeNode(
Node* sourceNode,
Model* targetModel,
int diffuseTextureAtlasSize,
const map<string, int>& diffuseTextureAtlasIndices,
const vector<string>& excludeDiffuseTextureFileNamePatterns) {
906 unordered_set<int> processedTextureCoordinates;
908 auto& sourceNormals = sourceNode->
getNormals();
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()) {
935 targetFacesEntitiesKeptMaterials.push_back(targetFacesEntity);
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>();
942 auto material = sourceFacesEntity.getMaterial();
945 string keptMaterialId;
946 for (
auto& excludeDiffuseTextureFileNamePattern: excludeDiffuseTextureFileNamePatterns) {
947 if (
StringTools::startsWith(sourceFacesEntity.getMaterial()->getSpecularMaterialProperties()->getDiffuseTextureFileName(), excludeDiffuseTextureFileNamePattern) ==
true) {
948 keptMaterialId = sourceFacesEntity.getMaterial()->getId();
952 auto targetFacesKeptMaterial = (tmpFacesEntity = targetNode->getFacesEntity(
"tdme.facesentity.kept." + keptMaterialId)) !=
nullptr?tmpFacesEntity->
getFaces():vector<Face>();
954 auto diffuseTextureAtlasIndexIt = diffuseTextureAtlasIndices.find(material->getId());
955 auto diffuseTextureAtlasIndex = -1;
956 if (diffuseTextureAtlasIndexIt != diffuseTextureAtlasIndices.end()) {
957 diffuseTextureAtlasIndex = diffuseTextureAtlasIndices.find(material->getId())->second;
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;
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();
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]);
989 if (keptMaterialId.empty() ==
false) {
990 targetFacesKeptMaterial.push_back(
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
1005 if (material->getSpecularMaterialProperties()->hasDiffuseTextureTransparency() ==
true) {
1006 if (material->getSpecularMaterialProperties()->hasDiffuseTextureMaskedTransparency() ==
true) {
1007 targetFacesMaskedTransparency.push_back(
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
1022 targetFacesTransparency.push_back(
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
1038 targetFaces.push_back(
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
1054 if (targetFacesKeptMaterial.size() > 0) {
1056 for (
auto& targetFacesEntityKeptMaterial: targetFacesEntitiesKeptMaterials) {
1057 if (targetFacesEntityKeptMaterial.getId() ==
"tdme.facesentity.kept." + keptMaterialId) {
1058 facesEntity = &targetFacesEntityKeptMaterial;
1062 if (facesEntity ==
nullptr) {
1063 targetFacesEntitiesKeptMaterials.push_back(
FacesEntity(targetNode,
"tdme.facesentity.kept." + keptMaterialId));
1064 facesEntity = &targetFacesEntitiesKeptMaterials[targetFacesEntitiesKeptMaterials.size() - 1];
1066 facesEntity->
setFaces(targetFacesKeptMaterial);
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);
1076 if (targetFacesMaskedTransparency.size() > 0) {
1077 targetFacesEntities.push_back(
FacesEntity(targetNode,
"tdme.facesentity.optimized.maskedtransparency"));
1078 targetFacesEntities[targetFacesEntities.size() - 1].setFaces(targetFacesMaskedTransparency);
1080 if (targetFacesTransparency.size() > 0) {
1081 targetFacesEntities.push_back(
FacesEntity(targetNode,
"tdme.facesentity.optimized.transparency"));
1082 targetFacesEntities[targetFacesEntities.size() - 1].setFaces(targetFacesTransparency);
1084 for (
auto& targetFacesEntityKeptMaterial: targetFacesEntitiesKeptMaterials) {
1085 targetFacesEntities.push_back(targetFacesEntityKeptMaterial);
1087 targetNode->setFacesEntities(targetFacesEntities);
1089 for (
auto& subNodeIt: sourceNode->
getSubNodes()) {
1090 optimizeNode(subNodeIt.second, targetModel, diffuseTextureAtlasSize, diffuseTextureAtlasIndices, excludeDiffuseTextureFileNamePatterns);
1095 auto textureAtlasSize =
static_cast<int>(Math::ceil(sqrt(textureAtlasTextures.size())));
1096 #define ATLAS_TEXTURE_SIZE 512
1097 #define ATLAS_TEXTURE_BORDER 32
1101 for (
auto y = 0; y < textureHeight; y++)
1102 for (
auto x = 0; x < textureWidth; x++) {
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));
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));
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);
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);
1145 auto atlasTexture =
new Texture(
1148 textureWidth, textureHeight,
1149 textureWidth, textureHeight,
1152 atlasTexture->setAtlasSize(textureAtlasSize);
1153 return atlasTexture;
1157 return model->
getNodes()[
"tdme.node.optimized"] !=
nullptr;
1166 map<string, int> materialUseCount;
1171 excludeDiffuseTextureFileNamePatterns
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;
1192 if (diffuseTextureCount < 2)
return model;
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;
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;
1222 if (keepDiffuseTexture ==
false)
continue;
1223 optimizedModel->getMaterials()[material->getId()] =
cloneMaterial(material);
1227 auto optimizedMaterial =
new Material(
"tdme.material.optimized");
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();
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);
1260 auto optimizedMaterialMaskedTransparency =
cloneMaterial(optimizedMaterial,
"tdme.material.optimized.maskedtransparency");
1261 optimizedMaterialMaskedTransparency->getSpecularMaterialProperties()->setDiffuseTextureTransparency(
true);
1262 optimizedMaterialMaskedTransparency->getSpecularMaterialProperties()->setDiffuseTextureMaskedTransparency(
true);
1265 auto optimizedMaterialTransparency =
cloneMaterial(optimizedMaterial,
"tdme.material.optimized.transparency");
1266 optimizedMaterialTransparency->getSpecularMaterialProperties()->setDiffuseTextureTransparency(
true);
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);
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);
1283 cloneNode(node, optimizedModel,
nullptr,
false);
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);
1293 delete optimizedMaterial;
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);
1302 delete optimizedMaterialMaskedTransparency;
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);
1311 delete optimizedMaterialTransparency;
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()
1328 optimizedModel->addAnimationSetup(
1329 animationSetup->getId(),
1330 animationSetup->getStartFrame(),
1331 animationSetup->getEndFrame(),
1332 animationSetup->isLoop(),
1333 animationSetup->getSpeed()
1342 return optimizedModel;
1349 vector<Vector3> tangents;
1350 vector<Vector3> bitangents;
1364 for (
auto& faceEntity: facesEntities) {
1365 auto faces = faceEntity.getFaces();
1366 for (
auto& face: faces) {
1368 auto verticesIndexes = face.getVertexIndices();
1369 auto v0 = vertices[verticesIndexes[0]];
1370 auto v1 = vertices[verticesIndexes[1]];
1371 auto v2 = vertices[verticesIndexes[2]];
1373 auto textureCoordinatesIndexes = face.getTextureCoordinateIndices();
1374 uv0.
set(textureCoordinates[textureCoordinatesIndexes[0]].getArray());
1376 uv1.
set(textureCoordinates[textureCoordinatesIndexes[1]].getArray());
1378 uv2.
set(textureCoordinates[textureCoordinatesIndexes[2]].getArray());
1381 deltaPos1.
set(v1).
sub(v0);
1382 deltaPos2.
set(v2).
sub(v0);
1384 deltaUV1.
set(uv1).
sub(uv0);
1385 deltaUV2.
set(uv2).
sub(uv0);
1387 auto r = 1.0f / (deltaUV1.
getX() * deltaUV2.
getY() - deltaUV1.
getY() * deltaUV2.
getX());
1391 face.setTangentIndices(tangents.size() + 0, tangents.size() + 1, tangents.size() + 2);
1393 face.setBitangentIndices(bitangents.size() + 0, bitangents.size() + 1, bitangents.size() + 2);
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);
1402 faceEntity.setFaces(faces);
1407 if (tangents.size() > 0 && bitangents.size() > 0) {
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);
1421 bitangent.normalize();
1424 faceEntity.setFaces(faces);
const vector< Matrix4x4 > & getTransformationsMatrices() const
Returns transformation matrices.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
void setTextureCoordinateIndices(int32_t vt0, int32_t vt1, int32_t vt2)
Set up optional texture coordinate indices.
void setTangentIndices(int32_t ti0, int32_t ti1, int32_t ti2)
Set tangent indices.
void setBitangentIndices(int32_t bi0, int32_t bi1, int32_t bi2)
Set bitangent indices.
Node faces entity A node can have multiple entities containing faces and a applied material.
void setMaterial(Material *material)
Set up the entity's material.
const vector< Face > & getFaces() const
void setFaces(const vector< Face > &faces)
Set up entity's faces.
const string & getId() const
const SpecularMaterialProperties * getSpecularMaterialProperties() const
Representation of a 3d model.
map< string, AnimationSetup * > & getAnimationSetups()
TODO: return const map.
RotationOrder * getRotationOrder()
const Matrix4x4 & getImportTransformationsMatrix()
AuthoringTool getAuthoringTool()
void setUpVector(UpVector *upVector)
Set up vector.
map< string, Node * > & getNodes()
Returns all object's nodes.
AnimationSetup * addAnimationSetup(const string &id, int32_t startFrame, int32_t endFrame, bool loop, float speed=1.0f)
Adds an base animation setup.
AnimationSetup * getAnimationSetup(const string &id)
void setImportTransformationsMatrix(const Matrix4x4 &importTransformationsMatrix)
Set import transformations matrix.
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
BoundingBox * getBoundingBox()
map< string, Material * > & getMaterials()
Returns all object materials.
int32_t getFaceCount() const
const vector< Vector3 > & getTangents() const
const vector< Vector3 > & getOrigins() const
const Matrix4x4 & getTransformationsMatrix() const
void setAnimation(Animation *animation)
Sets animation object.
void setTextureCoordinates(const vector< TextureCoordinate > &textureCoordinates)
Set texture coordinates.
void setOrigins(const vector< Vector3 > &origins)
Set origins.
void setVertices(const vector< Vector3 > &vertices)
Set vertices.
const vector< Vector3 > & getBitangents() const
void setFacesEntities(const vector< FacesEntity > &facesEntities)
Set up faces entities.
const vector< TextureCoordinate > & getTextureCoordinates() const
void setJoint(bool isJoint)
Sets up if this node is a joint or not.
const string & getId()
Returns id.
const vector< Vector3 > & getNormals() const
void setTransformationsMatrix(const Matrix4x4 &transformationsMatrix)
Animation * getAnimation()
const vector< FacesEntity > & getFacesEntities() const
void setNormals(const vector< Vector3 > &normals)
Set normals.
const vector< Vector3 > & getVertices() const
map< string, Node * > & getSubNodes()
void setBitangents(const vector< Vector3 > &bitangents)
Set bitangents.
void setTangents(const vector< Vector3 > &tangents)
Set tangents.
Skinning definition for nodes.
void setVerticesJointsWeights(const vector< vector< JointWeight > > &verticesJointsWeights)
Sets up vertices joints weights.
const vector< vector< JointWeight > > & getVerticesJointsWeights()
Represents specular material properties.
Class representing texture UV coordinates data.
TextureCoordinate & set(const TextureCoordinate &textureCoordinate)
Set texture coordinate.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Matrix4x4 clone()
Clones this matrix.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Vector2 & set(float x, float y)
Set up vector.
Vector2 & sub(const Vector2 &v)
Subtracts a vector.
Vector2 & setY(float y)
set Y
Vector3 & set(float x, float y, float z)
Set up vector.
Vector3 clone() const
Clones the vector.
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Vector3 & add(const Vector3 &v)
Adds a vector.
Vector3 & scale(float scale)
Scale this vector.
array< float, 4 > & getArray() const
static ByteBuffer * allocate(int32_t capacity)
Allocate byte buffer.
static void println()
Print new line to console.
virtual void progress(float value)=0
Perform action.