45BoundingBox* ModelUtilitiesInternal::createBoundingBox(
Model* model,
const map<string, Matrix4x4*> overriddenNodeTransformationsMatrices)
48 object3dModel.
instanceAnimations[0]->overriddenTransformationsMatrices = overriddenNodeTransformationsMatrices;
56 auto model = object3DModelInternal->
getModel();
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;
63 animationState.
setup = defaultAnimation;
66 animationState.
time = 0.0f;
68 for (
auto t = 0.0f; t <= (defaultAnimation != nullptr ? static_cast<float>(defaultAnimation->getFrames()) : 0.0f) / model->getFPS(); t += 1.0f / model->getFPS()) {
72 object3DModelInternal->
instanceAnimations[0]->computeTransformationsMatrices(object3DModelInternal->
instanceAnimations[0]->nodeLists[0], parentTransformationsMatrix, &animationState);
75 for (
auto object3DNode : object3DModelInternal->
object3dNodes) {
76 for (
auto& vertex : *object3DNode->mesh->vertices) {
77 auto& vertexXYZ = vertex.getArray();
78 if (firstVertex ==
true) {
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];
96 animationState.
currentAtTime =
static_cast< int64_t
>((t * 1000.0f));
97 animationState.
lastAtTime =
static_cast< int64_t
>((t * 1000.0f));
100 if (firstVertex ==
true)
return nullptr;
107 auto model = object3DModelInternal->
getModel();
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;
114 animationState.
setup = defaultAnimation;
117 animationState.
time = 0.0f;
120 for (
auto t = 0.0f; t <= (defaultAnimation != nullptr ? static_cast<float>(defaultAnimation->getFrames()) : 0.0f) / model->getFPS(); t += 1.0f / model->getFPS()) {
124 object3DModelInternal->
instanceAnimations[0]->computeTransformationsMatrices(object3DModelInternal->
instanceAnimations[0]->nodeLists[0], parentTransformationsMatrix, &animationState);
125 for (
auto nodeIt: model->getNodes()) {
127 vertex = transformedNodeMatrix.multiply(vertex.
set(0.0f, 0.0f, 0.0f));
128 if (firstVertex ==
true) {
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];
145 animationState.
currentAtTime =
static_cast< int64_t
>((t * 1000.0f));
146 animationState.
lastAtTime =
static_cast< int64_t
>((t * 1000.0f));
149 if (firstVertex ==
true)
return nullptr;
161 for (
auto it: nodes) {
162 Node* node = it.second;
164 for (
auto& normal : normals) {
182 map<string, int32_t> materialCountById;
183 auto opaqueFaceCount = 0;
184 auto transparentFaceCount = 0;
185 for (
auto object3DNode : object3DModelInternal->
object3dNodes) {
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();
193 auto material = facesEntity.getMaterial();
195 auto transparentFacesEntity =
false;
197 if (material !=
nullptr) {
198 auto specularMaterialProperties = material->getSpecularMaterialProperties();
199 if (specularMaterialProperties !=
nullptr &&
200 (specularMaterialProperties->hasColorTransparency() ==
true || specularMaterialProperties->hasTextureTransparency() ==
true))
201 transparentFacesEntity =
true;
205 auto materialId = material ==
nullptr ?
"tdme.material.none" : material->getId();
206 materialCountById[materialId]++;
208 if (transparentFacesEntity ==
true) {
210 transparentFaceCount += faces;
214 opaqueFaceCount += faces;
218 auto materialCount = materialCountById.size();
237 for (
auto i = 0; i < object3DModel1Internal->
object3dNodes.size(); i++) {
238 auto object3DNodeModel1 = object3DModel1Internal->
object3dNodes[i];
239 auto object3DNodeModel2 = object3DModel2Internal->
object3dNodes[i];
242 auto& facesEntitiesModel1 = object3DNodeModel1->node->getFacesEntities();
243 auto& facesEntitiesModel2 = object3DNodeModel2->node->getFacesEntities();
245 if (object3DNodeModel1->node->getTransformationsMatrix().equals(object3DNodeModel2->node->getTransformationsMatrix()) ==
false)
248 if (node1->getVertices().size() != node2->getVertices().size())
251 for (
auto j = 0; j < node1->getVertices().size(); j++) {
252 if (node1->getVertices()[j].equals(node2->getVertices()[j]) ==
false)
256 if (node1->getNormals().size() != node2->getNormals().size())
259 for (
auto j = 0; j < node1->getNormals().size(); j++) {
260 if (node1->getNormals()[j].equals(node2->getNormals()[j]) ==
false)
264 if (facesEntitiesModel1.size() != facesEntitiesModel2.size())
267 for (
auto j = 0; j < facesEntitiesModel1.size(); j++) {
268 auto facesEntityModel1 = facesEntitiesModel1[j];
269 auto facesEntityModel2 = facesEntitiesModel2[j];
272 if (facesEntityModel1.getMaterial() ==
nullptr && facesEntityModel2.getMaterial() !=
nullptr)
275 if (facesEntityModel1.getMaterial() !=
nullptr && facesEntityModel2.getMaterial() ==
nullptr)
278 if (facesEntityModel1.getMaterial() !=
nullptr && facesEntityModel2.getMaterial() !=
nullptr &&
279 facesEntityModel1.getMaterial()->getId() != facesEntityModel2.getMaterial()->getId()) {
283 auto& facesModel1 = facesEntityModel1.getFaces();
284 auto& facesModel2 = facesEntityModel2.getFaces();
286 if (facesModel1.size() != facesModel2.size())
289 for (
auto k = 0; k < facesModel1.size(); k++) {
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]) {
static constexpr int64_t UNDEFINED
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.
Representation of a 3d model.
const Matrix4x4 & getImportTransformationsMatrix()
AnimationSetup * getAnimationSetup(const string &id)
map< string, Node * > & getSubNodes()
Returns object's sub nodes.
const vector< Vector3 > & getNormals() const
void setNormals(const vector< Vector3 > &normals)
Set normals.
map< string, Node * > & getSubNodes()
Represents specular material properties.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
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
vector< Object3DNode * > object3dNodes
vector< Object3DAnimation * > instanceAnimations
const Matrix4x4 getNodeTransformationsMatrix(const string &id)
Returns transformation matrix for given node.
Object 3D model To be used in non engine context.
Object 3D node mesh specifically for rendering.
Object 3d node specifically for rendering.
static void computeTransformations(int contextIdx, vector< Object3DNode * > &object3DNodes)
Applies transformations to meshes for given object 3d nodes.
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Vector3 & set(float x, float y, float z)
Set up vector.
int32_t transparentFaceCount