mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-10-23 16:51:21 -06:00
New tech ENABLE_SMOOTH_NORMALS (disabled) -> Added two experimental functions to smooth normals using libigl (none of them working properly in detecting edges) when calling GLIndexedVertexArray::load_mesh_full_shading()
This commit is contained in:
parent
21de9d23ec
commit
aa92cbf051
4 changed files with 131 additions and 13 deletions
|
@ -45,5 +45,8 @@
|
||||||
// Enable rendering of objects using environment map
|
// Enable rendering of objects using environment map
|
||||||
#define ENABLE_ENVIRONMENT_MAP (1 && ENABLE_2_3_0_ALPHA1)
|
#define ENABLE_ENVIRONMENT_MAP (1 && ENABLE_2_3_0_ALPHA1)
|
||||||
|
|
||||||
|
// Enable smoothing of objects normals
|
||||||
|
#define ENABLE_SMOOTH_NORMALS (0 && ENABLE_2_3_0_ALPHA1)
|
||||||
|
|
||||||
|
|
||||||
#endif // _prusaslicer_technologies_h_
|
#endif // _prusaslicer_technologies_h_
|
||||||
|
|
|
@ -1,5 +1,11 @@
|
||||||
#include <GL/glew.h>
|
#include <GL/glew.h>
|
||||||
|
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
#include <igl/per_face_normals.h>
|
||||||
|
#include <igl/per_corner_normals.h>
|
||||||
|
#include <igl/per_vertex_normals.h>
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
|
||||||
#include "3DScene.hpp"
|
#include "3DScene.hpp"
|
||||||
#if ENABLE_ENVIRONMENT_MAP
|
#if ENABLE_ENVIRONMENT_MAP
|
||||||
#include "GUI_App.hpp"
|
#include "GUI_App.hpp"
|
||||||
|
@ -57,23 +63,107 @@ void glAssertRecentCallImpl(const char *file_name, unsigned int line, const char
|
||||||
|
|
||||||
namespace Slic3r {
|
namespace Slic3r {
|
||||||
|
|
||||||
void GLIndexedVertexArray::load_mesh_full_shading(const TriangleMesh &mesh)
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
static void smooth_normals_corner(TriangleMesh& mesh, std::vector<stl_normal>& normals)
|
||||||
|
{
|
||||||
|
mesh.repair();
|
||||||
|
|
||||||
|
using MapMatrixXfUnaligned = Eigen::Map<const Eigen::Matrix<float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>>;
|
||||||
|
using MapMatrixXiUnaligned = Eigen::Map<const Eigen::Matrix<int, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>>;
|
||||||
|
|
||||||
|
std::vector<stl_normal> face_normals(mesh.stl.stats.number_of_facets);
|
||||||
|
for (uint32_t i = 0; i < mesh.stl.stats.number_of_facets; ++i) {
|
||||||
|
face_normals[i] = mesh.stl.facet_start[i].normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
Eigen::MatrixXd vertices = MapMatrixXfUnaligned(mesh.its.vertices.front().data(),
|
||||||
|
Eigen::Index(mesh.its.vertices.size()), 3).cast<double>();
|
||||||
|
Eigen::MatrixXi indices = MapMatrixXiUnaligned(mesh.its.indices.front().data(),
|
||||||
|
Eigen::Index(mesh.its.indices.size()), 3);
|
||||||
|
Eigen::MatrixXd in_normals = MapMatrixXfUnaligned(face_normals.front().data(),
|
||||||
|
Eigen::Index(face_normals.size()), 3).cast<double>();
|
||||||
|
Eigen::MatrixXd out_normals;
|
||||||
|
|
||||||
|
igl::per_corner_normals(vertices, indices, in_normals, 1.0, out_normals);
|
||||||
|
|
||||||
|
normals = std::vector<stl_normal>(mesh.its.vertices.size());
|
||||||
|
for (size_t i = 0; i < mesh.its.indices.size(); ++i) {
|
||||||
|
for (size_t j = 0; j < 3; ++j) {
|
||||||
|
normals[mesh.its.indices[i][j]] = out_normals.row(i * 3 + j).cast<float>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void smooth_normals_vertex(TriangleMesh& mesh, std::vector<stl_normal>& normals)
|
||||||
|
{
|
||||||
|
mesh.repair();
|
||||||
|
|
||||||
|
using MapMatrixXfUnaligned = Eigen::Map<const Eigen::Matrix<float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>>;
|
||||||
|
using MapMatrixXiUnaligned = Eigen::Map<const Eigen::Matrix<int, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>>;
|
||||||
|
|
||||||
|
Eigen::MatrixXd vertices = MapMatrixXfUnaligned(mesh.its.vertices.front().data(),
|
||||||
|
Eigen::Index(mesh.its.vertices.size()), 3).cast<double>();
|
||||||
|
Eigen::MatrixXi indices = MapMatrixXiUnaligned(mesh.its.indices.front().data(),
|
||||||
|
Eigen::Index(mesh.its.indices.size()), 3);
|
||||||
|
Eigen::MatrixXd out_normals;
|
||||||
|
|
||||||
|
// igl::per_vertex_normals(vertices, indices, igl::PER_VERTEX_NORMALS_WEIGHTING_TYPE_UNIFORM, out_normals);
|
||||||
|
// igl::per_vertex_normals(vertices, indices, igl::PER_VERTEX_NORMALS_WEIGHTING_TYPE_AREA, out_normals);
|
||||||
|
igl::per_vertex_normals(vertices, indices, igl::PER_VERTEX_NORMALS_WEIGHTING_TYPE_ANGLE, out_normals);
|
||||||
|
// igl::per_vertex_normals(vertices, indices, igl::PER_VERTEX_NORMALS_WEIGHTING_TYPE_DEFAULT, out_normals);
|
||||||
|
|
||||||
|
normals = std::vector<stl_normal>(mesh.its.vertices.size());
|
||||||
|
for (size_t i = 0; i < static_cast<size_t>(out_normals.rows()); ++i) {
|
||||||
|
normals[i] = out_normals.row(i).cast<float>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
void GLIndexedVertexArray::load_mesh_full_shading(const TriangleMesh& mesh, bool smooth_normals)
|
||||||
|
#else
|
||||||
|
void GLIndexedVertexArray::load_mesh_full_shading(const TriangleMesh& mesh)
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
{
|
{
|
||||||
assert(triangle_indices.empty() && vertices_and_normals_interleaved_size == 0);
|
assert(triangle_indices.empty() && vertices_and_normals_interleaved_size == 0);
|
||||||
assert(quad_indices.empty() && triangle_indices_size == 0);
|
assert(quad_indices.empty() && triangle_indices_size == 0);
|
||||||
assert(vertices_and_normals_interleaved.size() % 6 == 0 && quad_indices_size == vertices_and_normals_interleaved.size());
|
assert(vertices_and_normals_interleaved.size() % 6 == 0 && quad_indices_size == vertices_and_normals_interleaved.size());
|
||||||
|
|
||||||
this->vertices_and_normals_interleaved.reserve(this->vertices_and_normals_interleaved.size() + 3 * 3 * 2 * mesh.facets_count());
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
if (smooth_normals) {
|
||||||
|
TriangleMesh new_mesh(mesh);
|
||||||
|
std::vector<stl_normal> normals;
|
||||||
|
smooth_normals_corner(new_mesh, normals);
|
||||||
|
// smooth_normals_vertex(new_mesh, normals);
|
||||||
|
|
||||||
unsigned int vertices_count = 0;
|
this->vertices_and_normals_interleaved.reserve(this->vertices_and_normals_interleaved.size() + 3 * 2 * new_mesh.its.vertices.size());
|
||||||
for (int i = 0; i < (int)mesh.stl.stats.number_of_facets; ++i) {
|
for (size_t i = 0; i < new_mesh.its.vertices.size(); ++i) {
|
||||||
const stl_facet &facet = mesh.stl.facet_start[i];
|
const stl_vertex& v = new_mesh.its.vertices[i];
|
||||||
for (int j = 0; j < 3; ++j)
|
const stl_normal& n = normals[i];
|
||||||
this->push_geometry(facet.vertex[j](0), facet.vertex[j](1), facet.vertex[j](2), facet.normal(0), facet.normal(1), facet.normal(2));
|
this->push_geometry(v(0), v(1), v(2), n(0), n(1), n(2));
|
||||||
|
}
|
||||||
|
|
||||||
this->push_triangle(vertices_count, vertices_count + 1, vertices_count + 2);
|
for (size_t i = 0; i < new_mesh.its.indices.size(); ++i) {
|
||||||
vertices_count += 3;
|
const stl_triangle_vertex_indices& idx = new_mesh.its.indices[i];
|
||||||
|
this->push_triangle(idx(0), idx(1), idx(2));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
this->vertices_and_normals_interleaved.reserve(this->vertices_and_normals_interleaved.size() + 3 * 3 * 2 * mesh.facets_count());
|
||||||
|
|
||||||
|
unsigned int vertices_count = 0;
|
||||||
|
for (int i = 0; i < (int)mesh.stl.stats.number_of_facets; ++i) {
|
||||||
|
const stl_facet& facet = mesh.stl.facet_start[i];
|
||||||
|
for (int j = 0; j < 3; ++j)
|
||||||
|
this->push_geometry(facet.vertex[j](0), facet.vertex[j](1), facet.vertex[j](2), facet.normal(0), facet.normal(1), facet.normal(2));
|
||||||
|
|
||||||
|
this->push_triangle(vertices_count, vertices_count + 1, vertices_count + 2);
|
||||||
|
vertices_count += 3;
|
||||||
|
}
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
}
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
}
|
}
|
||||||
|
|
||||||
void GLIndexedVertexArray::finalize_geometry(bool opengl_initialized)
|
void GLIndexedVertexArray::finalize_geometry(bool opengl_initialized)
|
||||||
|
@ -462,7 +552,11 @@ int GLVolumeCollection::load_object_volume(
|
||||||
this->volumes.emplace_back(new GLVolume(color));
|
this->volumes.emplace_back(new GLVolume(color));
|
||||||
GLVolume& v = *this->volumes.back();
|
GLVolume& v = *this->volumes.back();
|
||||||
v.set_color_from_model_volume(model_volume);
|
v.set_color_from_model_volume(model_volume);
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
v.indexed_vertex_array.load_mesh(mesh, true);
|
||||||
|
#else
|
||||||
v.indexed_vertex_array.load_mesh(mesh);
|
v.indexed_vertex_array.load_mesh(mesh);
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
v.indexed_vertex_array.finalize_geometry(opengl_initialized);
|
v.indexed_vertex_array.finalize_geometry(opengl_initialized);
|
||||||
v.composite_id = GLVolume::CompositeID(obj_idx, volume_idx, instance_idx);
|
v.composite_id = GLVolume::CompositeID(obj_idx, volume_idx, instance_idx);
|
||||||
if (model_volume->is_model_part())
|
if (model_volume->is_model_part())
|
||||||
|
@ -504,8 +598,12 @@ void GLVolumeCollection::load_object_auxiliary(
|
||||||
const ModelInstance& model_instance = *print_object->model_object()->instances[instance_idx.first];
|
const ModelInstance& model_instance = *print_object->model_object()->instances[instance_idx.first];
|
||||||
this->volumes.emplace_back(new GLVolume((milestone == slaposPad) ? GLVolume::SLA_PAD_COLOR : GLVolume::SLA_SUPPORT_COLOR));
|
this->volumes.emplace_back(new GLVolume((milestone == slaposPad) ? GLVolume::SLA_PAD_COLOR : GLVolume::SLA_SUPPORT_COLOR));
|
||||||
GLVolume& v = *this->volumes.back();
|
GLVolume& v = *this->volumes.back();
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
v.indexed_vertex_array.load_mesh(mesh, true);
|
||||||
|
#else
|
||||||
v.indexed_vertex_array.load_mesh(mesh);
|
v.indexed_vertex_array.load_mesh(mesh);
|
||||||
v.indexed_vertex_array.finalize_geometry(opengl_initialized);
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
v.indexed_vertex_array.finalize_geometry(opengl_initialized);
|
||||||
v.composite_id = GLVolume::CompositeID(obj_idx, -int(milestone), (int)instance_idx.first);
|
v.composite_id = GLVolume::CompositeID(obj_idx, -int(milestone), (int)instance_idx.first);
|
||||||
v.geometry_id = std::pair<size_t, size_t>(timestamp, model_instance.id().id);
|
v.geometry_id = std::pair<size_t, size_t>(timestamp, model_instance.id().id);
|
||||||
// Create a copy of the convex hull mesh for each instance. Use a move operator on the last instance.
|
// Create a copy of the convex hull mesh for each instance. Use a move operator on the last instance.
|
||||||
|
|
|
@ -125,8 +125,13 @@ public:
|
||||||
unsigned int triangle_indices_VBO_id{ 0 };
|
unsigned int triangle_indices_VBO_id{ 0 };
|
||||||
unsigned int quad_indices_VBO_id{ 0 };
|
unsigned int quad_indices_VBO_id{ 0 };
|
||||||
|
|
||||||
void load_mesh_full_shading(const TriangleMesh &mesh);
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
void load_mesh_full_shading(const TriangleMesh& mesh, bool smooth_normals = false);
|
||||||
|
void load_mesh(const TriangleMesh& mesh, bool smooth_normals = false) { this->load_mesh_full_shading(mesh, smooth_normals); }
|
||||||
|
#else
|
||||||
|
void load_mesh_full_shading(const TriangleMesh& mesh);
|
||||||
void load_mesh(const TriangleMesh& mesh) { this->load_mesh_full_shading(mesh); }
|
void load_mesh(const TriangleMesh& mesh) { this->load_mesh_full_shading(mesh); }
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
|
||||||
inline bool has_VBOs() const { return vertices_and_normals_interleaved_VBO_id != 0; }
|
inline bool has_VBOs() const { return vertices_and_normals_interleaved_VBO_id != 0; }
|
||||||
|
|
||||||
|
|
|
@ -2491,11 +2491,19 @@ void GLCanvas3D::reload_scene(bool refresh_immediately, bool force_full_scene_re
|
||||||
TriangleMesh mesh = print_object->get_mesh(slaposDrillHoles);
|
TriangleMesh mesh = print_object->get_mesh(slaposDrillHoles);
|
||||||
assert(! mesh.empty());
|
assert(! mesh.empty());
|
||||||
mesh.transform(sla_print->sla_trafo(*m_model->objects[volume.object_idx()]).inverse());
|
mesh.transform(sla_print->sla_trafo(*m_model->objects[volume.object_idx()]).inverse());
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
volume.indexed_vertex_array.load_mesh(mesh, true);
|
||||||
|
#else
|
||||||
volume.indexed_vertex_array.load_mesh(mesh);
|
volume.indexed_vertex_array.load_mesh(mesh);
|
||||||
} else {
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
} else {
|
||||||
// Reload the original volume.
|
// Reload the original volume.
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
volume.indexed_vertex_array.load_mesh(m_model->objects[volume.object_idx()]->volumes[volume.volume_idx()]->mesh(), true);
|
||||||
|
#else
|
||||||
volume.indexed_vertex_array.load_mesh(m_model->objects[volume.object_idx()]->volumes[volume.volume_idx()]->mesh());
|
volume.indexed_vertex_array.load_mesh(m_model->objects[volume.object_idx()]->volumes[volume.volume_idx()]->mesh());
|
||||||
}
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
|
}
|
||||||
volume.finalize_geometry(true);
|
volume.finalize_geometry(true);
|
||||||
}
|
}
|
||||||
//FIXME it is an ugly hack to write the timestamp into the "offsets" field to not have to add another member variable
|
//FIXME it is an ugly hack to write the timestamp into the "offsets" field to not have to add another member variable
|
||||||
|
@ -6814,7 +6822,11 @@ void GLCanvas3D::_load_sla_shells()
|
||||||
const TriangleMesh &mesh, const float color[4], bool outside_printer_detection_enabled) {
|
const TriangleMesh &mesh, const float color[4], bool outside_printer_detection_enabled) {
|
||||||
m_volumes.volumes.emplace_back(new GLVolume(color));
|
m_volumes.volumes.emplace_back(new GLVolume(color));
|
||||||
GLVolume& v = *m_volumes.volumes.back();
|
GLVolume& v = *m_volumes.volumes.back();
|
||||||
|
#if ENABLE_SMOOTH_NORMALS
|
||||||
|
v.indexed_vertex_array.load_mesh(mesh, true);
|
||||||
|
#else
|
||||||
v.indexed_vertex_array.load_mesh(mesh);
|
v.indexed_vertex_array.load_mesh(mesh);
|
||||||
|
#endif // ENABLE_SMOOTH_NORMALS
|
||||||
v.indexed_vertex_array.finalize_geometry(this->m_initialized);
|
v.indexed_vertex_array.finalize_geometry(this->m_initialized);
|
||||||
v.shader_outside_printer_detection_enabled = outside_printer_detection_enabled;
|
v.shader_outside_printer_detection_enabled = outside_printer_detection_enabled;
|
||||||
v.composite_id.volume_id = volume_id;
|
v.composite_id.volume_id = volume_id;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue