TDME2 1.9.121
ModelUtilitiesInternal.cpp
Go to the documentation of this file.
2
3#include <map>
4#include <string>
5
6#include <tdme/tdme.h>
20#include <tdme/engine/Timing.h>
21#include <tdme/math/Matrix4x4.h>
22#include <tdme/math/Vector3.h>
23
24using std::map;
25using std::string;
26
44
45BoundingBox* ModelUtilitiesInternal::createBoundingBox(Model* model, const map<string, Matrix4x4*> overriddenNodeTransformationsMatrices)
46{
47 Object3DModelInternal object3dModel(model);
48 object3dModel.instanceAnimations[0]->overriddenTransformationsMatrices = overriddenNodeTransformationsMatrices;
49 auto boundingBox = ModelUtilitiesInternal::createBoundingBox(&object3dModel);
50 if (boundingBox == nullptr) boundingBox = ModelUtilitiesInternal::createBoundingBoxNoMesh(&object3dModel);
51 return boundingBox;
52}
53
55{
56 auto model = object3DModelInternal->getModel();
57 auto defaultAnimation = model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT);
58 float minX = 0.0f, minY = 0.0f, minZ = 0.0f;
59 float maxX = 0.0f, maxY = 0.0f, maxZ = 0.0f;
60 auto firstVertex = true;
61 // create bounding box for whole animation at 60fps
62 AnimationState animationState;
63 animationState.setup = defaultAnimation;
64 animationState.lastAtTime = Timing::UNDEFINED;
65 animationState.currentAtTime = 0LL;
66 animationState.time = 0.0f;
67 animationState.finished = false;
68 for (auto t = 0.0f; t <= (defaultAnimation != nullptr ? static_cast<float>(defaultAnimation->getFrames()) : 0.0f) / model->getFPS(); t += 1.0f / model->getFPS()) {
69 // calculate transformations matrices without world transformations
70 auto parentTransformationsMatrix = object3DModelInternal->getModel()->getImportTransformationsMatrix();
71 parentTransformationsMatrix.multiply(object3DModelInternal->getTransformationsMatrix());
72 object3DModelInternal->instanceAnimations[0]->computeTransformationsMatrices(object3DModelInternal->instanceAnimations[0]->nodeLists[0], parentTransformationsMatrix, &animationState);
73 Object3DNode::computeTransformations(0, object3DModelInternal->object3dNodes);
74 // parse through object nodes to determine min, max
75 for (auto object3DNode : object3DModelInternal->object3dNodes) {
76 for (auto& vertex : *object3DNode->mesh->vertices) {
77 auto& vertexXYZ = vertex.getArray();
78 if (firstVertex == true) {
79 minX = vertexXYZ[0];
80 minY = vertexXYZ[1];
81 minZ = vertexXYZ[2];
82 maxX = vertexXYZ[0];
83 maxY = vertexXYZ[1];
84 maxZ = vertexXYZ[2];
85 firstVertex = false;
86 } else {
87 if (vertexXYZ[0] < minX) minX = vertexXYZ[0];
88 if (vertexXYZ[1] < minY) minY = vertexXYZ[1];
89 if (vertexXYZ[2] < minZ) minZ = vertexXYZ[2];
90 if (vertexXYZ[0] > maxX) maxX = vertexXYZ[0];
91 if (vertexXYZ[1] > maxY) maxY = vertexXYZ[1];
92 if (vertexXYZ[2] > maxZ) maxZ = vertexXYZ[2];
93 }
94 }
95 }
96 animationState.currentAtTime = static_cast< int64_t >((t * 1000.0f));
97 animationState.lastAtTime = static_cast< int64_t >((t * 1000.0f));
98 }
99 // skip on models without mesh
100 if (firstVertex == true) return nullptr;
101 // otherwise go with bounding box
102 return new BoundingBox(Vector3(minX, minY, minZ), Vector3(maxX, maxY, maxZ));
103}
104
106{
107 auto model = object3DModelInternal->getModel();
108 auto defaultAnimation = model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT);
109 float minX = 0.0f, minY = 0.0f, minZ = 0.0f;
110 float maxX = 0.0f, maxY = 0.0f, maxZ = 0.0f;
111 auto firstVertex = true;
112 // create bounding box for whole animation at 60fps
113 AnimationState animationState;
114 animationState.setup = defaultAnimation;
115 animationState.lastAtTime = Timing::UNDEFINED;
116 animationState.currentAtTime = 0LL;
117 animationState.time = 0.0f;
118 animationState.finished = false;
119 Vector3 vertex;
120 for (auto t = 0.0f; t <= (defaultAnimation != nullptr ? static_cast<float>(defaultAnimation->getFrames()) : 0.0f) / model->getFPS(); t += 1.0f / model->getFPS()) {
121 // calculate transformations matrices without world transformations
122 auto parentTransformationsMatrix = object3DModelInternal->getModel()->getImportTransformationsMatrix();
123 parentTransformationsMatrix.multiply(object3DModelInternal->getTransformationsMatrix());
124 object3DModelInternal->instanceAnimations[0]->computeTransformationsMatrices(object3DModelInternal->instanceAnimations[0]->nodeLists[0], parentTransformationsMatrix, &animationState);
125 for (auto nodeIt: model->getNodes()) {
126 auto& transformedNodeMatrix = object3DModelInternal->getNodeTransformationsMatrix(nodeIt.second->getId());
127 vertex = transformedNodeMatrix.multiply(vertex.set(0.0f, 0.0f, 0.0f));
128 if (firstVertex == true) {
129 minX = vertex[0];
130 minY = vertex[1];
131 minZ = vertex[2];
132 maxX = vertex[0];
133 maxY = vertex[1];
134 maxZ = vertex[2];
135 firstVertex = false;
136 } else {
137 if (vertex[0] < minX) minX = vertex[0];
138 if (vertex[1] < minY) minY = vertex[1];
139 if (vertex[2] < minZ) minZ = vertex[2];
140 if (vertex[0] > maxX) maxX = vertex[0];
141 if (vertex[1] > maxY) maxY = vertex[1];
142 if (vertex[2] > maxZ) maxZ = vertex[2];
143 }
144 }
145 animationState.currentAtTime = static_cast< int64_t >((t * 1000.0f));
146 animationState.lastAtTime = static_cast< int64_t >((t * 1000.0f));
147 }
148 // skip on models without nodes
149 if (firstVertex == true) return nullptr;
150 // otherwise go with bounding box
151 return new BoundingBox(Vector3(minX, minY, minZ), Vector3(maxX, maxY, maxZ));
152}
153
155{
156 invertNormals(model->getSubNodes());
157}
158
159void ModelUtilitiesInternal::invertNormals(const map<string, Node*>& nodes)
160{
161 for (auto it: nodes) {
162 Node* node = it.second;
163 auto normals = node->getNormals();
164 for (auto& normal : normals) {
165 // invert
166 normal.scale(-1.0f);
167 }
168 node->setNormals(normals);
169 // process sub nodes
170 invertNormals(node->getSubNodes());
171 }
172}
173
175{
176 Object3DModelInternal object3DModelInternal(model);
177 computeModelStatistics(&object3DModelInternal, modelStatistics);
178}
179
181{
182 map<string, int32_t> materialCountById;
183 auto opaqueFaceCount = 0;
184 auto transparentFaceCount = 0;
185 for (auto object3DNode : object3DModelInternal->object3dNodes) {
186 // check each faces entity
187 auto& facesEntities = object3DNode->node->getFacesEntities();
188 auto facesEntityIdxCount = facesEntities.size();
189 for (auto faceEntityIdx = 0; faceEntityIdx < facesEntityIdxCount; faceEntityIdx++) {
190 auto& facesEntity = facesEntities[faceEntityIdx];
191 auto faces = facesEntity.getFaces().size();
192 // material
193 auto material = facesEntity.getMaterial();
194 // determine if transparent
195 auto transparentFacesEntity = false;
196 // via material
197 if (material != nullptr) {
198 auto specularMaterialProperties = material->getSpecularMaterialProperties();
199 if (specularMaterialProperties != nullptr &&
200 (specularMaterialProperties->hasColorTransparency() == true || specularMaterialProperties->hasTextureTransparency() == true))
201 transparentFacesEntity = true;
202
203 }
204 // setup material usage
205 auto materialId = material == nullptr ? "tdme.material.none" : material->getId();
206 materialCountById[materialId]++;
207 // skip, if requested
208 if (transparentFacesEntity == true) {
209 // keep track of rendered faces
210 transparentFaceCount += faces;
211 // skip to next entity
212 continue;
213 }
214 opaqueFaceCount += faces;
215 }
216 }
217 // determine final material count
218 auto materialCount = materialCountById.size();
219 modelStatistics->opaqueFaceCount = opaqueFaceCount;
220 modelStatistics->transparentFaceCount = transparentFaceCount;
221 modelStatistics->materialCount = materialCount;
222}
223
225{
226 Object3DModelInternal object3DModel1(model1);
227 Object3DModelInternal object3DModel2(model2);
228 return ModelUtilitiesInternal::equals(&object3DModel1, &object3DModel2);
229}
230
231bool ModelUtilitiesInternal::equals(Object3DModelInternal* object3DModel1Internal, Object3DModelInternal* object3DModel2Internal)
232{
233 // check number of object 3d nodes
234 if (object3DModel1Internal->object3dNodes.size() != object3DModel2Internal->object3dNodes.size())
235 return false;
236
237 for (auto i = 0; i < object3DModel1Internal->object3dNodes.size(); i++) {
238 auto object3DNodeModel1 = object3DModel1Internal->object3dNodes[i];
239 auto object3DNodeModel2 = object3DModel2Internal->object3dNodes[i];
240 auto node1 = object3DModel1Internal->object3dNodes[i]->node;
241 auto node2 = object3DModel2Internal->object3dNodes[i]->node;
242 auto& facesEntitiesModel1 = object3DNodeModel1->node->getFacesEntities();
243 auto& facesEntitiesModel2 = object3DNodeModel2->node->getFacesEntities();
244 // check transformation matrix
245 if (object3DNodeModel1->node->getTransformationsMatrix().equals(object3DNodeModel2->node->getTransformationsMatrix()) == false)
246 return false;
247 // check vertices count
248 if (node1->getVertices().size() != node2->getVertices().size())
249 return false;
250 // check vertices
251 for (auto j = 0; j < node1->getVertices().size(); j++) {
252 if (node1->getVertices()[j].equals(node2->getVertices()[j]) == false)
253 return false;
254 }
255 // check normals count
256 if (node1->getNormals().size() != node2->getNormals().size())
257 return false;
258 // check normals
259 for (auto j = 0; j < node1->getNormals().size(); j++) {
260 if (node1->getNormals()[j].equals(node2->getNormals()[j]) == false)
261 return false;
262 }
263 // check number of faces entities
264 if (facesEntitiesModel1.size() != facesEntitiesModel2.size())
265 return false;
266 // check each faces entity
267 for (auto j = 0; j < facesEntitiesModel1.size(); j++) {
268 auto facesEntityModel1 = facesEntitiesModel1[j];
269 auto facesEntityModel2 = facesEntitiesModel2[j];
270 // check material
271 // TODO: check if it should be allowed to have NULL material
272 if (facesEntityModel1.getMaterial() == nullptr && facesEntityModel2.getMaterial() != nullptr)
273 return false;
274
275 if (facesEntityModel1.getMaterial() != nullptr && facesEntityModel2.getMaterial() == nullptr)
276 return false;
277
278 if (facesEntityModel1.getMaterial() != nullptr && facesEntityModel2.getMaterial() != nullptr &&
279 facesEntityModel1.getMaterial()->getId() != facesEntityModel2.getMaterial()->getId()) {
280 return false;
281 }
282 // check faces
283 auto& facesModel1 = facesEntityModel1.getFaces();
284 auto& facesModel2 = facesEntityModel2.getFaces();
285 // number of faces in faces entity
286 if (facesModel1.size() != facesModel2.size())
287 return false;
288 // face indices
289 for (auto k = 0; k < facesModel1.size(); k++) {
290 // vertex indices
291 auto vertexIndicesModel1 = facesModel1[k].getVertexIndices();
292 auto vertexIndicesModel2 = facesModel2[k].getVertexIndices();
293 if (vertexIndicesModel1[0] != vertexIndicesModel2[0] ||
294 vertexIndicesModel1[1] != vertexIndicesModel2[1] ||
295 vertexIndicesModel1[2] != vertexIndicesModel2[2]) {
296 return false;
297 }
298 // TODO: maybe other indices
299 }
300 // TODO: check vertices, normals and such
301 }
302 }
303 //
304 return true;
305}
Timing class.
Definition: Timing.h:17
static constexpr int64_t UNDEFINED
Definition: Timing.h:21
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
Represents a material.
Definition: Material.h:21
Representation of a 3d model.
Definition: Model.h:32
const Matrix4x4 & getImportTransformationsMatrix()
Definition: Model.h:293
AnimationSetup * getAnimationSetup(const string &id)
Definition: Model.cpp:116
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
Definition: Model.h:194
Model node.
Definition: Node.h:31
const vector< Vector3 > & getNormals() const
Definition: Node.h:173
void setNormals(const vector< Vector3 > &normals)
Set normals.
Definition: Node.cpp:60
map< string, Node * > & getSubNodes()
Definition: Node.h:289
Represents specular material properties.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
static void invertNormals(Model *model)
Invert normals of a model.
static BoundingBox * createBoundingBoxNoMesh(Object3DModelInternal *object3DModelInternal)
Creates a bounding box from given object3d model without mesh.
static void computeModelStatistics(Model *model, ModelStatistics *modelStatistics)
Compute model statistics.
static bool equals(Model *model1, Model *model2)
Compute if model 1 equals model 2.
static BoundingBox * createBoundingBox(Model *model, const map< string, Matrix4x4 * > overriddenNodeTransformationsMatrices=map< string, Matrix4x4 * >())
Creates a bounding box from given model.
const Matrix4x4 & getTransformationsMatrix() const
Definition: Object3DBase.h:279
vector< Object3DAnimation * > instanceAnimations
Definition: Object3DBase.h:54
const Matrix4x4 getNodeTransformationsMatrix(const string &id)
Returns transformation matrix for given node.
Definition: Object3DBase.h:255
Object 3D model To be used in non engine context.
Object 3D node mesh specifically for rendering.
Object 3d node specifically for rendering.
Definition: Object3DNode.h:39
static void computeTransformations(int contextIdx, vector< Object3DNode * > &object3DNodes)
Applies transformations to meshes for given object 3d nodes.
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
3D vector 3 class
Definition: Vector3.h:22
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73