TDME2 1.9.121
Object3DRenderGroup.cpp
Go to the documentation of this file.
2
3#include <string>
4#include <vector>
5
6#include <tdme/tdme.h>
16#include <tdme/engine/Engine.h>
21#include <tdme/math/Math.h>
22#include <tdme/math/Matrix4x4.h>
23#include <tdme/math/Vector3.h>
25
26using std::string;
27using std::to_string;
28using std::vector;
29
49
50Object3DRenderGroup::Object3DRenderGroup(
51 const string& id,
52 int lodLevels,
53 float modelLOD2MinDistance,
54 float modelLOD3MinDistance,
55 int modelLOD2ReduceBy,
56 int modelLOD3ReduceBy,
57 bool optimizeModels
58):
59 id(id)
60{
61 this->enabled = true;
62 this->pickable = false;
63 this->contributesShadows = false;
64 this->receivesShadows = false;
65 this->effectColorMul.set(1.0f, 1.0f, 1.0f, 1.0f);
66 this->effectColorAdd.set(0.0f, 0.0f, 0.0f, 0.0f);
67 this->combinedModels.resize(Math::clamp(lodLevels, 1, 3));
68 this->combinedEntity = nullptr;
69 this->lodReduceBy[0] = 1;
70 this->lodReduceBy[1] = modelLOD2ReduceBy;
71 this->lodReduceBy[2] = modelLOD3ReduceBy;
72 this->modelLOD2MinDistance = modelLOD2MinDistance;
73 this->modelLOD3MinDistance = modelLOD3MinDistance;
74 this->optimizeModels = optimizeModels;
75}
76
78 if (combinedEntity != nullptr) delete combinedEntity;
79 for (auto combinedModel: combinedModels) {
80 if (combinedModel != nullptr) delete combinedModel;
81 }
82}
83
84void Object3DRenderGroup::combineNode(Node* sourceNode, const vector<Vector3>& origins, const vector<Matrix4x4>& objectParentTransformationsMatrices, Model* combinedModel) {
85 // create node in combined model
86 auto combinedModelNode = combinedModel->getNodeById(sourceNode->getId());
87 if (combinedModelNode == nullptr) {
88 combinedModelNode = new Node(
89 combinedModel,
90 sourceNode->getParentNode() == nullptr?nullptr:combinedModel->getNodeById(sourceNode->getParentNode()->getId()),
91 sourceNode->getId(),
92 sourceNode->getName()
93 );
94 if (sourceNode->getParentNode() == nullptr) {
95 combinedModel->getSubNodes()[combinedModelNode->getId()] = combinedModelNode;
96 } else {
97 combinedModelNode->getParentNode()->getSubNodes()[combinedModelNode->getId()] = combinedModelNode;
98 }
99 combinedModel->getNodes()[combinedModelNode->getId()] = combinedModelNode;
100 }
101
102 {
103 auto sourceNodeVerticesSize = sourceNode->getVertices().size();
104 auto sourceNodeNormalsSize = sourceNode->getNormals().size();
105 auto sourceNodeTextureCoordinatesSize = sourceNode->getTextureCoordinates().size();
106 auto sourceNodeTangentsSize = sourceNode->getTangents().size();
107 auto sourceNodeBitangentsSize = sourceNode->getBitangents().size();
108
109 // vertices and such from new model
110 auto combinedModelNodeVertices = combinedModelNode->getVertices();
111 auto combinedModelNodeNormals = combinedModelNode->getNormals();
112 auto combinedModelNodeTextureCoordinates = combinedModelNode->getTextureCoordinates();
113 auto combinedModelNodeTangents = combinedModelNode->getTangents();
114 auto combinedModelNodeBitangents = combinedModelNode->getBitangents();
115 auto combinedModelNodeFacesEntities = combinedModelNode->getFacesEntities();
116 auto combinedModelNodeOrigins = combinedModelNode->getOrigins();
117
118 // current indices
119 auto combinedModelNodeVerticesIdxStart = combinedModelNodeVertices.size();
120 auto combinedModelNodeNormalsIdxStart = combinedModelNodeNormals.size();
121 auto combinedModelNodeTextureCoordinatesIdxStart = combinedModelNodeTextureCoordinates.size();
122 auto combinedModelNodeTangentsIdxStart = combinedModelNodeTangents.size();
123 auto combinedModelNodeBitangentsIdxStart = combinedModelNodeBitangents.size();
124
125 // add vertices and such from source node to new node
126 {
127 auto i = 0;
128 for (auto& objectParentTransformationsMatrix: objectParentTransformationsMatrices) {
131 transformationsMatrix.multiply(objectParentTransformationsMatrix);
132
133 //
134 for (auto& vertex: sourceNode->getVertices()) {
135 combinedModelNodeOrigins.push_back(origins[i]);
136 combinedModelNodeVertices.push_back(transformationsMatrix.multiply(vertex));
137 }
138 for (auto& normal: sourceNode->getNormals()) {
139 combinedModelNodeNormals.push_back(transformationsMatrix.multiplyNoTranslation(normal));
140 }
141 for (auto& textureCoordinate: sourceNode->getTextureCoordinates()) {
142 combinedModelNodeTextureCoordinates.push_back(textureCoordinate);
143 }
144 for (auto& tangent: sourceNode->getTangents()) {
145 combinedModelNodeTangents.push_back(transformationsMatrix.multiplyNoTranslation(tangent));
146 }
147 for (auto& bitangent: sourceNode->getBitangents()) {
148 combinedModelNodeBitangents.push_back(transformationsMatrix.multiplyNoTranslation(bitangent));
149 }
150
151 //
152 i++;
153 }
154 }
155
156 // add source node faces to new new faces entity
157 for (auto& facesEntity: sourceNode->getFacesEntities()) {
158 bool haveTextureCoordinates = facesEntity.isTextureCoordinatesAvailable();
159 bool haveTangentsBitangents = facesEntity.isTangentBitangentAvailable();
160
161 // get faces entity
162 FacesEntity* combinedModelNodeFacesEntity = nullptr;
163 for (auto& combinedModelNodeFacesEntityExisting: combinedModelNodeFacesEntities) {
164 if (combinedModelNodeFacesEntityExisting.getId() == facesEntity.getId()) {
165 combinedModelNodeFacesEntity = &combinedModelNodeFacesEntityExisting;
166 break;
167 }
168 }
169 // create
170 if (combinedModelNodeFacesEntity == nullptr) {
171 auto newFacesEntity = FacesEntity(
172 combinedModelNode,
173 facesEntity.getId()
174 );
175 combinedModelNodeFacesEntities.push_back(newFacesEntity);
176 combinedModelNodeFacesEntity = &combinedModelNodeFacesEntities[combinedModelNodeFacesEntities.size() - 1];
177 auto combinedModelNodeFacesEntityMaterial = combinedModel->getMaterials()[facesEntity.getMaterial()->getId()];
178 if (combinedModelNodeFacesEntityMaterial == nullptr) {
179 combinedModelNodeFacesEntityMaterial = ModelTools::cloneMaterial(facesEntity.getMaterial());
180 combinedModel->getMaterials()[combinedModelNodeFacesEntityMaterial->getId()] = combinedModelNodeFacesEntityMaterial;
181 }
182 combinedModelNodeFacesEntity->setMaterial(combinedModelNodeFacesEntityMaterial);
183 }
184
185 //
186 auto combinedModelNodeFaces = combinedModelNodeFacesEntity->getFaces();
187
188 //
189 auto combinedModelNodeVerticesIdx = combinedModelNodeVerticesIdxStart;
190 auto combinedModelNodeNormalsIdx = combinedModelNodeNormalsIdxStart;
191 auto combinedModelNodeTextureCoordinatesIdx = combinedModelNodeTextureCoordinatesIdxStart;
192 auto combinedModelNodeTangentsIdx = combinedModelNodeTangentsIdxStart;
193 auto combinedModelNodeBitangentsIdx = combinedModelNodeBitangentsIdxStart;
194 for (auto& objectParentTransformationsMatrix: objectParentTransformationsMatrices) {
195 // add faces
196 for (auto& face: facesEntity.getFaces()) {
197 // get face vertices and such
198 auto& faceVertexIndices = face.getVertexIndices();
199 auto& faceNormalIndices = face.getNormalIndices();
200 auto& faceTextureCoordinatesIndices = face.getTextureCoordinateIndices();
201 auto& faceTangentIndices = face.getTangentIndices();
202 auto& faceBitangentIndices = face.getBitangentIndices();
203
204 //
205 auto combinedModelNodeFace =
206 Face(
207 combinedModelNode,
208 combinedModelNodeVerticesIdx + faceVertexIndices[0],
209 combinedModelNodeVerticesIdx + faceVertexIndices[1],
210 combinedModelNodeVerticesIdx + faceVertexIndices[2],
211 combinedModelNodeNormalsIdx + faceNormalIndices[0],
212 combinedModelNodeNormalsIdx + faceNormalIndices[1],
213 combinedModelNodeNormalsIdx + faceNormalIndices[2]
214 );
215 if (haveTextureCoordinates == true) {
216 combinedModelNodeFace.setTextureCoordinateIndices(
217 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[0],
218 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[1],
219 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[2]
220 );
221 }
222 if (haveTangentsBitangents == true) {
223 combinedModelNodeFace.setTangentIndices(
224 combinedModelNodeTangentsIdx + faceTangentIndices[0],
225 combinedModelNodeTangentsIdx + faceTangentIndices[1],
226 combinedModelNodeTangentsIdx + faceTangentIndices[2]
227 );
228 combinedModelNodeFace.setBitangentIndices(
229 combinedModelNodeBitangentsIdx + faceBitangentIndices[0],
230 combinedModelNodeBitangentsIdx + faceBitangentIndices[1],
231 combinedModelNodeBitangentsIdx + faceBitangentIndices[2]
232 );
233 }
234 combinedModelNodeFaces.push_back(combinedModelNodeFace);
235 }
236
237 //
238 combinedModelNodeVerticesIdx+= sourceNodeVerticesSize;
239 combinedModelNodeNormalsIdx+= sourceNodeNormalsSize;
240 combinedModelNodeTextureCoordinatesIdx+= sourceNodeTextureCoordinatesSize;
241 combinedModelNodeTangentsIdx+= sourceNodeTangentsSize;
242 combinedModelNodeBitangentsIdx+= sourceNodeBitangentsSize;
243 }
244 combinedModelNodeFacesEntity->setFaces(combinedModelNodeFaces);
245 }
246
247 // store back to model
248 combinedModelNode->setVertices(combinedModelNodeVertices);
249 combinedModelNode->setNormals(combinedModelNodeNormals);
250 combinedModelNode->setTextureCoordinates(combinedModelNodeTextureCoordinates);
251 combinedModelNode->setTangents(combinedModelNodeTangents);
252 combinedModelNode->setBitangents(combinedModelNodeBitangents);
253 combinedModelNode->setFacesEntities(combinedModelNodeFacesEntities);
254 combinedModelNode->setOrigins(combinedModelNodeOrigins);
255 }
256
257 // do child nodes
258 for (auto nodeIt: sourceNode->getSubNodes()) {
259 combineNode(nodeIt.second, origins, objectParentTransformationsMatrices, combinedModel);
260 }
261}
262
263void Object3DRenderGroup::combineObjects(Model* model, const vector<Transformations>& objectsTransformations, Model* combinedModel) {
264 vector<Matrix4x4> objectTransformationMatrices;
265 vector<Vector3> origins;
266 for (auto& objectTransformations: objectsTransformations) {
269 transformationsMatrix.multiply(objectTransformations.getTransformationsMatrix());
270 objectTransformationMatrices.push_back(transformationsMatrix);
271 origins.push_back(objectTransformations.getTranslation());
272 }
273 for (auto nodeIt: model->getSubNodes()) {
274 combineNode(nodeIt.second, origins, objectTransformationMatrices, combinedModel);
275 }
276}
277
279 // dispose old object and combined model
280 if (combinedEntity != nullptr) {
283 delete combinedEntity;
284 combinedEntity = nullptr;
285 }
286
287 // combine objects to a new model
288 for (auto i = 0; i < combinedModels.size(); i++) {
289 if (combinedModels[i] != nullptr) delete combinedModels[i];
290 combinedModels[i] = new Model(
291 id + ".o3rg.lod." + to_string(i),
292 id + ".o3rg.lod." + to_string(i),
293 UpVector::Y_UP,
294 RotationOrder::ZYX,
295 nullptr
296 );
297 }
298
299 //
300 for (auto& transformationsByModelIt: transformationsByModel) {
301 auto model = transformationsByModelIt.first;
302 auto& objectsTransformations = transformationsByModelIt.second;
303 auto lodLevel = 0;
304 for (auto combinedModel: combinedModels) {
305 auto reduceByFactor = lodReduceBy[lodLevel];
306 lodLevel++;
307 auto objectCount = 0;
308 vector<Transformations> reducedObjectsTransformations;
309 for (auto& objectTransformations: objectsTransformations) {
310 if (objectCount % reduceByFactor != 0) {
311 objectCount++;
312 continue;
313 }
314 reducedObjectsTransformations.push_back(objectTransformations);
315 objectCount++;
316 }
317 combineObjects(model, reducedObjectsTransformations, combinedModel);
318 }
319 }
320
321 // create new combined object
322 for (auto combinedModel: combinedModels) {
323 if (combinedModel != nullptr) {
324 // post process combined model
325 ModelTools::shrinkToFit(combinedModel);
326 ModelTools::createDefaultAnimation(combinedModel, 0);
327 ModelTools::setupJoints(combinedModel);
328 ModelTools::fixAnimationLength(combinedModel);
329 }
330 }
331
332 // optimize models
333 if (optimizeModels == true) {
334 for (auto i = 0; i < combinedModels.size(); i++) {
335 combinedModels[i] = ModelTools::optimizeModel(combinedModels[i]);
336 }
337 }
338
339 // no lod
340 if (combinedModels.size() == 1) {
341 auto combinedObject3D = new Object3D(id + ".o3rg", combinedModels[0]);
342 combinedObject3D->setParentEntity(this);
343 combinedObject3D->setShader(shaderId);
344 combinedObject3D->setDistanceShader(distanceShaderId);
345 combinedObject3D->setContributesShadows(contributesShadows);
346 combinedObject3D->setReceivesShadows(receivesShadows);
347 combinedObject3D->setEngine(engine);
348 combinedObject3D->setEnableEarlyZRejection(enableEarlyZRejection);
349 combinedObject3D->shaderParameters = shaderParameters;
350 combinedObject3D->distanceShaderParameters = distanceShaderParameters;
351 combinedObject3D->update();
352 combinedEntity = combinedObject3D;
353 } else
354 // lod
355 if (combinedModels.size() > 1) {
356 // create object, initialize and
357 auto combinedLODObject3D = new LODObject3D(
358 id + ".o3rg",
366 );
367 combinedLODObject3D->setParentEntity(this);
368 combinedLODObject3D->setShader(shaderId);
369 combinedLODObject3D->setDistanceShader(distanceShaderId);
370 combinedLODObject3D->setDistanceShaderDistance(distanceShaderDistance);
371 combinedLODObject3D->setContributesShadows(contributesShadows);
372 combinedLODObject3D->setReceivesShadows(receivesShadows);
373 combinedLODObject3D->setEngine(engine);
374 combinedLODObject3D->setEnableEarlyZRejection(enableEarlyZRejection);
375 combinedLODObject3D->shaderParameters = shaderParameters;
376 combinedLODObject3D->distanceShaderParameters = distanceShaderParameters;
377 if (combinedLODObject3D->objectLOD1 != nullptr) {
378 combinedLODObject3D->objectLOD1->shaderParameters = shaderParameters;
379 combinedLODObject3D->objectLOD1->distanceShaderParameters = distanceShaderParameters;
380 }
381 if (combinedLODObject3D->objectLOD2 != nullptr) {
382 combinedLODObject3D->objectLOD2->shaderParameters = shaderParameters;
383 combinedLODObject3D->objectLOD2->distanceShaderParameters = distanceShaderParameters;
384 }
385 if (combinedLODObject3D->objectLOD3 != nullptr) {
386 combinedLODObject3D->objectLOD3->shaderParameters = shaderParameters;
387 combinedLODObject3D->objectLOD3->distanceShaderParameters = distanceShaderParameters;
388 }
389 combinedLODObject3D->update();
390 combinedEntity = combinedLODObject3D;
391 }
392
393 //
395 update();
396}
397
398void Object3DRenderGroup::addObject(Model* model, const Transformations& transformations) {
399 transformationsByModel[model].push_back(transformations);
400}
401
403{
404 if (this->engine != nullptr) this->engine->deregisterEntity(this);
405 this->engine = engine;
406 if (engine != nullptr) engine->registerEntity(this);
408}
409
411{
412}
413
415{
417 // update bounding box transformed
419 // update object
420 if (parentEntity == nullptr && frustumCulling == true && engine != nullptr && enabled == true) engine->partition->updateEntity(this);
421}
422
424{
426 // update bounding box transformed
428 // update object
429 if (parentEntity == nullptr && frustumCulling == true && engine != nullptr && enabled == true) engine->partition->updateEntity(this);
430}
431
433{
434 // return if enable state has not changed
435 if (this->enabled == enabled) return;
436
437 // frustum if root entity
438 if (parentEntity == nullptr) {
439 // frustum culling enabled?
440 if (frustumCulling == true) {
441 // yeo, add or remove from partition
442 if (enabled == true) {
443 if (engine != nullptr) engine->partition->addEntity(this);
444 } else {
445 if (engine != nullptr) engine->partition->removeEntity(this);
446 }
447 }
448 }
449
450 //
451 this->enabled = enabled;
452}
453
455 return frustumCulling;
456}
457
458void Object3DRenderGroup::setFrustumCulling(bool frustumCulling) {
459 // check if enabled and engine attached
460 if (enabled == true && engine != nullptr) {
461 // had frustum culling
462 if (this->frustumCulling == true) {
463 // yep, remove if set to false now
464 if (frustumCulling == false) engine->partition->removeEntity(this);
465 } else {
466 // yep, add if set to true now
467 if (frustumCulling == true) engine->partition->addEntity(this);
468 }
469 }
470 this->frustumCulling = frustumCulling;
471 // delegate change to engine
472 if (engine != nullptr) engine->registerEntity(this);
473}
474
476{
477 // delegate to combined object
478 if (combinedEntity != nullptr) {
480 delete combinedEntity;
481 combinedEntity = nullptr;
482 }
483 // disose combined model
484 for (auto& combinedModel: combinedModels) {
485 if (combinedModel != nullptr) {
486 delete combinedModel;
487 combinedModel = nullptr;
488 }
489 }
490}
491
493{
494 if (combinedEntity != nullptr) combinedEntity->initialize();
495}
496
Engine main class.
Definition: Engine.h:122
void deregisterEntity(Entity *entity)
Removes a entity from internal lists, those entities can also be sub entities from entity hierarchy o...
Definition: Engine.cpp:392
void removeEntityFromLists(Entity *entity)
Remove entity.
Definition: Engine.cpp:615
void registerEntity(Entity *entity)
Adds a entity to internal lists, those entities can also be sub entities from entity hierarchy or par...
Definition: Engine.cpp:406
Partition * partition
Definition: Engine.h:255
virtual void setEngine(Engine *engine)=0
Set up engine.
virtual void initialize()=0
Initiates this object 3d.
virtual void dispose()=0
Dispose this object 3d.
LOD object 3D to be used with engine class.
Definition: LODObject3D.h:47
Object 3D render group for static objects that might be animated by shaders.
map< Model *, vector< Transformations > > transformationsByModel
void dispose() override
Dispose this object 3d.
void initialize() override
Initiates this object 3d.
EntityShaderParameters shaderParameters
void update() override
Update transformations.
static void combineObjects(Model *model, const vector< Transformations > &objectsTransformations, Model *combinedModel)
Combine model with transformations into current model.
void addObject(Model *model, const Transformations &transformations)
Adds a instance to this render group.
void updateRenderGroup()
Update render group model and bounding box.
static void combineNode(Node *sourceNode, const vector< Vector3 > &origins, const vector< Matrix4x4 > &objectParentTransformationsMatrices, Model *combinedModel)
Combine node into given combined model.
void fromTransformations(const Transformations &transformations) override
Set up this transformations from given transformations.
void updateBoundingBox()
Compute bounding box.
EntityShaderParameters distanceShaderParameters
void setFrustumCulling(bool frustumCulling) override
Set frustum culling.
void setEngine(Engine *engine) override
Set up engine.
void setEnabled(bool enabled) override
Enable/disable rendering.
void setRenderer(Renderer *renderer) override
Set up renderer.
Object 3D to be used with engine class.
Definition: Object3D.h:60
Transformations which contain scale, rotations and translation.
virtual void fromTransformations(const Transformations &transformations)
Set up this transformations from given transformations.
virtual void update()
Computes transformation matrix.
void set(const array< float, 4 > &color)
Set up color.
Definition: Color4Base.h:68
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
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
Represents a material.
Definition: Material.h:21
Representation of a 3d model.
Definition: Model.h:32
const Matrix4x4 & getImportTransformationsMatrix()
Definition: Model.h:293
map< string, Node * > & getNodes()
Returns all object's nodes.
Definition: Model.h:179
Node * getNodeById(const string &id)
Returns a node by given name or null.
Definition: Model.cpp:78
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
Definition: Model.h:194
map< string, Material * > & getMaterials()
Returns all object materials.
Definition: Model.h:171
Model node.
Definition: Node.h:31
const vector< Vector3 > & getTangents() const
Definition: Node.h:199
const Matrix4x4 & getTransformationsMatrix() const
Definition: Node.h:121
Node * getParentNode()
Definition: Node.h:77
const vector< Vector3 > & getBitangents() const
Definition: Node.h:212
const vector< TextureCoordinate > & getTextureCoordinates() const
Definition: Node.h:186
const string & getId()
Returns id.
Definition: Node.h:85
const vector< Vector3 > & getNormals() const
Definition: Node.h:173
const string & getName()
Definition: Node.h:92
const vector< FacesEntity > & getFacesEntities() const
Definition: Node.h:256
const vector< Vector3 > & getVertices() const
Definition: Node.h:151
map< string, Node * > & getSubNodes()
Definition: Node.h:289
Represents rotation orders of a model.
Definition: RotationOrder.h:23
Class representing texture UV coordinates data.
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
void fromBoundingVolumeWithTransformations(BoundingBox *original, const Transformations &transformations)
Create bounding volume from given original(of same type) with applied transformations.
Definition: BoundingBox.cpp:79
Standard math functions.
Definition: Math.h:21
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
Vector3 multiplyNoTranslation(const Vector3 &v) const
Multiplies a vector3 with this matrix ignoring translation.
Definition: Matrix4x4.h:364
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
Model tools functions class.
Definition: ModelTools.h:38
Partition interface.
Definition: Partition.h:19
virtual void updateEntity(Entity *entity)=0
Updates a entity.
virtual void addEntity(Entity *entity)=0
Adds a entity.
virtual void removeEntity(Entity *entity)=0
Removes a entity.