#include #include "libslic3r/KDTreeIndirect.hpp" #include "libslic3r/Execution/ExecutionSeq.hpp" #include "libslic3r/BoundingBox.hpp" #include "libslic3r/PointGrid.hpp" using namespace Slic3r; //template //struct Within { // Wrapper for the `within` predicate that counts calls. // kdtree::Within pred; // Within(G box): pred{box} {} // // Number of times the predicate was called // mutable size_t call_count = 0; // std::pair operator() (const Vec3f &p, size_t dim) // { // ++call_count; // return pred(p, dim); // } //}; static double volume(const BoundingBox3Base &box) { auto sz = box.size(); return sz.x() * sz.y() * sz.z(); } static double volume(const Eigen::AlignedBox &box) { return box.volume(); } TEST_CASE("Test kdtree query for a Box", "[KDTreeIndirect]") { auto vol = BoundingBox3Base{{0.f, 0.f, 0.f}, {10.f, 10.f, 10.f}}; auto pgrid = point_grid(ex_seq, vol, Vec3f{0.1f, 0.1f, 0.1f}); REQUIRE(!pgrid.empty()); auto coordfn = [&pgrid] (size_t i, size_t D) { return pgrid.get(i)(int(D)); }; KDTreeIndirect<3, float, decltype(coordfn)> tree{coordfn, pgrid.point_count()}; std::vector out; auto qbox = BoundingBox3Base{Vec3f{0.f, 0.f, 0.f}, Vec3f{.5f, .5f, .5f}}; size_t call_count = 0; out = find_nearby_points(tree, qbox.min, qbox.max, [&call_count](size_t) { call_count++; return true; }); // Output shall be non-empty REQUIRE(!out.empty()); std::sort(out.begin(), out.end()); // No duplicates allowed in the output auto it = std::unique(out.begin(), out.end()); REQUIRE(it == out.end()); // Test if inside points are in the output and outside points are not. bool succ = true; for (size_t i = 0; i < pgrid.point_count(); ++i) { auto foundit = std::find(out.begin(), out.end(), i); bool contains = qbox.contains(pgrid.get(i)); succ = succ && contains ? foundit != out.end() : foundit == out.end(); if (!succ) { std::cout << "invalid point: " << i << " " << pgrid.get(i).transpose() << std::endl; break; } } REQUIRE(succ); // Test for the expected cost of the query. double gridvolume = volume(vol); double queryvolume = volume(qbox); double volratio = (queryvolume / gridvolume); REQUIRE(call_count < 3 * volratio * pgrid.point_count()); REQUIRE(call_count < pgrid.point_count()); } //TEST_CASE("Test kdtree query for a Sphere", "[KDTreeIndirect]") { // auto vol = BoundingBox3Base{{0.f, 0.f, 0.f}, {10.f, 10.f, 10.f}}; // auto pgrid = point_grid(ex_seq, vol, Vec3f{0.1f, 0.1f, 0.1f}); // REQUIRE(!pgrid.empty()); // auto coordfn = [&pgrid] (size_t i, size_t D) { return pgrid.get(i)(int(D)); }; // kdtree::KDTreeIndirect<3, float, decltype(coordfn)> tree{coordfn, pgrid.point_count()}; // std::vector out; // auto querysphere = kdtree::Sphere{Vec3f{5.f, 5.f, 5.f}, 2.f}; // auto pred = Within(querysphere); // kdtree::query(tree, pred, std::back_inserter(out)); // // Output shall be non-empty // REQUIRE(!out.empty()); // std::sort(out.begin(), out.end()); // // No duplicates allowed in the output // auto it = std::unique(out.begin(), out.end()); // REQUIRE(it == out.end()); // // Test if inside points are in the output and outside points are not. // bool succ = true; // for (size_t i = 0; i < pgrid.point_count(); ++i) { // auto foundit = std::find(out.begin(), out.end(), i); // bool contains = (querysphere.center - pgrid.get(i)).squaredNorm() < pred.pred.r2; // succ = succ && contains ? foundit != out.end() : foundit == out.end(); // if (!succ) { // std::cout << "invalid point: " << i << " " << pgrid.get(i).transpose() // << std::endl; // break; // } // } // REQUIRE(succ); // // Test for the expected cost of the query. // double gridvolume = volume(vol); // double queryvolume = volume(querysphere); // double volratio = (queryvolume / gridvolume); // REQUIRE(pred.call_count < 3 * volratio * pgrid.point_count()); // REQUIRE(pred.call_count < pgrid.point_count()); //}