4#include <unordered_map>
7#include <ext/v-hacd/src/VHACD_Lib/public/VHACD.h>
47using std::unordered_map;
85void GenerateConvexMeshes::removeConvexMeshes(
Prototype* prototype)
90 if (boundingVolume->isGenerated() ==
false) {
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())
102 if (boundingVolume->getConvexMeshData().empty() ==
false) {
114 class VHACDCallback :
public IVHACD::IUserCallback {
118 VHACDCallback(
ProgressBarScreenController* progressBarScreenController): progressBarScreenController(progressBarScreenController) {}
121 const double overallProgress,
122 const double stageProgress,
123 const double operationProgress,
124 const char*
const stage,
125 const char*
const operation)
127 progressBarScreenController->
progress((
int)(overallProgress + 0.5) / 100.0f);
131 class VHACDLogger :
public IVHACD::IUserLogger {
135 void Log(
const char*
const msg)
137 Console::println(msg);
143 IVHACD* vhacd = CreateVHACD();
145 if (parameters.m_resolution < 10000 || parameters.m_resolution > 64000000) {
146 throw ExceptionBase(
"Resolution must be between 10000 and 64000000");
148 if (parameters.m_concavity < 0.0f || parameters.m_concavity > 1.0f) {
149 throw ExceptionBase(
"Concavity must be between 0.0 and 1.0");
151 if (parameters.m_planeDownsampling < 1 || parameters.m_planeDownsampling > 16) {
152 throw ExceptionBase(
"Plane down sampling must be between 1 and 16");
154 if (parameters.m_convexhullDownsampling < 1 || parameters.m_convexhullDownsampling > 16) {
155 throw ExceptionBase(
"Convex hull down sampling must be between 1 and 16");
157 if (parameters.m_alpha < 0.0f || parameters.m_alpha > 1.0f) {
160 if (parameters.m_beta < 0.0f || parameters.m_beta > 1.0f) {
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");
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");
169 if (parameters.m_pca > 1) {
172 VHACDLogger vhacdLogger;
173 parameters.m_logger = &vhacdLogger;
174 if (popUps !=
nullptr) {
176 parameters.m_callback = &vhacdCallback;
178 vector<float> meshPoints;
179 vector<int> meshTriangles;
180 auto meshModel = ModelReader::read(
186 vector<Triangle> 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());
203 (
unsigned int)meshPoints.size() / 3,
204 (
const unsigned int *)&meshTriangles[0],
205 (
unsigned int)meshTriangles.size() / 3,
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";
215 "GenerateConvexMeshes::generateConvexMeshes(): VHACD: Saving convex hull@" +
217 ": " + convexHullFileName +
218 ", points = " + to_string(convexHull.m_nPoints) +
219 ", triangles = " + to_string(convexHull.m_nTriangles)
224 convexHull.m_triangles,
225 convexHull.m_nPoints,
226 convexHull.m_nTriangles
228 convexMeshTMsData.push_back(vector<uint8_t>());
229 TMWriter::write(convexHullModel, convexMeshTMsData[convexMeshTMsData.size() - 1]);
230 delete convexHullModel;
234 if (popUps !=
nullptr) {
236 "Warning: Could not create convex hulls",
240 Console::println(
string(
"Could not create convex hulls: ") + exception.what());
241 convexMeshTMsData.clear();
250 auto meshModel = ModelReader::read(
256 for (
auto i = 0; i < meshObject3DModel.
getNodeCount(); i++) {
257 vector<Triangle> nodeTriangles;
259 auto convexHullFileName = fileName +
".cm." + to_string(i) +
".tm";
261 "GenerateConvexMeshes::generateConvexMeshes(): Model: Saving convex hull@" +
263 ": " + convexHullFileName +
264 ", triangles = " + to_string(nodeTriangles.size())
270 convexMeshTMsData.push_back(vector<uint8_t>());
271 TMWriter::write(convexHullModel, convexMeshTMsData[convexMeshTMsData.size() - 1]);
272 delete convexHullModel;
277 if (popUps !=
nullptr) {
279 "Warning: Could not create convex hulls",
283 Console::println(
string(
"Could not create convex hulls: ") + exception.what());
284 convexMeshTMsData.clear();
292 auto model =
new Model(
id,
id, UpVector::Y_UP, RotationOrder::XYZ,
nullptr);
293 auto material =
new Material(
"primitive");
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;
303 int normalIndex = -1;
304 for (
auto i = 0; i < pointCount; i++) {
307 static_cast<float>(points[i * 3 + 0]),
308 static_cast<float>(points[i * 3 + 1]),
309 static_cast<float>(points[i * 3 + 2])
313 for (
auto i = 0; i < triangleCount; i++) {
314 normalIndex = normals.size();
316 array<Vector3, 3> faceVertices = {
317 vertices[triangles[i * 3 + 0]],
318 vertices[triangles[i * 3 + 1]],
319 vertices[triangles[i * 3 + 2]]
321 for (
auto& normal: ModelTools::computeNormals(faceVertices)) {
322 normals.push_back(normal);
328 triangles[i * 3 + 0],
329 triangles[i * 3 + 1],
330 triangles[i * 3 + 2],
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);
352 auto model =
new Model(
id,
id, UpVector::Y_UP, RotationOrder::XYZ,
nullptr);
353 auto material =
new Material(
"primitive");
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;
364 for (
auto& triangle: triangles) {
365 for (
auto& vertex: triangle.getVertices()) {
366 vertices.push_back(vertex);
369 array<Vector3, 3> faceVertices = {
370 triangle.getVertices()[0],
371 triangle.getVertices()[1],
372 triangle.getVertices()[2],
374 for (
auto& normal: ModelTools::computeNormals(faceVertices)) {
375 normals.push_back(normal);
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);
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.
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Representation of a 3d model.
Represents rotation orders of a model.
Represents specular material properties.
Triangle entity, this is not directly connectable with physics engine.
Prototype bounding volume definition.
int getBoundingVolumeCount()
PrototypeBoundingVolume * getBoundingVolume(int idx)
Get bounding volume at given index.
void removeBoundingVolume(int idx)
Remove bounding volume at given index.
void getTriangles(vector< Triangle > &triangles, int nodeIdx=-1)
Retrieves list of triangles of all or given nodes.
GUI node controller base class.
GUI parent node base class thats supporting child nodes.
GUI screen node that represents a screen that can be rendered via GUI system.
File system singleton class.
Standard file system implementation.
std::exception Exception
Exception base class.