TDME2 1.9.121
GenerateConvexMeshes.cpp
Go to the documentation of this file.
2
3#include <string>
4#include <unordered_map>
5#include <vector>
6
7#include <ext/v-hacd/src/VHACD_Lib/public/VHACD.h>
8
9#include <tdme/tdme.h>
42
44
45using std::string;
46using std::to_string;
47using std::unordered_map;
48using std::vector;
49
50using namespace VHACD;
51
84
85void GenerateConvexMeshes::removeConvexMeshes(Prototype* prototype)
86{
87 // delete old convex meshes
88 for (int i = 0; i < prototype->getBoundingVolumeCount(); i++) {
89 auto boundingVolume = prototype->getBoundingVolume(i);
90 if (boundingVolume->isGenerated() == false) {
91 continue;
92 } else {
93 if (boundingVolume->getConvexMeshFile().empty() == false &&
94 FileSystem::getInstance()->fileExists(boundingVolume->getConvexMeshFile()) == true) {
95 FileSystem::getInstance()->removeFile(
96 FileSystem::getInstance()->getPathName(boundingVolume->getConvexMeshFile()),
97 FileSystem::getInstance()->getFileName(boundingVolume->getConvexMeshFile())
98 );
99 prototype->removeBoundingVolume(i);
100 i--;
101 } else
102 if (boundingVolume->getConvexMeshData().empty() == false) {
103 prototype->removeBoundingVolume(i);
104 i--;
105 }
106 }
107 }
108}
109
110bool GenerateConvexMeshes::generateConvexMeshes(Prototype* prototype, Mode mode, PopUps* popUps, const string& pathName, const string& fileName, vector<vector<uint8_t>>& convexMeshTMsData, VHACD::IVHACD::Parameters parameters)
111{
112 auto success = true;
113 if (mode == MODE_GENERATE) {
114 class VHACDCallback : public IVHACD::IUserCallback {
115 private:
116 ProgressBarScreenController* progressBarScreenController;
117 public:
118 VHACDCallback(ProgressBarScreenController* progressBarScreenController): progressBarScreenController(progressBarScreenController) {}
119 ~VHACDCallback() {};
120 void Update(
121 const double overallProgress,
122 const double stageProgress,
123 const double operationProgress,
124 const char* const stage,
125 const char* const operation)
126 {
127 progressBarScreenController->progress((int)(overallProgress + 0.5) / 100.0f);
128 };
129 };
130
131 class VHACDLogger : public IVHACD::IUserLogger {
132 public:
133 VHACDLogger() {}
134 ~VHACDLogger() {};
135 void Log(const char* const msg)
136 {
137 Console::println(msg);
138 }
139 };
140
141 //
142 if (popUps != nullptr) popUps->getProgressBarScreenController()->show("Generate convex meshes ...");
143 IVHACD* vhacd = CreateVHACD();
144 try {
145 if (parameters.m_resolution < 10000 || parameters.m_resolution > 64000000) {
146 throw ExceptionBase("Resolution must be between 10000 and 64000000");
147 }
148 if (parameters.m_concavity < 0.0f || parameters.m_concavity > 1.0f) {
149 throw ExceptionBase("Concavity must be between 0.0 and 1.0");
150 }
151 if (parameters.m_planeDownsampling < 1 || parameters.m_planeDownsampling > 16) {
152 throw ExceptionBase("Plane down sampling must be between 1 and 16");
153 }
154 if (parameters.m_convexhullDownsampling < 1 || parameters.m_convexhullDownsampling > 16) {
155 throw ExceptionBase("Convex hull down sampling must be between 1 and 16");
156 }
157 if (parameters.m_alpha < 0.0f || parameters.m_alpha > 1.0f) {
158 throw ExceptionBase("Alpha must be between 0.0 and 1.0");
159 }
160 if (parameters.m_beta < 0.0f || parameters.m_beta > 1.0f) {
161 throw ExceptionBase("Beta must be between 0.0 and 1.0");
162 }
163 if (parameters.m_maxNumVerticesPerCH < 4 || parameters.m_maxNumVerticesPerCH > 1024) {
164 throw ExceptionBase("Max number of vertices per convex hull must be between 4 and 1024");
165 }
166 if (parameters.m_minVolumePerCH < 0.0f || parameters.m_minVolumePerCH > 0.01f) {
167 throw ExceptionBase("Min volume per convex hull must be between 0.0 and 0.01");
168 }
169 if (parameters.m_pca > 1) {
170 throw ExceptionBase("PCA must be between 0 and 1");
171 }
172 VHACDLogger vhacdLogger;
173 parameters.m_logger = &vhacdLogger;
174 if (popUps != nullptr) {
175 VHACDCallback vhacdCallback(popUps->getProgressBarScreenController());
176 parameters.m_callback = &vhacdCallback;
177 }
178 vector<float> meshPoints;
179 vector<int> meshTriangles;
180 auto meshModel = ModelReader::read(
181 pathName,
182 fileName
183 );
184 {
185 Object3DModel meshObject3DModel(meshModel);
186 vector<Triangle> meshFaceTriangles;
187 meshObject3DModel.getTriangles(meshFaceTriangles);
188 for (auto& triangle: meshFaceTriangles) {
189 meshTriangles.push_back(meshPoints.size() / 3 + 0);
190 meshTriangles.push_back(meshPoints.size() / 3 + 1);
191 meshTriangles.push_back(meshPoints.size() / 3 + 2);
192 for (auto i = 0; i < triangle.getVertices().size(); i++) {
193 meshPoints.push_back(triangle.getVertices()[i].getX());
194 meshPoints.push_back(triangle.getVertices()[i].getY());
195 meshPoints.push_back(triangle.getVertices()[i].getZ());
196 }
197 }
198 }
199 delete meshModel;
200 bool vhacdResult =
201 vhacd->Compute(
202 &meshPoints[0],
203 (unsigned int)meshPoints.size() / 3,
204 (const unsigned int *)&meshTriangles[0],
205 (unsigned int)meshTriangles.size() / 3,
206 parameters
207 );
208 if (vhacdResult == true) {
209 auto convexHulls = vhacd->GetNConvexHulls();
210 IVHACD::ConvexHull convexHull;
211 for (auto i = 0; i < convexHulls; i++) {
212 vhacd->GetConvexHull(i, convexHull);
213 auto convexHullFileName = fileName + ".cm." + to_string(i) + ".tm";
214 Console::println(
215 "GenerateConvexMeshes::generateConvexMeshes(): VHACD: Saving convex hull@" +
216 to_string(i) +
217 ": " + convexHullFileName +
218 ", points = " + to_string(convexHull.m_nPoints) +
219 ", triangles = " + to_string(convexHull.m_nTriangles)
220 );
221 auto convexHullModel = createModel(
222 convexHullFileName,
223 convexHull.m_points,
224 convexHull.m_triangles,
225 convexHull.m_nPoints,
226 convexHull.m_nTriangles
227 );
228 convexMeshTMsData.push_back(vector<uint8_t>());
229 TMWriter::write(convexHullModel, convexMeshTMsData[convexMeshTMsData.size() - 1]);
230 delete convexHullModel;
231 }
232 }
233 } catch (Exception &exception) {
234 if (popUps != nullptr) {
236 "Warning: Could not create convex hulls",
237 exception.what()
238 );
239 }
240 Console::println(string("Could not create convex hulls: ") + exception.what());
241 convexMeshTMsData.clear();
242 success = false;
243 }
244 vhacd->Clean();
245 vhacd->Release();
246 if (popUps != nullptr) popUps->getProgressBarScreenController()->close();
247 } else
248 if (mode == MODE_IMPORT) {
249 try {
250 auto meshModel = ModelReader::read(
251 pathName,
252 fileName
253 );
254 {
255 Object3DModel meshObject3DModel(meshModel);
256 for (auto i = 0; i < meshObject3DModel.getNodeCount(); i++) {
257 vector<Triangle> nodeTriangles;
258 meshObject3DModel.getTriangles(nodeTriangles, i);
259 auto convexHullFileName = fileName + ".cm." + to_string(i) + ".tm";
260 Console::println(
261 "GenerateConvexMeshes::generateConvexMeshes(): Model: Saving convex hull@" +
262 to_string(i) +
263 ": " + convexHullFileName +
264 ", triangles = " + to_string(nodeTriangles.size())
265 );
266 auto convexHullModel = createModel(
267 convexHullFileName,
268 nodeTriangles
269 );
270 convexMeshTMsData.push_back(vector<uint8_t>());
271 TMWriter::write(convexHullModel, convexMeshTMsData[convexMeshTMsData.size() - 1]);
272 delete convexHullModel;
273 }
274 }
275 delete meshModel;
276 } catch (Exception &exception) {
277 if (popUps != nullptr) {
279 "Warning: Could not create convex hulls",
280 exception.what()
281 );
282 }
283 Console::println(string("Could not create convex hulls: ") + exception.what());
284 convexMeshTMsData.clear();
285 success = false;
286 }
287 }
288 return success;
289}
290
291Model* GenerateConvexMeshes::createModel(const string& id, double* points, unsigned int* triangles, unsigned int pointCount, unsigned int triangleCount) {
292 auto model = new Model(id, id, UpVector::Y_UP, RotationOrder::XYZ, nullptr);
293 auto material = new Material("primitive");
294 material->setSpecularMaterialProperties(new SpecularMaterialProperties());
295 material->getSpecularMaterialProperties()->setAmbientColor(Color4(0.5f, 0.5f, 0.5f, 1.0f));
296 material->getSpecularMaterialProperties()->setDiffuseColor(Color4(1.0f, 0.5f, 0.5f, 0.5f));
297 material->getSpecularMaterialProperties()->setSpecularColor(Color4(0.0f, 0.0f, 0.0f, 1.0f));
298 model->getMaterials()[material->getId()] = material;
299 auto node = new Node(model, nullptr, "node", "node");
300 vector<Vector3> vertices;
301 vector<Vector3> normals;
302 vector<Face> faces;
303 int normalIndex = -1;
304 for (auto i = 0; i < pointCount; i++) {
305 vertices.push_back(
306 Vector3(
307 static_cast<float>(points[i * 3 + 0]),
308 static_cast<float>(points[i * 3 + 1]),
309 static_cast<float>(points[i * 3 + 2])
310 )
311 );
312 }
313 for (auto i = 0; i < triangleCount; i++) {
314 normalIndex = normals.size();
315 {
316 array<Vector3, 3> faceVertices = {
317 vertices[triangles[i * 3 + 0]],
318 vertices[triangles[i * 3 + 1]],
319 vertices[triangles[i * 3 + 2]]
320 };
321 for (auto& normal: ModelTools::computeNormals(faceVertices)) {
322 normals.push_back(normal);
323 }
324 }
325 faces.push_back(
326 Face(
327 node,
328 triangles[i * 3 + 0],
329 triangles[i * 3 + 1],
330 triangles[i * 3 + 2],
331 normalIndex + 0,
332 normalIndex + 1,
333 normalIndex + 2
334 )
335 );
336 }
337 FacesEntity nodeFacesEntity(node, "faces entity");
338 nodeFacesEntity.setMaterial(material);
339 nodeFacesEntity.setFaces(faces);
340 vector<FacesEntity> nodeFacesEntities;
341 nodeFacesEntities.push_back(nodeFacesEntity);
342 node->setVertices(vertices);
343 node->setNormals(normals);
344 node->setFacesEntities(nodeFacesEntities);
345 model->getNodes()["node"] = node;
346 model->getSubNodes()["node"] = node;
347 ModelTools::prepareForIndexedRendering(model);
348 return model;
349}
350
351Model* GenerateConvexMeshes::createModel(const string& id, vector<Triangle>& triangles) {
352 auto model = new Model(id, id, UpVector::Y_UP, RotationOrder::XYZ, nullptr);
353 auto material = new Material("primitive");
354 material->setSpecularMaterialProperties(new SpecularMaterialProperties());
355 material->getSpecularMaterialProperties()->setAmbientColor(Color4(0.5f, 0.5f, 0.5f, 1.0f));
356 material->getSpecularMaterialProperties()->setDiffuseColor(Color4(1.0f, 0.5f, 0.5f, 0.5f));
357 material->getSpecularMaterialProperties()->setSpecularColor(Color4(0.0f, 0.0f, 0.0f, 1.0f));
358 model->getMaterials()[material->getId()] = material;
359 auto node = new Node(model, nullptr, "node", "node");
360 vector<Vector3> vertices;
361 vector<Vector3> normals;
362 vector<Face> faces;
363 auto index = 0;
364 for (auto& triangle: triangles) {
365 for (auto& vertex: triangle.getVertices()) {
366 vertices.push_back(vertex);
367 }
368 {
369 array<Vector3, 3> faceVertices = {
370 triangle.getVertices()[0],
371 triangle.getVertices()[1],
372 triangle.getVertices()[2],
373 };
374 for (auto& normal: ModelTools::computeNormals(faceVertices)) {
375 normals.push_back(normal);
376 }
377 }
378 faces.push_back(
379 Face(
380 node,
381 index + 0,
382 index + 1,
383 index + 2,
384 index + 0,
385 index + 1,
386 index + 2
387 )
388 );
389 index+= 3;
390 }
391 FacesEntity nodeFacesEntity(node, "faces entity");
392 nodeFacesEntity.setMaterial(material);
393 nodeFacesEntity.setFaces(faces);
394 vector<FacesEntity> nodeFacesEntities;
395 nodeFacesEntities.push_back(nodeFacesEntity);
396 node->setVertices(vertices);
397 node->setNormals(normals);
398 node->setFacesEntities(nodeFacesEntities);
399 model->getNodes()["node"] = node;
400 model->getSubNodes()["node"] = node;
401 ModelTools::prepareForIndexedRendering(model);
402 return model;
403}
Color 4 definition.
Definition: Color4.h:20
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
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
Model node.
Definition: Node.h:31
Represents rotation orders of a model.
Definition: RotationOrder.h:23
Represents specular material properties.
Model up vector.
Definition: UpVector.h:20
Triangle entity, this is not directly connectable with physics engine.
Definition: Triangle.h:19
Prototype definition.
Definition: Prototype.h:49
PrototypeBoundingVolume * getBoundingVolume(int idx)
Get bounding volume at given index.
Definition: Prototype.h:223
void removeBoundingVolume(int idx)
Remove bounding volume at given index.
Definition: Prototype.cpp:85
void getTriangles(vector< Triangle > &triangles, int nodeIdx=-1)
Retrieves list of triangles of all or given nodes.
GUI node controller base class.
GUI node base class.
Definition: GUINode.h:63
GUI parent node base class thats supporting child nodes.
Definition: GUIParentNode.h:43
GUI screen node that represents a screen that can be rendered via GUI system.
Definition: GUIScreenNode.h:57
3D vector 3 class
Definition: Vector3.h:22
File system singleton class.
Definition: FileSystem.h:14
Standard file system implementation.
void show(const string &caption, const string &message)
Shows the pop up.
void show(const string &message, bool showProgressBar=true)
Shows the pop up.
static bool generateConvexMeshes(Prototype *prototype, Mode mode, PopUps *popUps, const string &pathName, const string &fileName, vector< vector< uint8_t > > &convexMeshTMsData, VHACD::IVHACD::Parameters parameters=VHACD::IVHACD::Parameters())
Generateconvex meshes.
static Model * createModel(const string &id, double *points, unsigned int *triangles, unsigned int pointCount, unsigned int triangleCount)
Create model.
Pop ups controller accessor class.
Definition: PopUps.h:19
ProgressBarScreenController * getProgressBarScreenController()
Definition: PopUps.h:56
InfoDialogScreenController * getInfoDialogScreenController()
Definition: PopUps.h:49
Console class.
Definition: Console.h:26
Exception base class.
Definition: ExceptionBase.h:20
Model tools functions class.
Definition: ModelTools.h:38
Mutable string class.
Definition: MutableString.h:16
std::exception Exception
Exception base class.
Definition: Exception.h:19