12#include <Eigen/Geometry>
31 :
ICP(), overlap_rate(config.get<double>(
"overlap_rate", 0.9)) {}
36 a_current.resize(
a.size());
43 const size_t n =
a.size();
45 for (
size_t i = 0; i < n; i++) {
62 [](
const auto&
a,
const auto&
b) { return a.cost < b.cost; });
63 size_t new_n =
static_cast<size_t>(overlap_rate * n);
64 new_n = std::max<size_t>(new_n, 1);
67 std::vector<icp::Vector> trimmed_current(new_n);
68 std::vector<icp::Vector> trimmed_b(new_n);
69 for (
size_t i = 0; i < new_n; i++) {
70 trimmed_current[i] = a_current[
matches[i].point];
79 for (
size_t i = 0; i < new_n; i++) {
80 N += (trimmed_current[i] - trimmed_cm) * (trimmed_b[i] - trimmed_b_cm).transpose();
83 auto svd = N.jacobiSvd(Eigen::ComputeFullU | Eigen::ComputeFullV);
86 Matrix R = V * U.transpose();
89 if (R.determinant() < 0) {
90 V = V * Eigen::DiagonalMatrix<double, 2>(1, -1);
91 R = V * U.transpose();
100 void Trimmed::compute_matches() {
101 const size_t n =
a.size();
102 const size_t m =
b.size();
104 for (
size_t i = 0; i < n; i++) {
106 matches[i].cost = std::numeric_limits<double>::infinity();
107 for (
size_t j = 0; j < m; j++) {
109 double dist_ij = (
b[j] - a_current[i]).squaredNorm();
111 if (dist_ij <
matches[i].cost) {
Configuration for ICP instances.
Interface for iterative closest points.
RBTransform transform
The current point cloud transformation that is being optimized.
std::vector< Vector > a
The source point cloud.
std::vector< Vector > b
The destination point cloud.
std::vector< Match > matches
The pairing of each point in a to its closest in b.
void setup() override
Per-method setup code.
Trimmed(double overlap_rate)
void iterate() override
Perform one iteration of ICP for the point clouds a and b provided with ICP::begin.
Vector get_centroid(const std::vector< Vector > &points)