#include "MeshUtils.hpp" #include "libslic3r/Tesselate.hpp" #include "libslic3r/TriangleMesh.hpp" #include "slic3r/GUI/Camera.hpp" // There is an L function in igl that would be overridden by our localization macro. #undef L #include #include namespace Slic3r { namespace GUI { void MeshClipper::set_plane(const ClippingPlane& plane) { if (m_plane != plane) { m_plane = plane; m_triangles_valid = false; } } void MeshClipper::set_mesh(const TriangleMesh& mesh) { if (m_mesh != &mesh) { m_mesh = &mesh; m_triangles_valid = false; m_triangles2d.resize(0); m_triangles3d.resize(0); m_tms.reset(nullptr); } } void MeshClipper::set_transformation(const Geometry::Transformation& trafo) { if (! m_trafo.get_matrix().isApprox(trafo.get_matrix())) { m_trafo = trafo; m_triangles_valid = false; m_triangles2d.resize(0); m_triangles3d.resize(0); } } const std::vector& MeshClipper::get_triangles() { if (! m_triangles_valid) recalculate_triangles(); return m_triangles3d; } void MeshClipper::recalculate_triangles() { if (! m_tms) { m_tms.reset(new TriangleMeshSlicer); m_tms->init(m_mesh, [](){}); } const Transform3f& instance_matrix_no_translation_no_scaling = m_trafo.get_matrix(true,false,true).cast(); const Vec3f& scaling = m_trafo.get_scaling_factor().cast(); // Calculate clipping plane normal in mesh coordinates. Vec3f up_noscale = instance_matrix_no_translation_no_scaling.inverse() * m_plane.get_normal().cast(); 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()); // Now do the cutting std::vector list_of_expolys; m_tms->set_up_direction(up); m_tms->slice(std::vector{height_mesh}, 0.f, &list_of_expolys, [](){}); 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() * 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(); } m_triangles_valid = true; } class MeshRaycaster::AABBWrapper { public: AABBWrapper(const TriangleMesh* mesh); ~AABBWrapper() { m_AABB.deinit(); } typedef Eigen::Map> MapMatrixXfUnaligned; typedef Eigen::Map> MapMatrixXiUnaligned; igl::AABB m_AABB; Vec3f get_hit_pos(const igl::Hit& hit) const; Vec3f get_hit_normal(const igl::Hit& hit) const; private: const TriangleMesh* m_mesh; }; MeshRaycaster::AABBWrapper::AABBWrapper(const TriangleMesh* mesh) : m_mesh(mesh) { m_AABB.init( 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)); } MeshRaycaster::MeshRaycaster(const TriangleMesh& mesh) : m_AABB_wrapper(new AABBWrapper(&mesh)), m_mesh(&mesh) { } MeshRaycaster::~MeshRaycaster() { delete m_AABB_wrapper; } 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)); } bool MeshRaycaster::unproject_on_mesh(const Vec2d& mouse_pos, const Transform3d& trafo, const Camera& camera, Vec3f& position, Vec3f& normal, const ClippingPlane* clipping_plane) const { const std::array& viewport = camera.get_viewport(); const Transform3d& model_mat = camera.get_view_matrix(); const Transform3d& proj_mat = camera.get_projection_matrix(); Vec3d pt1; Vec3d pt2; ::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)); std::vector 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(), (pt2-pt1).cast(), 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; }); unsigned i = 0; // Remove points that are obscured or cut by the clipping plane if (clipping_plane) { for (i=0; iis_point_clipped(trafo * m_AABB_wrapper->get_hit_pos(hits[i]).cast())) 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; } } // 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]); return true; } std::vector MeshRaycaster::get_unobscured_idxs(const Geometry::Transformation& trafo, const Camera& camera, const std::vector& points, const ClippingPlane* clipping_plane) const { std::vector out; const Transform3d& instance_matrix_no_translation_no_scaling = trafo.get_matrix(true,false,true); Vec3f direction_to_camera = -camera.get_dir_forward().cast(); Vec3f direction_to_camera_mesh = (instance_matrix_no_translation_no_scaling.inverse().cast() * direction_to_camera).normalized().eval(); Vec3f scaling = trafo.get_scaling_factor().cast(); 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)); const Transform3f inverse_trafo = trafo.get_matrix().inverse().cast(); for (size_t i=0; iis_point_clipped(pt.cast())) continue; bool is_obscured = false; // Cast a ray in the direction of the camera and look for intersection with the mesh: std::vector hits; // Offset the start of the ray by EPSILON to account for numerical inaccuracies. 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), inverse_trafo * pt + direction_to_camera_mesh * EPSILON, direction_to_camera_mesh, hits)) { std::sort(hits.begin(), hits.end(), [](const igl::Hit& h1, const igl::Hit& h2) { return h1.t < h2.t; }); // 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; jis_point_clipped(trafo.get_matrix() * m_AABB_wrapper->get_hit_pos(hit).cast())) { hits.erase(hits.begin()+j); --j; } } // 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; } Vec3f MeshRaycaster::get_closest_point(const Vec3f& point, Vec3f* normal) const { int idx = 0; Eigen::Matrix 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; } } // namespace GUI } // namespace Slic3r