TDME2 1.9.121
ConvexMesh.cpp
Go to the documentation of this file.
2
3#include <algorithm>
4#include <array>
5#include <map>
6#include <unordered_set>
7#include <vector>
8
9#include <ext/reactphysics3d/src/collision/shapes/ConvexMeshShape.h>
10
11#include <tdme/tdme.h>
17#include <tdme/math/Math.h>
18#include <tdme/math/Matrix4x4.h>
19#include <tdme/math/Vector3.h>
25
26using std::array;
27using std::find;
28using std::map;
29using std::reverse;
30using std::sort;
31using std::unique;
32using std::unordered_set;
33using std::vector;
34
48
49ConvexMesh::ConvexMesh()
50{
51}
52
54{
55 if (polyhedronMesh != nullptr) delete polyhedronMesh;
56 if (polygonVertexArray != nullptr) delete polygonVertexArray;
57 if (verticesByteBuffer != nullptr) delete verticesByteBuffer;
58 if (indicesByteBuffer != nullptr) delete indicesByteBuffer;
59}
60
61inline bool ConvexMesh::isVertexOnTrianglePlane(Triangle& triangle, const Vector3& vertex) {
62 for (auto& triangleVertex: triangle.getVertices()) {
63 if (triangleVertex.equals(vertex) == true) return true;
64 }
65 // see: http://www.ambrsoft.com/TrigoCalc/Plan3D/PointsCoplanar.htm
66 Vector3 v1;
67 Vector3 v2;
68 Vector3 v3;
69 v1.set(triangle.getVertices()[1]).sub(triangle.getVertices()[0]).normalize();
70 v2.set(triangle.getVertices()[2]).sub(triangle.getVertices()[0]).normalize();
71 v3.set(vertex).sub(triangle.getVertices()[0]);
72 auto v1Dotv2v3Cross = Vector3::computeDotProduct(v1, Vector3::computeCrossProduct(v2, v3).normalize());
73 return Math::abs(v1Dotv2v3Cross) < Math::EPSILON;
74}
75
76inline bool ConvexMesh::areTrianglesAdjacent(Triangle& triangle1, Triangle& triangle2) {
77 auto equalVertices = 0;
78 for (auto& triangle1Vertex: triangle1.getVertices()) {
79 for (auto& triangle2Vertex: triangle2.getVertices()) {
80 if (triangle1Vertex.equals(triangle2Vertex)) equalVertices++;
81 }
82 }
83 return equalVertices == 2;
84}
85
86void ConvexMesh::createConvexMesh(const vector<Vector3>& vertices, const vector<int>& facesVerticesCount, const vector<int>& indices, const Vector3& scale) {
87 // delete old collision shape if we have any
88 if (collisionShape != nullptr) delete collisionShape;
89 if (polyhedronMesh != nullptr) delete polyhedronMesh;
90 if (polygonVertexArray != nullptr) delete polygonVertexArray;
91 if (verticesByteBuffer != nullptr) delete verticesByteBuffer;
92 if (indicesByteBuffer != nullptr) delete indicesByteBuffer;
93
94 // check if local translation is given
95 // determine center/position transformed
96 collisionShapeLocalTranslation.set(0.0f, 0.0f, 0.0f);
97 for (auto vertexIdx: indices) {
98 auto& vertex = vertices[vertexIdx];
100 }
102
103 // center
105
106 // scale collision shape local translation
108
109 // local transformations
111
112 // generate vertices and indices buffers
113 verticesByteBuffer = ByteBuffer::allocate(vertices.size() * 3 * sizeof(float));
114 indicesByteBuffer = ByteBuffer::allocate(indices.size() * sizeof(int));
115 auto verticesBuffer = verticesByteBuffer->asFloatBuffer();
116 auto indicesBuffer = indicesByteBuffer->asIntBuffer();
117 Vector3 vertexTransformed;
118 for (auto& vertex: vertices) {
119 vertexTransformed.set(vertex);
120 vertexTransformed.sub(center);
121 vertexTransformed.scale(scale);
122 verticesBuffer.put(vertexTransformed.getArray());
123 }
124 for (auto& index: indices) {
125 indicesBuffer.put(index);
126 }
127 faces.clear();
128 int indexIdx = 0;
129 for (auto faceVerticesCount: facesVerticesCount) {
130 reactphysics3d::PolygonVertexArray::PolygonFace face;
131 face.nbVertices = faceVerticesCount;
132 face.indexBase = indexIdx;
133 faces.push_back(face);
134 indexIdx+= faceVerticesCount;
135 }
136
137 //
138 polygonVertexArray = new reactphysics3d::PolygonVertexArray(
139 vertices.size(),
141 3 * sizeof(float),
143 sizeof(int),
144 faces.size(),
145 faces.data(),
146 reactphysics3d::PolygonVertexArray::VertexDataType::VERTEX_FLOAT_TYPE,
147 reactphysics3d::PolygonVertexArray::IndexDataType::INDEX_INTEGER_TYPE
148 );
149 polyhedronMesh = new reactphysics3d::PolyhedronMesh(polygonVertexArray);
150 // create convex mesh shape
151 auto convexMeshShape = new reactphysics3d::ConvexMeshShape(polyhedronMesh);
152 // set up new collision shape
153 collisionShape = convexMeshShape;
154 // transform
155 collisionShapeTransform = reactphysics3d::Transform();
156 // compute bounding box
158}
159
161{
162 vector<Triangle> triangles;
163 model->getTriangles(triangles);
164
165 // determine coplanar faces of model
166 map<int, vector<Triangle*>> trianglesCoplanar;
167 {
168 auto triangle1Idx = 0;
169 unordered_set<int> trianglesProcessed;
170
171 for (auto& triangle1: triangles) {
172 if (trianglesProcessed.find(triangle1Idx) != trianglesProcessed.end()) {
173 triangle1Idx++;
174 continue;
175 }
176 trianglesCoplanar[triangle1Idx].push_back(&triangle1);
177 trianglesProcessed.insert(triangle1Idx);
178 auto triangle2Added = 0;
179 do {
180 auto triangle2Idx = 0;
181 triangle2Added = 0;
182 for (auto& triangle2: triangles) {
183 if (trianglesProcessed.find(triangle2Idx) != trianglesProcessed.end()) {
184 triangle2Idx++;
185 continue;
186 }
187 auto adjacent = areTrianglesAdjacent(triangle1, triangle2);
188 auto triangle2OnTriangle1Plane =
189 isVertexOnTrianglePlane(triangle1, triangle2.getVertices()[0]) &&
190 isVertexOnTrianglePlane(triangle1, triangle2.getVertices()[1]) &&
191 isVertexOnTrianglePlane(triangle1, triangle2.getVertices()[2]);
192 if (adjacent == true && triangle2OnTriangle1Plane == true) {
193 trianglesCoplanar[triangle1Idx].push_back(&triangle2);
194 trianglesProcessed.insert(triangle2Idx);
195 triangle2Added++;
196 }
197 triangle2Idx++;
198 }
199 } while (triangle2Added > 0);
200 triangle1Idx++;
201 }
202 }
203
204 // iterate triangles that are coplanar and build a polygon
205 for (auto& trianglesCoplanarIt: trianglesCoplanar) {
206 auto& trianglesCoplanarVector = trianglesCoplanarIt.second;
207
208 // collect polygon vertices
209 vector<Vector3> polygonVertices;
210
211 // determine polygon vertices
212 for (auto& triangle: trianglesCoplanarVector) {
213 for (auto& triangleVertex: triangle->getVertices()) {
214 bool foundVertex = false;
215 for (auto& polygonVertex: polygonVertices) {
216 if (polygonVertex.equals(triangleVertex) == true) {
217 foundVertex = true;
218 break;
219 }
220 }
221 if (foundVertex == false) {
222 polygonVertices.push_back(triangleVertex);
223 }
224 }
225 }
226
227 // remove vertices that live on the line 2 other 2 vertices span
228 /*
229 {
230 vector<int> polygonVerticesToRemove;
231 Vector3 c;
232 for (auto i = 0; i < polygonVertices.size(); i++) {
233 for (auto j = 0; j < polygonVertices.size(); j++) {
234 if (i == j) continue;
235 for (auto k = 0; k < polygonVertices.size(); k++) {
236 if (i == k || j == k) continue;
237 LineSegment::computeClosestPointOnLineSegment(
238 polygonVertices[i],
239 polygonVertices[j],
240 polygonVertices[k],
241 c
242 );
243 if (polygonVertices[k].equals(c) == true) polygonVerticesToRemove.push_back(k);
244 }
245 }
246 }
247 sort(polygonVerticesToRemove.begin(), polygonVerticesToRemove.end());
248 polygonVerticesToRemove.erase(unique(polygonVerticesToRemove.begin(), polygonVerticesToRemove.end()), polygonVerticesToRemove.end());
249 auto polygonVerticesToRemoved = 0;
250 for (auto i: polygonVerticesToRemove) {
251 polygonVertices.erase(polygonVertices.begin() + i - polygonVerticesToRemoved);
252 polygonVerticesToRemoved++;
253 }
254 }
255 */
256
257 // check if to skip as combined polygons could already have the current polygon
258 if (polygonVertices.size() > 2) {
259 auto skip = false;
260 auto idx = 0;
261 for (auto faceVertexCount: facesVerticesCount) {
262 unordered_set<int> foundIndices;
263 for (auto i = 0; i < faceVertexCount; i++) {
264 auto foundVertex = false;
265 for (auto& polygonVertex: polygonVertices) {
266 if (polygonVertex.equals(vertices[indices[idx]]) == true) {
267 foundIndices.insert(indices[idx]);
268 break;
269 }
270 }
271 idx++;
272 }
273 if (foundIndices.size() == polygonVertices.size()) {
274 skip = true;
275 break;
276 }
277 }
278 if (skip == true) {
279 continue;
280 }
281 }
282
283 //
284 if (polygonVertices.size() < 3) continue;
285
286 // determine polygon center, a point outside of mesh viewing the polygon
287 Vector3 polygonCenter;
288 for (auto& polygonVertex: polygonVertices) {
289 polygonCenter.add(polygonVertex);
290 }
291 polygonCenter.scale(1.0f / polygonVertices.size());
292
293 // plane normal
294 Vector3 triangle1Edge1;
295 Vector3 triangle1Edge2;
296 triangle1Edge1.set(trianglesCoplanarVector[0]->getVertices()[1]).sub(trianglesCoplanarVector[0]->getVertices()[0]).normalize();
297 triangle1Edge2.set(trianglesCoplanarVector[0]->getVertices()[2]).sub(trianglesCoplanarVector[0]->getVertices()[0]).normalize();
298 auto polygonNormal = Vector3::computeCrossProduct(triangle1Edge1, triangle1Edge2).normalize();
299
300 // determine polygon vertices order
301 vector<int> polygonVerticesOrdered;
302 // add first vertex
303 polygonVerticesOrdered.push_back(0);
304
305 // then check vertex order if it matches
306 // if it matches we have the next vertex
307 Vector3 distanceVector;
308 // as long as we have vertices left
309 while (polygonVerticesOrdered.size() != polygonVertices.size()) {
310 // find next vertex with most little
311 auto hitVertexAngle = 0.0f;
312 auto hitVertexIdx = -1;
313 for (int i = 0; i < polygonVertices.size(); i++) {
314 // check if already add to ordered vertices list
315 if (find(polygonVerticesOrdered.begin(), polygonVerticesOrdered.end(), i) != polygonVerticesOrdered.end()) continue;
316
317 // otherwise check if angle is smaller
318 auto angleCurrent = Vector3::computeAngle(
319 polygonVertices[0].clone().sub(polygonCenter).normalize(),
320 polygonVertices[i].clone().sub(polygonCenter).normalize(),
321 polygonNormal
322 );
323 if (hitVertexIdx == -1 || angleCurrent < hitVertexAngle) {
324 hitVertexAngle = angleCurrent;
325 hitVertexIdx = i;
326 }
327 }
328 // yep
329 polygonVerticesOrdered.push_back(hitVertexIdx);
330 }
331
332 /*
333 {
334 // vertex order
335 // see: https://stackoverflow.com/questions/14370636/sorting-a-list-of-3d-coplanar-points-to-be-clockwise-or-counterclockwise
336 auto& polygonVertexOrderedFirst = polygonVertices[polygonVerticesOrdered[0]];
337 auto& polygonVertexOrderedLast = polygonVertices[polygonVerticesOrdered[1]];
338 Vector3 ac;
339 Vector3 bc;
340 Vector3 acbcCross;
341 ac.set(polygonVertexOrderedFirst).sub(polygonCenter);
342 bc.set(polygonVertexOrderedLast).sub(polygonCenter);
343 Vector3::computeCrossProduct(ac, bc, acbcCross);
344 // counter clockwise???
345 if ((Vector3::computeDotProduct(polygonNormal, acbcCross) > 0.0f) == false) {
346 // yep, reverse
347 reverse(begin(polygonVerticesOrdered), end(polygonVerticesOrdered));
348 }
349 }
350 */
351
352 // add face
353 facesVerticesCount.push_back(polygonVerticesOrdered.size());
354 for (auto polygonVerticesOrderedIdx: polygonVerticesOrdered) {
355 // polygon vertex
356 auto& polygonVertex = polygonVertices[polygonVerticesOrderedIdx];
357
358 // check if to insert vertex
359 int vertexIdx = 0;
360 for (auto& vertexExisting: vertices) {
361 if (vertexExisting.equals(polygonVertex) == true) {
362 break;
363 }
364 vertexIdx++;
365 }
366 if (vertexIdx == vertices.size()) {
367 vertices.push_back(polygonVertex);
368 }
369
370 // add index
371 indices.push_back(vertexIdx);
372 }
373 }
374
375 // create convex mesh
377}
378
379ConvexMesh::ConvexMesh(const vector<Vector3>& vertices, const vector<int>& facesVerticesCount, const vector<int>& indices, const Vector3& scale) {
380 this->vertices = vertices;
381 this->facesVerticesCount = facesVerticesCount;
382 this->indices = indices;
384}
385
386void ConvexMesh::setScale(const Vector3& scale) {
387 // store new scale
388 this->scale.set(scale);
389 // recreate convex mesh
391}
392
394{
396}
397
398const vector<Vector3>& ConvexMesh::getVertices() {
399 return vertices;
400}
Transformations which contain scale, rotations and translation.
reactphysics3d::Transform collisionShapeLocalTransform
reactphysics3d::Transform collisionShapeTransform
reactphysics3d::CollisionShape * collisionShape
void computeBoundingBox()
Compute bounding box.
Convex mesh physics primitive.
Definition: ConvexMesh.h:33
~ConvexMesh()
Public denstructor.
Definition: ConvexMesh.cpp:53
bool isVertexOnTrianglePlane(Triangle &triangle, const Vector3 &vertex)
Checks if vertex lives on triangle plane.
Definition: ConvexMesh.cpp:61
reactphysics3d::PolygonVertexArray * polygonVertexArray
Definition: ConvexMesh.h:40
bool areTrianglesAdjacent(Triangle &triangle1, Triangle &triangle2)
Checks if 2 triangles are adjacent.
Definition: ConvexMesh.cpp:76
reactphysics3d::PolyhedronMesh * polyhedronMesh
Definition: ConvexMesh.h:41
void createConvexMesh(const vector< Vector3 > &vertices, const vector< int > &facesVerticesCount, const vector< int > &indices, const Vector3 &scale)
Create convex mesh Note: it also translates center into origin.
Definition: ConvexMesh.cpp:86
void setScale(const Vector3 &scale) override
Set local scale.
Definition: ConvexMesh.cpp:386
const vector< Vector3 > & getVertices()
Definition: ConvexMesh.cpp:398
BoundingVolume * clone() const override
Clones this bounding volume.
Definition: ConvexMesh.cpp:393
vector< reactphysics3d::PolygonVertexArray::PolygonFace > faces
Definition: ConvexMesh.h:39
Line segment helper functions.
Definition: LineSegment.h:17
Triangle entity, this is not directly connectable with physics engine.
Definition: Triangle.h:19
vector< Vector3 > & getVertices()
Definition: Triangle.h:37
void getTriangles(vector< Triangle > &triangles, int nodeIdx=-1)
Retrieves list of triangles of all or given nodes.
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Definition: Vector3.h:325
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
array< float, 3 > & getArray() const
Definition: Vector3.h:171
uint8_t * getBuffer()
Definition: Buffer.h:131
Byte buffer class.
Definition: ByteBuffer.h:24
FloatBuffer asFloatBuffer()
Definition: ByteBuffer.h:36
Float buffer class.
Definition: FloatBuffer.h:18
Float class.
Definition: Float.h:23
Integer buffer class.
Definition: IntBuffer.h:14
Model tools functions class.
Definition: ModelTools.h:38