50Object3DRenderGroup::Object3DRenderGroup(
53 float modelLOD2MinDistance,
54 float modelLOD3MinDistance,
55 int modelLOD2ReduceBy,
56 int modelLOD3ReduceBy,
80 if (combinedModel !=
nullptr)
delete combinedModel;
87 if (combinedModelNode ==
nullptr) {
88 combinedModelNode =
new Node(
95 combinedModel->
getSubNodes()[combinedModelNode->getId()] = combinedModelNode;
97 combinedModelNode->getParentNode()->getSubNodes()[combinedModelNode->getId()] = combinedModelNode;
99 combinedModel->
getNodes()[combinedModelNode->getId()] = combinedModelNode;
103 auto sourceNodeVerticesSize = sourceNode->
getVertices().size();
104 auto sourceNodeNormalsSize = sourceNode->
getNormals().size();
106 auto sourceNodeTangentsSize = sourceNode->
getTangents().size();
107 auto sourceNodeBitangentsSize = sourceNode->
getBitangents().size();
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();
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();
128 for (
auto& objectParentTransformationsMatrix: objectParentTransformationsMatrices) {
135 combinedModelNodeOrigins.push_back(origins[i]);
138 for (
auto& normal: sourceNode->
getNormals()) {
142 combinedModelNodeTextureCoordinates.push_back(textureCoordinate);
158 bool haveTextureCoordinates = facesEntity.isTextureCoordinatesAvailable();
159 bool haveTangentsBitangents = facesEntity.isTangentBitangentAvailable();
162 FacesEntity* combinedModelNodeFacesEntity =
nullptr;
163 for (
auto& combinedModelNodeFacesEntityExisting: combinedModelNodeFacesEntities) {
164 if (combinedModelNodeFacesEntityExisting.getId() == facesEntity.getId()) {
165 combinedModelNodeFacesEntity = &combinedModelNodeFacesEntityExisting;
170 if (combinedModelNodeFacesEntity ==
nullptr) {
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;
182 combinedModelNodeFacesEntity->
setMaterial(combinedModelNodeFacesEntityMaterial);
186 auto combinedModelNodeFaces = combinedModelNodeFacesEntity->
getFaces();
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) {
196 for (
auto& face: facesEntity.getFaces()) {
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();
205 auto combinedModelNodeFace =
208 combinedModelNodeVerticesIdx + faceVertexIndices[0],
209 combinedModelNodeVerticesIdx + faceVertexIndices[1],
210 combinedModelNodeVerticesIdx + faceVertexIndices[2],
211 combinedModelNodeNormalsIdx + faceNormalIndices[0],
212 combinedModelNodeNormalsIdx + faceNormalIndices[1],
213 combinedModelNodeNormalsIdx + faceNormalIndices[2]
215 if (haveTextureCoordinates ==
true) {
216 combinedModelNodeFace.setTextureCoordinateIndices(
217 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[0],
218 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[1],
219 combinedModelNodeTextureCoordinatesIdx + faceTextureCoordinatesIndices[2]
222 if (haveTangentsBitangents ==
true) {
223 combinedModelNodeFace.setTangentIndices(
224 combinedModelNodeTangentsIdx + faceTangentIndices[0],
225 combinedModelNodeTangentsIdx + faceTangentIndices[1],
226 combinedModelNodeTangentsIdx + faceTangentIndices[2]
228 combinedModelNodeFace.setBitangentIndices(
229 combinedModelNodeBitangentsIdx + faceBitangentIndices[0],
230 combinedModelNodeBitangentsIdx + faceBitangentIndices[1],
231 combinedModelNodeBitangentsIdx + faceBitangentIndices[2]
234 combinedModelNodeFaces.push_back(combinedModelNodeFace);
238 combinedModelNodeVerticesIdx+= sourceNodeVerticesSize;
239 combinedModelNodeNormalsIdx+= sourceNodeNormalsSize;
240 combinedModelNodeTextureCoordinatesIdx+= sourceNodeTextureCoordinatesSize;
241 combinedModelNodeTangentsIdx+= sourceNodeTangentsSize;
242 combinedModelNodeBitangentsIdx+= sourceNodeBitangentsSize;
244 combinedModelNodeFacesEntity->
setFaces(combinedModelNodeFaces);
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);
259 combineNode(nodeIt.second, origins, objectParentTransformationsMatrices, combinedModel);
264 vector<Matrix4x4> objectTransformationMatrices;
265 vector<Vector3> origins;
266 for (
auto& objectTransformations: objectsTransformations) {
271 origins.push_back(objectTransformations.getTranslation());
274 combineNode(nodeIt.second, origins, objectTransformationMatrices, combinedModel);
291 id +
".o3rg.lod." + to_string(i),
292 id +
".o3rg.lod." + to_string(i),
301 auto model = transformationsByModelIt.first;
302 auto& objectsTransformations = transformationsByModelIt.second;
307 auto objectCount = 0;
308 vector<Transformations> reducedObjectsTransformations;
309 for (
auto& objectTransformations: objectsTransformations) {
310 if (objectCount % reduceByFactor != 0) {
314 reducedObjectsTransformations.push_back(objectTransformations);
317 combineObjects(model, reducedObjectsTransformations, combinedModel);
323 if (combinedModel !=
nullptr) {
325 ModelTools::shrinkToFit(combinedModel);
326 ModelTools::createDefaultAnimation(combinedModel, 0);
327 ModelTools::setupJoints(combinedModel);
328 ModelTools::fixAnimationLength(combinedModel);
342 combinedObject3D->setParentEntity(
this);
343 combinedObject3D->setShader(
shaderId);
347 combinedObject3D->setEngine(
engine);
351 combinedObject3D->update();
367 combinedLODObject3D->setParentEntity(
this);
368 combinedLODObject3D->setShader(
shaderId);
373 combinedLODObject3D->setEngine(
engine);
377 if (combinedLODObject3D->objectLOD1 !=
nullptr) {
381 if (combinedLODObject3D->objectLOD2 !=
nullptr) {
385 if (combinedLODObject3D->objectLOD3 !=
nullptr) {
389 combinedLODObject3D->update();
435 if (this->enabled ==
enabled)
return;
462 if (this->frustumCulling ==
true) {
485 if (combinedModel !=
nullptr) {
486 delete combinedModel;
487 combinedModel =
nullptr;
void deregisterEntity(Entity *entity)
Removes a entity from internal lists, those entities can also be sub entities from entity hierarchy o...
void removeEntityFromLists(Entity *entity)
Remove entity.
void registerEntity(Entity *entity)
Adds a entity to internal lists, those entities can also be sub entities from entity hierarchy or par...
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.
Object 3D render group for static objects that might be animated by shaders.
map< Model *, vector< Transformations > > transformationsByModel
BoundingBox boundingBoxTransformed
void dispose() override
Dispose this object 3d.
~Object3DRenderGroup()
Destructor.
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.
array< int, 3 > lodReduceBy
float modelLOD3MinDistance
static void combineNode(Node *sourceNode, const vector< Vector3 > &origins, const vector< Matrix4x4 > &objectParentTransformationsMatrices, Model *combinedModel)
Combine node into given combined model.
float distanceShaderDistance
void fromTransformations(const Transformations &transformations) override
Set up this transformations from given transformations.
void updateBoundingBox()
Compute bounding box.
bool enableEarlyZRejection
float modelLOD2MinDistance
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.
vector< Model * > combinedModels
bool isFrustumCulling() override
void setRenderer(Renderer *renderer) override
Set up renderer.
Object 3D to be used with engine class.
void set(const array< float, 4 > &color)
Set up color.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
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.
Representation of a 3d model.
const Matrix4x4 & getImportTransformationsMatrix()
map< string, Node * > & getNodes()
Returns all object's nodes.
Node * getNodeById(const string &id)
Returns a node by given name or null.
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
map< string, Material * > & getMaterials()
Returns all object materials.
const vector< Vector3 > & getTangents() const
const Matrix4x4 & getTransformationsMatrix() const
const vector< Vector3 > & getBitangents() const
const vector< TextureCoordinate > & getTextureCoordinates() const
const string & getId()
Returns id.
const vector< Vector3 > & getNormals() const
const vector< FacesEntity > & getFacesEntities() const
const vector< Vector3 > & getVertices() const
map< string, Node * > & getSubNodes()
Represents rotation orders of a model.
Class representing texture UV coordinates data.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
void fromBoundingVolumeWithTransformations(BoundingBox *original, const Transformations &transformations)
Create bounding volume from given original(of same type) with applied transformations.
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.
Vector3 multiplyNoTranslation(const Vector3 &v) const
Multiplies a vector3 with this matrix ignoring translation.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
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.