#include "SLA/SLASupportTree.hpp" #include "SLA/SLABoilerPlate.hpp" #include "SLA/SLASpatIndex.hpp" // HEAVY headers... takes eternity to compile // for concave hull merging decisions #include "SLABoostAdapter.hpp" #include "boost/geometry/index/rtree.hpp" #include //#if !defined(_MSC_VER) || defined(_WIN64) #if 1 #define IGL_COMPATIBLE #endif #ifdef IGL_COMPATIBLE #include #endif #include "SLASpatIndex.hpp" #include "ClipperUtils.hpp" namespace Slic3r { namespace sla { class SpatIndex::Impl { public: using BoostIndex = boost::geometry::index::rtree< SpatElement, boost::geometry::index::rstar<16, 4> /* ? */ >; BoostIndex m_store; }; SpatIndex::SpatIndex(): m_impl(new Impl()) {} SpatIndex::~SpatIndex() {} SpatIndex::SpatIndex(const SpatIndex &cpy): m_impl(new Impl(*cpy.m_impl)) {} SpatIndex::SpatIndex(SpatIndex&& cpy): m_impl(std::move(cpy.m_impl)) {} SpatIndex& SpatIndex::operator=(const SpatIndex &cpy) { m_impl.reset(new Impl(*cpy.m_impl)); return *this; } SpatIndex& SpatIndex::operator=(SpatIndex &&cpy) { m_impl.swap(cpy.m_impl); return *this; } void SpatIndex::insert(const SpatElement &el) { m_impl->m_store.insert(el); } bool SpatIndex::remove(const SpatElement& el) { return m_impl->m_store.remove(el) == 1; } std::vector SpatIndex::query(std::function fn) { namespace bgi = boost::geometry::index; std::vector ret; m_impl->m_store.query(bgi::satisfies(fn), std::back_inserter(ret)); return ret; } std::vector SpatIndex::nearest(const Vec3d &el, unsigned k = 1) { namespace bgi = boost::geometry::index; std::vector ret; ret.reserve(k); m_impl->m_store.query(bgi::nearest(el, k), std::back_inserter(ret)); return ret; } size_t SpatIndex::size() const { return m_impl->m_store.size(); } PointSet normals(const PointSet& points, const EigenMesh3D& mesh) { if(points.rows() == 0 || mesh.V.rows() == 0 || mesh.F.rows() == 0) return {}; #ifdef IGL_COMPATIBLE Eigen::VectorXd dists; Eigen::VectorXi I; // Eigen::Matrix dists; // Eigen::Matrix I; PointSet C; igl::point_mesh_squared_distance( points, mesh.V, mesh.F, dists, I, C); PointSet ret(I.rows(), 3); for(int i = 0; i < I.rows(); i++) { auto idx = I(i); auto trindex = mesh.F.row(idx); auto& p1 = mesh.V.row(trindex(0)); auto& p2 = mesh.V.row(trindex(1)); auto& p3 = mesh.V.row(trindex(2)); Eigen::Vector3d U = p2 - p1; Eigen::Vector3d V = p3 - p1; ret.row(i) = U.cross(V).normalized(); } return ret; #else // TODO: do something on 32 bit windows return {}; #endif } double ray_mesh_intersect(const Vec3d& s, const Vec3d& dir, const EigenMesh3D& m) { igl::Hit hit; hit.t = std::numeric_limits::infinity(); igl::ray_mesh_intersect(s, dir, m.V, m.F, hit); return hit.t; } // Clustering a set of points by the given criteria ClusteredPoints cluster( const sla::PointSet& points, std::function pred, unsigned max_points = 0) { namespace bgi = boost::geometry::index; using Index3D = bgi::rtree< SpatElement, bgi::rstar<16, 4> /* ? */ >; // A spatial index for querying the nearest points Index3D sindex; // Build the index for(unsigned idx = 0; idx < points.rows(); idx++) sindex.insert( std::make_pair(points.row(idx), idx)); using Elems = std::vector; // Recursive function for visiting all the points in a given distance to // each other std::function group = [&sindex, &group, pred, max_points](Elems& pts, Elems& cluster) { for(auto& p : pts) { std::vector tmp; sindex.query( bgi::satisfies([p, pred](const SpatElement& se) { return pred(p, se); }), std::back_inserter(tmp) ); auto cmp = [](const SpatElement& e1, const SpatElement& e2){ return e1.second < e2.second; }; std::sort(tmp.begin(), tmp.end(), cmp); Elems newpts; std::set_difference(tmp.begin(), tmp.end(), cluster.begin(), cluster.end(), std::back_inserter(newpts), cmp); int c = max_points && newpts.size() + cluster.size() > max_points? int(max_points - cluster.size()) : int(newpts.size()); cluster.insert(cluster.end(), newpts.begin(), newpts.begin() + c); std::sort(cluster.begin(), cluster.end(), cmp); if(!newpts.empty() && (!max_points || cluster.size() < max_points)) group(newpts, cluster); } }; std::vector clusters; for(auto it = sindex.begin(); it != sindex.end();) { Elems cluster = {}; Elems pts = {*it}; group(pts, cluster); for(auto& c : cluster) sindex.remove(c); it = sindex.begin(); clusters.emplace_back(cluster); } ClusteredPoints result; for(auto& cluster : clusters) { result.emplace_back(); for(auto c : cluster) result.back().emplace_back(c.second); } return result; } using Segments = std::vector>; Segments model_boundary(const EigenMesh3D& emesh, double offs) { Segments ret; Polygons pp; pp.reserve(emesh.F.rows()); for (int i = 0; i < emesh.F.rows(); i++) { auto trindex = emesh.F.row(i); auto& p1 = emesh.V.row(trindex(0)); auto& p2 = emesh.V.row(trindex(1)); auto& p3 = emesh.V.row(trindex(2)); Polygon p; p.points.resize(3); p.points[0] = Point::new_scale(p1(X), p1(Y)); p.points[1] = Point::new_scale(p2(X), p2(Y)); p.points[2] = Point::new_scale(p3(X), p3(Y)); p.make_counter_clockwise(); pp.emplace_back(p); } ExPolygons merged = union_ex(offset(pp, float(scale_(offs))), true); for(auto& expoly : merged) { auto lines = expoly.lines(); for(Line& l : lines) { Vec2d a(l.a(X) * SCALING_FACTOR, l.a(Y) * SCALING_FACTOR); Vec2d b(l.b(X) * SCALING_FACTOR, l.b(Y) * SCALING_FACTOR); ret.emplace_back(std::make_pair(a, b)); } } return ret; } //struct SegmentIndex { //}; //using SegmentIndexEl = std::pair; //SegmentIndexEl } }