2019-09-13 09:30:50 +00:00
|
|
|
#include "MeshUtils.hpp"
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
#include "libslic3r/Tesselate.hpp"
|
2019-09-13 09:30:50 +00:00
|
|
|
#include "libslic3r/TriangleMesh.hpp"
|
2019-09-12 12:58:03 +00:00
|
|
|
|
2019-09-17 07:17:53 +00:00
|
|
|
#include "slic3r/GUI/Camera.hpp"
|
|
|
|
|
|
|
|
// There is an L function in igl that would be overridden by our localization macro.
|
|
|
|
#undef L
|
|
|
|
#include <igl/AABB.h>
|
|
|
|
|
|
|
|
#include <GL/glew.h>
|
|
|
|
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
|
|
|
void MeshClipper::set_plane(const ClippingPlane& plane)
|
|
|
|
{
|
|
|
|
if (m_plane != plane) {
|
|
|
|
m_plane = plane;
|
|
|
|
m_triangles_valid = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-13 09:30:50 +00:00
|
|
|
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
void MeshClipper::set_mesh(const TriangleMesh& mesh)
|
|
|
|
{
|
|
|
|
if (m_mesh != &mesh) {
|
|
|
|
m_mesh = &mesh;
|
|
|
|
m_triangles_valid = false;
|
2019-09-12 14:57:30 +00:00
|
|
|
m_triangles2d.resize(0);
|
|
|
|
m_triangles3d.resize(0);
|
2019-09-12 12:58:03 +00:00
|
|
|
m_tms.reset(nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-13 09:30:50 +00:00
|
|
|
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
void MeshClipper::set_transformation(const Geometry::Transformation& trafo)
|
|
|
|
{
|
|
|
|
if (! m_trafo.get_matrix().isApprox(trafo.get_matrix())) {
|
|
|
|
m_trafo = trafo;
|
|
|
|
m_triangles_valid = false;
|
2019-09-12 14:57:30 +00:00
|
|
|
m_triangles2d.resize(0);
|
|
|
|
m_triangles3d.resize(0);
|
2019-09-12 12:58:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-12 14:57:30 +00:00
|
|
|
|
|
|
|
const std::vector<Vec3f>& MeshClipper::get_triangles()
|
2019-09-12 12:58:03 +00:00
|
|
|
{
|
|
|
|
if (! m_triangles_valid)
|
|
|
|
recalculate_triangles();
|
|
|
|
|
2019-09-12 14:57:30 +00:00
|
|
|
return m_triangles3d;
|
2019-09-12 12:58:03 +00:00
|
|
|
}
|
|
|
|
|
2019-09-13 09:30:50 +00:00
|
|
|
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
void MeshClipper::recalculate_triangles()
|
|
|
|
{
|
|
|
|
if (! m_tms) {
|
|
|
|
m_tms.reset(new TriangleMeshSlicer);
|
|
|
|
m_tms->init(m_mesh, [](){});
|
|
|
|
}
|
|
|
|
|
2019-09-12 14:57:30 +00:00
|
|
|
const Transform3f& instance_matrix_no_translation_no_scaling = m_trafo.get_matrix(true,false,true).cast<float>();
|
|
|
|
const Vec3f& scaling = m_trafo.get_scaling_factor().cast<float>();
|
|
|
|
// Calculate clipping plane normal in mesh coordinates.
|
|
|
|
Vec3f up_noscale = instance_matrix_no_translation_no_scaling.inverse() * m_plane.get_normal().cast<float>();
|
|
|
|
Vec3f up (up_noscale(0)*scaling(0), up_noscale(1)*scaling(1), up_noscale(2)*scaling(2));
|
|
|
|
// Calculate distance from mesh origin to the clipping plane (in mesh coordinates).
|
|
|
|
float height_mesh = m_plane.distance(m_trafo.get_offset()) * (up_noscale.norm()/up.norm());
|
2019-09-12 12:58:03 +00:00
|
|
|
|
2019-09-12 14:57:30 +00:00
|
|
|
// Now do the cutting
|
2019-09-12 12:58:03 +00:00
|
|
|
std::vector<ExPolygons> list_of_expolys;
|
|
|
|
m_tms->set_up_direction(up);
|
|
|
|
m_tms->slice(std::vector<float>{height_mesh}, 0.f, &list_of_expolys, [](){});
|
2019-09-12 14:57:30 +00:00
|
|
|
m_triangles2d = triangulate_expolygons_2f(list_of_expolys[0], m_trafo.get_matrix().matrix().determinant() < 0.);
|
|
|
|
|
|
|
|
// Rotate the cut into world coords:
|
|
|
|
Eigen::Quaternionf q;
|
|
|
|
q.setFromTwoVectors(Vec3f::UnitZ(), up);
|
|
|
|
Transform3f tr = Transform3f::Identity();
|
|
|
|
tr.rotate(q);
|
|
|
|
tr = m_trafo.get_matrix().cast<float>() * tr;
|
|
|
|
|
|
|
|
m_triangles3d.clear();
|
|
|
|
m_triangles3d.reserve(m_triangles2d.size());
|
|
|
|
for (const Vec2f& pt : m_triangles2d) {
|
|
|
|
m_triangles3d.push_back(Vec3f(pt(0), pt(1), height_mesh+0.001f));
|
|
|
|
m_triangles3d.back() = tr * m_triangles3d.back();
|
|
|
|
}
|
2019-09-12 12:58:03 +00:00
|
|
|
|
|
|
|
m_triangles_valid = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-17 07:17:53 +00:00
|
|
|
class MeshRaycaster::AABBWrapper {
|
|
|
|
public:
|
|
|
|
AABBWrapper(const TriangleMesh* mesh);
|
|
|
|
~AABBWrapper() { m_AABB.deinit(); }
|
|
|
|
|
|
|
|
typedef Eigen::Map<const Eigen::Matrix<float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>> MapMatrixXfUnaligned;
|
|
|
|
typedef Eigen::Map<const Eigen::Matrix<int, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>> MapMatrixXiUnaligned;
|
|
|
|
igl::AABB<MapMatrixXfUnaligned, 3> m_AABB;
|
|
|
|
|
2019-09-17 08:47:01 +00:00
|
|
|
Vec3f get_hit_pos(const igl::Hit& hit) const;
|
|
|
|
Vec3f get_hit_normal(const igl::Hit& hit) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
const TriangleMesh* m_mesh;
|
2019-09-17 07:17:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
MeshRaycaster::AABBWrapper::AABBWrapper(const TriangleMesh* mesh)
|
2019-09-17 08:47:01 +00:00
|
|
|
: m_mesh(mesh)
|
2019-09-17 07:17:53 +00:00
|
|
|
{
|
|
|
|
m_AABB.init(
|
2019-09-17 08:47:01 +00:00
|
|
|
MapMatrixXfUnaligned(m_mesh->its.vertices.front().data(), m_mesh->its.vertices.size(), 3),
|
|
|
|
MapMatrixXiUnaligned(m_mesh->its.indices.front().data(), m_mesh->its.indices.size(), 3));
|
2019-09-17 07:17:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MeshRaycaster::MeshRaycaster(const TriangleMesh& mesh)
|
|
|
|
: m_AABB_wrapper(new AABBWrapper(&mesh)), m_mesh(&mesh)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
MeshRaycaster::~MeshRaycaster()
|
|
|
|
{
|
|
|
|
delete m_AABB_wrapper;
|
|
|
|
}
|
|
|
|
|
2019-09-17 08:47:01 +00:00
|
|
|
Vec3f MeshRaycaster::AABBWrapper::get_hit_pos(const igl::Hit& hit) const
|
|
|
|
{
|
|
|
|
const stl_triangle_vertex_indices& indices = m_mesh->its.indices[hit.id];
|
|
|
|
return Vec3f((1-hit.u-hit.v) * m_mesh->its.vertices[indices(0)]
|
|
|
|
+ hit.u * m_mesh->its.vertices[indices(1)]
|
|
|
|
+ hit.v * m_mesh->its.vertices[indices(2)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Vec3f MeshRaycaster::AABBWrapper::get_hit_normal(const igl::Hit& hit) const
|
|
|
|
{
|
|
|
|
const stl_triangle_vertex_indices& indices = m_mesh->its.indices[hit.id];
|
|
|
|
Vec3f a(m_mesh->its.vertices[indices(1)] - m_mesh->its.vertices[indices(0)]);
|
|
|
|
Vec3f b(m_mesh->its.vertices[indices(2)] - m_mesh->its.vertices[indices(0)]);
|
|
|
|
return Vec3f(a.cross(b));
|
|
|
|
}
|
|
|
|
|
2019-09-17 07:17:53 +00:00
|
|
|
|
2019-09-24 10:48:05 +00:00
|
|
|
bool MeshRaycaster::unproject_on_mesh(const Vec2d& mouse_pos, const Transform3d& trafo, const Camera& camera,
|
|
|
|
Vec3f& position, Vec3f& normal, const ClippingPlane* clipping_plane) const
|
2019-09-17 07:17:53 +00:00
|
|
|
{
|
|
|
|
const std::array<int, 4>& viewport = camera.get_viewport();
|
|
|
|
const Transform3d& model_mat = camera.get_view_matrix();
|
|
|
|
const Transform3d& proj_mat = camera.get_projection_matrix();
|
|
|
|
|
|
|
|
Vec3d pt1;
|
|
|
|
Vec3d pt2;
|
2019-09-17 08:47:01 +00:00
|
|
|
::gluUnProject(mouse_pos(0), viewport[3] - mouse_pos(1), 0., model_mat.data(), proj_mat.data(), viewport.data(), &pt1(0), &pt1(1), &pt1(2));
|
|
|
|
::gluUnProject(mouse_pos(0), viewport[3] - mouse_pos(1), 1., model_mat.data(), proj_mat.data(), viewport.data(), &pt2(0), &pt2(1), &pt2(2));
|
2019-09-17 07:17:53 +00:00
|
|
|
|
|
|
|
std::vector<igl::Hit> hits;
|
|
|
|
|
|
|
|
Transform3d inv = trafo.inverse();
|
|
|
|
|
|
|
|
pt1 = inv * pt1;
|
|
|
|
pt2 = inv * pt2;
|
|
|
|
|
|
|
|
if (! m_AABB_wrapper->m_AABB.intersect_ray(
|
|
|
|
AABBWrapper::MapMatrixXfUnaligned(m_mesh->its.vertices.front().data(), m_mesh->its.vertices.size(), 3),
|
|
|
|
AABBWrapper::MapMatrixXiUnaligned(m_mesh->its.indices.front().data(), m_mesh->its.indices.size(), 3),
|
|
|
|
pt1.cast<float>(), (pt2-pt1).cast<float>(), hits))
|
|
|
|
return false; // no intersection found
|
|
|
|
|
|
|
|
std::sort(hits.begin(), hits.end(), [](const igl::Hit& a, const igl::Hit& b) { return a.t < b.t; });
|
|
|
|
|
2019-09-24 10:48:05 +00:00
|
|
|
unsigned i = 0;
|
|
|
|
|
|
|
|
// Remove points that are obscured or cut by the clipping plane
|
|
|
|
if (clipping_plane) {
|
|
|
|
for (i=0; i<hits.size(); ++i)
|
|
|
|
if (! clipping_plane->is_point_clipped(trafo * m_AABB_wrapper->get_hit_pos(hits[i]).cast<double>()))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (i==hits.size() || (hits.size()-i) % 2 != 0) {
|
|
|
|
// All hits are either clipped, or there is an odd number of unclipped
|
|
|
|
// hits - meaning the nearest must be from inside the mesh.
|
|
|
|
return false;
|
2019-09-17 07:17:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-24 10:48:05 +00:00
|
|
|
// Now stuff the points in the provided vector and calculate normals if asked about them:
|
|
|
|
position = m_AABB_wrapper->get_hit_pos(hits[i]);
|
|
|
|
normal = m_AABB_wrapper->get_hit_normal(hits[i]);
|
2019-09-17 07:17:53 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-17 12:14:26 +00:00
|
|
|
std::vector<unsigned> MeshRaycaster::get_unobscured_idxs(const Geometry::Transformation& trafo, const Camera& camera, const std::vector<Vec3f>& points,
|
2019-09-24 10:48:05 +00:00
|
|
|
const ClippingPlane* clipping_plane) const
|
2019-09-17 08:47:01 +00:00
|
|
|
{
|
2019-09-17 12:14:26 +00:00
|
|
|
std::vector<unsigned> out;
|
|
|
|
|
|
|
|
const Transform3d& instance_matrix_no_translation_no_scaling = trafo.get_matrix(true,false,true);
|
|
|
|
Vec3f direction_to_camera = -camera.get_dir_forward().cast<float>();
|
|
|
|
Vec3f direction_to_camera_mesh = (instance_matrix_no_translation_no_scaling.inverse().cast<float>() * direction_to_camera).normalized().eval();
|
|
|
|
Vec3f scaling = trafo.get_scaling_factor().cast<float>();
|
|
|
|
direction_to_camera_mesh = Vec3f(direction_to_camera_mesh(0)*scaling(0), direction_to_camera_mesh(1)*scaling(1), direction_to_camera_mesh(2)*scaling(2));
|
2019-09-24 10:48:05 +00:00
|
|
|
const Transform3f inverse_trafo = trafo.get_matrix().inverse().cast<float>();
|
2019-09-17 12:14:26 +00:00
|
|
|
|
|
|
|
for (size_t i=0; i<points.size(); ++i) {
|
|
|
|
const Vec3f& pt = points[i];
|
2019-09-24 10:48:05 +00:00
|
|
|
if (clipping_plane && clipping_plane->is_point_clipped(pt.cast<double>()))
|
|
|
|
continue;
|
|
|
|
|
2019-09-17 12:14:26 +00:00
|
|
|
bool is_obscured = false;
|
|
|
|
// Cast a ray in the direction of the camera and look for intersection with the mesh:
|
|
|
|
std::vector<igl::Hit> hits;
|
2019-09-24 10:48:05 +00:00
|
|
|
// Offset the start of the ray by EPSILON to account for numerical inaccuracies.
|
2019-09-17 12:14:26 +00:00
|
|
|
if (m_AABB_wrapper->m_AABB.intersect_ray(
|
|
|
|
AABBWrapper::MapMatrixXfUnaligned(m_mesh->its.vertices.front().data(), m_mesh->its.vertices.size(), 3),
|
|
|
|
AABBWrapper::MapMatrixXiUnaligned(m_mesh->its.indices.front().data(), m_mesh->its.indices.size(), 3),
|
2019-09-24 10:48:05 +00:00
|
|
|
inverse_trafo * pt + direction_to_camera_mesh * EPSILON, direction_to_camera_mesh, hits)) {
|
2019-09-17 12:14:26 +00:00
|
|
|
|
|
|
|
std::sort(hits.begin(), hits.end(), [](const igl::Hit& h1, const igl::Hit& h2) { return h1.t < h2.t; });
|
2019-09-24 10:48:05 +00:00
|
|
|
|
2019-09-17 12:14:26 +00:00
|
|
|
// If the closest hit facet normal points in the same direction as the ray,
|
|
|
|
// we are looking through the mesh and should therefore discard the point:
|
|
|
|
if (m_AABB_wrapper->get_hit_normal(hits.front()).dot(direction_to_camera_mesh) > 0.f)
|
|
|
|
is_obscured = true;
|
|
|
|
|
|
|
|
// Eradicate all hits that the caller wants to ignore
|
|
|
|
for (unsigned j=0; j<hits.size(); ++j) {
|
|
|
|
const igl::Hit& hit = hits[j];
|
2019-09-24 10:48:05 +00:00
|
|
|
if (clipping_plane && clipping_plane->is_point_clipped(trafo.get_matrix() * m_AABB_wrapper->get_hit_pos(hit).cast<double>())) {
|
2019-09-17 12:14:26 +00:00
|
|
|
hits.erase(hits.begin()+j);
|
|
|
|
--j;
|
|
|
|
}
|
|
|
|
}
|
2019-09-24 10:48:05 +00:00
|
|
|
|
2019-09-17 12:14:26 +00:00
|
|
|
// FIXME: the intersection could in theory be behind the camera, but as of now we only have camera direction.
|
|
|
|
// Also, the threshold is in mesh coordinates, not in actual dimensions.
|
|
|
|
if (! hits.empty())
|
|
|
|
is_obscured = true;
|
|
|
|
}
|
|
|
|
if (! is_obscured)
|
|
|
|
out.push_back(i);
|
|
|
|
}
|
|
|
|
return out;
|
2019-09-17 08:47:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-17 12:56:46 +00:00
|
|
|
Vec3f MeshRaycaster::get_closest_point(const Vec3f& point, Vec3f* normal) const
|
|
|
|
{
|
|
|
|
int idx = 0;
|
|
|
|
Eigen::Matrix<float, 1, 3> closest_point;
|
|
|
|
m_AABB_wrapper->m_AABB.squared_distance(
|
|
|
|
AABBWrapper::MapMatrixXfUnaligned(m_mesh->its.vertices.front().data(), m_mesh->its.vertices.size(), 3),
|
|
|
|
AABBWrapper::MapMatrixXiUnaligned(m_mesh->its.indices.front().data(), m_mesh->its.indices.size(), 3),
|
|
|
|
point, idx, closest_point);
|
|
|
|
if (normal) {
|
|
|
|
igl::Hit imag_hit;
|
|
|
|
imag_hit.id = idx;
|
|
|
|
*normal = m_AABB_wrapper->get_hit_normal(imag_hit);
|
|
|
|
}
|
|
|
|
return closest_point;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-12 12:58:03 +00:00
|
|
|
|
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|