6#include <unordered_set>
9#include <ext/reactphysics3d/src/collision/shapes/ConvexMeshShape.h>
32using std::unordered_set;
49ConvexMesh::ConvexMesh()
62 for (
auto& triangleVertex: triangle.
getVertices()) {
63 if (triangleVertex.equals(vertex) ==
true)
return true;
72 auto v1Dotv2v3Cross = Vector3::computeDotProduct(v1, Vector3::computeCrossProduct(v2, v3).normalize());
73 return Math::abs(v1Dotv2v3Cross) < Math::EPSILON;
77 auto equalVertices = 0;
78 for (
auto& triangle1Vertex: triangle1.
getVertices()) {
79 for (
auto& triangle2Vertex: triangle2.
getVertices()) {
80 if (triangle1Vertex.equals(triangle2Vertex)) equalVertices++;
83 return equalVertices == 2;
119 vertexTransformed.
set(vertex);
122 verticesBuffer.put(vertexTransformed.
getArray());
125 indicesBuffer.put(index);
130 reactphysics3d::PolygonVertexArray::PolygonFace face;
131 face.nbVertices = faceVerticesCount;
132 face.indexBase = indexIdx;
133 faces.push_back(face);
134 indexIdx+= faceVerticesCount;
146 reactphysics3d::PolygonVertexArray::VertexDataType::VERTEX_FLOAT_TYPE,
147 reactphysics3d::PolygonVertexArray::IndexDataType::INDEX_INTEGER_TYPE
151 auto convexMeshShape =
new reactphysics3d::ConvexMeshShape(
polyhedronMesh);
162 vector<Triangle> triangles;
166 map<int, vector<Triangle*>> trianglesCoplanar;
168 auto triangle1Idx = 0;
169 unordered_set<int> trianglesProcessed;
171 for (
auto& triangle1: triangles) {
172 if (trianglesProcessed.find(triangle1Idx) != trianglesProcessed.end()) {
176 trianglesCoplanar[triangle1Idx].push_back(&triangle1);
177 trianglesProcessed.insert(triangle1Idx);
178 auto triangle2Added = 0;
180 auto triangle2Idx = 0;
182 for (
auto& triangle2: triangles) {
183 if (trianglesProcessed.find(triangle2Idx) != trianglesProcessed.end()) {
188 auto triangle2OnTriangle1Plane =
192 if (adjacent ==
true && triangle2OnTriangle1Plane ==
true) {
193 trianglesCoplanar[triangle1Idx].push_back(&triangle2);
194 trianglesProcessed.insert(triangle2Idx);
199 }
while (triangle2Added > 0);
205 for (
auto& trianglesCoplanarIt: trianglesCoplanar) {
206 auto& trianglesCoplanarVector = trianglesCoplanarIt.second;
209 vector<Vector3> polygonVertices;
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) {
221 if (foundVertex ==
false) {
222 polygonVertices.push_back(triangleVertex);
258 if (polygonVertices.size() > 2) {
262 unordered_set<int> foundIndices;
263 for (
auto i = 0; i < faceVertexCount; i++) {
264 auto foundVertex =
false;
265 for (
auto& polygonVertex: polygonVertices) {
267 foundIndices.insert(
indices[idx]);
273 if (foundIndices.size() == polygonVertices.size()) {
284 if (polygonVertices.size() < 3)
continue;
288 for (
auto& polygonVertex: polygonVertices) {
289 polygonCenter.
add(polygonVertex);
291 polygonCenter.
scale(1.0f / polygonVertices.size());
298 auto polygonNormal = Vector3::computeCrossProduct(triangle1Edge1, triangle1Edge2).
normalize();
301 vector<int> polygonVerticesOrdered;
303 polygonVerticesOrdered.push_back(0);
309 while (polygonVerticesOrdered.size() != polygonVertices.size()) {
311 auto hitVertexAngle = 0.0f;
312 auto hitVertexIdx = -1;
313 for (
int i = 0; i < polygonVertices.size(); i++) {
315 if (find(polygonVerticesOrdered.begin(), polygonVerticesOrdered.end(), i) != polygonVerticesOrdered.end())
continue;
318 auto angleCurrent = Vector3::computeAngle(
319 polygonVertices[0].
clone().sub(polygonCenter).normalize(),
320 polygonVertices[i].
clone().sub(polygonCenter).normalize(),
323 if (hitVertexIdx == -1 || angleCurrent < hitVertexAngle) {
324 hitVertexAngle = angleCurrent;
329 polygonVerticesOrdered.push_back(hitVertexIdx);
354 for (
auto polygonVerticesOrderedIdx: polygonVerticesOrdered) {
356 auto& polygonVertex = polygonVertices[polygonVerticesOrderedIdx];
360 for (
auto& vertexExisting:
vertices) {
361 if (vertexExisting.equals(polygonVertex) ==
true) {
Bounding volume interface.
reactphysics3d::Transform collisionShapeLocalTransform
reactphysics3d::Transform collisionShapeTransform
Vector3 collisionShapeLocalTranslation
reactphysics3d::CollisionShape * collisionShape
void computeBoundingBox()
Compute bounding box.
Convex mesh physics primitive.
~ConvexMesh()
Public denstructor.
bool isVertexOnTrianglePlane(Triangle &triangle, const Vector3 &vertex)
Checks if vertex lives on triangle plane.
reactphysics3d::PolygonVertexArray * polygonVertexArray
vector< int > facesVerticesCount
bool areTrianglesAdjacent(Triangle &triangle1, Triangle &triangle2)
Checks if 2 triangles are adjacent.
reactphysics3d::PolyhedronMesh * polyhedronMesh
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.
ByteBuffer * verticesByteBuffer
void setScale(const Vector3 &scale) override
Set local scale.
ConvexMesh()
Public constructor.
const vector< Vector3 > & getVertices()
vector< Vector3 > vertices
BoundingVolume * clone() const override
Clones this bounding volume.
vector< reactphysics3d::PolygonVertexArray::PolygonFace > faces
ByteBuffer * indicesByteBuffer
Line segment helper functions.
Triangle entity, this is not directly connectable with physics engine.
vector< Vector3 > & getVertices()
void getTriangles(vector< Triangle > &triangles, int nodeIdx=-1)
Retrieves list of triangles of all or given nodes.
Vector3 & normalize()
Normalize the vector.
Vector3 & set(float x, float y, float z)
Set up vector.
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Vector3 & add(const Vector3 &v)
Adds a vector.
Vector3 & scale(float scale)
Scale this vector.
array< float, 3 > & getArray() const
FloatBuffer asFloatBuffer()