38 #ifndef PCL_REGISTRATION_IMPL_IA_FPCS_H_
39 #define PCL_REGISTRATION_IMPL_IA_FPCS_H_
43 #include <pcl/common/utils.h>
44 #include <pcl/filters/random_sample.h>
45 #include <pcl/registration/ia_fpcs.h>
46 #include <pcl/registration/transformation_estimation_3point.h>
47 #include <pcl/sample_consensus/sac_model_plane.h>
52 template <
typename Po
intT>
58 const float max_dist_sqr = max_dist * max_dist;
59 const std::size_t s = cloud->
size();
64 float mean_dist = 0.f;
67 std::vector<float> dists_sqr(2);
70 #pragma omp parallel for default(none) shared(tree, cloud) \
71 firstprivate(ids, dists_sqr) reduction(+ : mean_dist, num) \
72 firstprivate(s, max_dist_sqr) num_threads(nr_threads)
73 for (
int i = 0; i < 1000; i++) {
75 if (dists_sqr[1] < max_dist_sqr) {
76 mean_dist += std::sqrt(dists_sqr[1]);
81 return (mean_dist / num);
85 template <
typename Po
intT>
92 const float max_dist_sqr = max_dist * max_dist;
93 const std::size_t s = indices.size();
98 float mean_dist = 0.f;
101 std::vector<float> dists_sqr(2);
104 #if OPENMP_LEGACY_CONST_DATA_SHARING_RULE
105 #pragma omp parallel for default(none) shared(tree, cloud, indices) \
106 firstprivate(ids, dists_sqr) reduction(+ : mean_dist, num) num_threads(nr_threads)
108 #pragma omp parallel for default(none) shared(tree, cloud, indices, s, max_dist_sqr) \
109 firstprivate(ids, dists_sqr) reduction(+ : mean_dist, num) num_threads(nr_threads)
111 for (
int i = 0; i < 1000; i++) {
112 tree.
nearestKSearch((*cloud)[indices[rand() % s]], 2, ids, dists_sqr);
113 if (dists_sqr[1] < max_dist_sqr) {
114 mean_dist += std::sqrt(dists_sqr[1]);
119 return (mean_dist / num);
123 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
128 , score_threshold_(std::numeric_limits<float>::max())
129 , fitness_score_(std::numeric_limits<float>::max())
131 reg_name_ =
"pcl::registration::FPCSInitialAlignment";
139 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
147 final_transformation_ = guess;
149 std::vector<MatchingCandidates> all_candidates(max_iterations_);
152 #pragma omp parallel default(none) shared(abort, all_candidates, timer) \
153 num_threads(nr_threads_)
156 const unsigned int seed =
157 static_cast<unsigned int>(std::time(
nullptr)) ^ omp_get_thread_num();
159 PCL_DEBUG(
"[%s::computeTransformation] Using seed=%u\n", reg_name_.c_str(), seed);
160 #pragma omp for schedule(dynamic)
162 for (
int i = 0; i < max_iterations_; i++) {
163 #pragma omp flush(abort)
167 all_candidates[i] = candidates;
172 if (selectBase(base_indices, ratio) == 0) {
175 if (bruteForceCorrespondences(base_indices[0], base_indices[1], pairs_a) ==
177 bruteForceCorrespondences(base_indices[2], base_indices[3], pairs_b) ==
181 std::vector<pcl::Indices> matches;
182 if (determineBaseMatches(base_indices, matches, pairs_a, pairs_b, ratio) ==
185 handleMatches(base_indices, matches, candidates);
186 if (!candidates.empty())
187 all_candidates[i] = candidates;
193 if (!candidates.empty() && candidates[0].fitness_score < score_threshold_) {
194 PCL_DEBUG(
"[%s::computeTransformation] Terminating because fitness score "
195 "(%g) is below threshold (%g).\n",
197 candidates[0].fitness_score,
202 PCL_DEBUG(
"[%s::computeTransformation] Terminating because time exceeded.\n",
207 #pragma omp flush(abort)
213 finalCompute(all_candidates);
222 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
227 const unsigned int seed = std::time(
nullptr);
229 PCL_DEBUG(
"[%s::initCompute] Using seed=%u\n", reg_name_.c_str(), seed);
236 if (!input_ || !target_) {
237 PCL_ERROR(
"[%s::initCompute] Source or target dataset not given!\n",
242 if (!target_indices_ || target_indices_->empty()) {
243 target_indices_.reset(
new pcl::Indices(target_->size()));
245 for (
auto& target_index : *target_indices_)
246 target_index = index++;
247 target_cloud_updated_ =
true;
252 if (nr_samples_ > 0 &&
static_cast<std::size_t
>(nr_samples_) < indices_->size()) {
259 random_sampling.
filter(*source_indices_);
262 source_indices_ = indices_;
265 if (source_normals_ && target_normals_ && source_normals_->size() == input_->size() &&
266 target_normals_->size() == target_->size())
270 if (target_cloud_updated_) {
271 tree_->setInputCloud(target_, target_indices_);
272 target_cloud_updated_ =
false;
276 constexpr
int min_iterations = 4;
277 constexpr
float diameter_fraction = 0.3f;
280 Eigen::Vector4f pt_min, pt_max;
282 diameter_ = (pt_max - pt_min).norm();
285 float max_base_diameter = diameter_ * approx_overlap_ * 2.f;
286 max_base_diameter_sqr_ = max_base_diameter * max_base_diameter;
289 if (nr_threads_ < 1) {
290 nr_threads_ = omp_get_num_procs();
291 PCL_DEBUG(
"[%s::initCompute] Setting number of threads to %i.\n",
298 if (normalize_delta_) {
299 float mean_dist = getMeanPointDensity<PointTarget>(
300 target_, *target_indices_, 0.05f * diameter_, nr_threads_);
306 if (max_iterations_ == 0) {
307 float first_est = std::log(small_error_) /
308 std::log(1.0 - std::pow(
static_cast<double>(approx_overlap_),
309 static_cast<double>(min_iterations)));
311 static_cast<int>(first_est / (diameter_fraction * approx_overlap_ * 2.f));
312 PCL_DEBUG(
"[%s::initCompute] Estimated max iterations as %i\n",
318 if (score_threshold_ == std::numeric_limits<float>::max())
319 score_threshold_ = 1.f - approx_overlap_;
321 if (max_iterations_ < 4)
324 if (max_runtime_ < 1)
325 max_runtime_ = std::numeric_limits<int>::max();
328 max_pair_diff_ = delta_ * 2.f;
329 max_edge_diff_ = delta_ * 4.f;
330 coincidation_limit_ = delta_ * 2.f;
331 max_mse_ = powf(delta_ * 2.f, 2.f);
332 max_inlier_dist_sqr_ = powf(delta_ * 2.f, 2.f);
333 PCL_DEBUG(
"[%s::initCompute] delta_=%g, max_inlier_dist_sqr_=%g, "
334 "coincidation_limit_=%g, max_edge_diff_=%g, max_pair_diff_=%g\n",
337 max_inlier_dist_sqr_,
343 fitness_score_ = std::numeric_limits<float>::max();
349 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
354 const float too_close_sqr = max_base_diameter_sqr_ * 0.01;
356 Eigen::VectorXf coefficients(4);
359 Eigen::Vector4f centre_pt;
360 float nearest_to_plane = std::numeric_limits<float>::max();
364 for (
int i = 0; i < ransac_iterations_; i++) {
366 if (selectBaseTriangle(base_indices) < 0)
369 pcl::Indices base_triple(base_indices.begin(), base_indices.end() - 1);
374 const PointTarget* pt1 = &((*target_)[base_indices[0]]);
375 const PointTarget* pt2 = &((*target_)[base_indices[1]]);
376 const PointTarget* pt3 = &((*target_)[base_indices[2]]);
378 for (
const auto& target_index : *target_indices_) {
379 const PointTarget* pt4 = &((*target_)[target_index]);
384 float d4 = (pt4->getVector3fMap() - centre_pt.head<3>()).squaredNorm();
388 if (d1 < too_close_sqr || d2 < too_close_sqr || d3 < too_close_sqr ||
389 d4 < too_close_sqr || d1 > max_base_diameter_sqr_ ||
390 d2 > max_base_diameter_sqr_ || d3 > max_base_diameter_sqr_)
395 if (dist_to_plane < nearest_to_plane) {
396 base_indices[3] = target_index;
397 nearest_to_plane = dist_to_plane;
402 if (nearest_to_plane != std::numeric_limits<float>::max()) {
405 setupBase(base_indices, ratio);
415 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
420 const auto nr_points = target_indices_->size();
424 base_indices[0] = (*target_indices_)[rand() % nr_points];
425 auto* index1 = base_indices.data();
428 for (
int i = 0; i < ransac_iterations_; i++) {
429 auto* index2 = &(*target_indices_)[rand() % nr_points];
430 auto* index3 = &(*target_indices_)[rand() % nr_points];
433 (*target_)[*index2].getVector3fMap() - (*target_)[*index1].getVector3fMap();
435 (*target_)[*index3].getVector3fMap() - (*target_)[*index1].getVector3fMap();
437 u.cross(v).squaredNorm();
440 if (t > best_t && u.squaredNorm() < max_base_diameter_sqr_ &&
441 v.squaredNorm() < max_base_diameter_sqr_) {
443 base_indices[1] = *index2;
444 base_indices[2] = *index3;
449 return (best_t == 0.f ? -1 : 0);
453 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
458 float best_t = std::numeric_limits<float>::max();
459 const pcl::Indices copy(base_indices.begin(), base_indices.end());
460 pcl::Indices temp(base_indices.begin(), base_indices.end());
463 for (
auto i = copy.begin(), i_e = copy.end(); i != i_e; ++i)
464 for (
auto j = copy.begin(), j_e = copy.end(); j != j_e; ++j) {
468 for (
auto k = copy.begin(), k_e = copy.end(); k != k_e; ++k) {
469 if (k == j || k == i)
472 auto l = copy.begin();
473 while (l == i || l == j || l == k)
484 float t = segmentToSegmentDist(temp, ratio_temp);
487 ratio[0] = ratio_temp[0];
488 ratio[1] = ratio_temp[1];
496 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
502 Eigen::Vector3f u = (*target_)[base_indices[1]].getVector3fMap() -
503 (*target_)[base_indices[0]].getVector3fMap();
504 Eigen::Vector3f v = (*target_)[base_indices[3]].getVector3fMap() -
505 (*target_)[base_indices[2]].getVector3fMap();
506 Eigen::Vector3f w = (*target_)[base_indices[0]].getVector3fMap() -
507 (*target_)[base_indices[2]].getVector3fMap();
515 float D = a * c - b * b;
516 float sN = 0.f, sD = D;
517 float tN = 0.f, tD = D;
520 if (D < small_error_) {
527 sN = (b * e - c * d);
528 tN = (a * e - b * d);
563 else if ((-d + b) > a)
573 ratio[0] = (std::abs(sN) < small_error_) ? 0.f : sN / sD;
574 ratio[1] = (std::abs(tN) < small_error_) ? 0.f : tN / tD;
576 Eigen::Vector3f x = w + (ratio[0] * u) - (ratio[1] * v);
581 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
586 const float max_norm_diff = 0.5f * max_norm_diff_ *
M_PI / 180.f;
590 float ref_norm_angle =
591 (use_normals_ ? ((*target_normals_)[idx1].getNormalVector3fMap() -
592 (*target_normals_)[idx2].getNormalVector3fMap())
597 auto it_out = source_indices_->begin(), it_out_e = source_indices_->end() - 1;
598 auto it_in_e = source_indices_->end();
599 for (; it_out != it_out_e; it_out++) {
600 auto it_in = it_out + 1;
601 const PointSource* pt1 = &(*input_)[*it_out];
602 for (; it_in != it_in_e; it_in++) {
603 const PointSource* pt2 = &(*input_)[*it_in];
607 if (std::abs(dist - ref_dist) < max_pair_diff_) {
610 const NormalT* pt1_n = &((*source_normals_)[*it_out]);
611 const NormalT* pt2_n = &((*source_normals_)[*it_in]);
614 (pt1_n->getNormalVector3fMap() - pt2_n->getNormalVector3fMap()).norm();
616 (pt1_n->getNormalVector3fMap() + pt2_n->getNormalVector3fMap()).norm();
618 float norm_diff = std::min<float>(std::abs(norm_angle_1 - ref_norm_angle),
619 std::abs(norm_angle_2 - ref_norm_angle));
620 if (norm_diff > max_norm_diff)
624 pairs.emplace_back(*it_in, *it_out, dist);
625 pairs.emplace_back(*it_out, *it_in, dist);
631 return (pairs.empty() ? -1 : 0);
635 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
639 std::vector<pcl::Indices>& matches,
642 const float (&ratio)[2])
658 cloud_e->resize(pairs_a.size() * 2);
659 auto it_pt = cloud_e->begin();
660 for (
const auto& pair : pairs_a) {
661 const PointSource* pt1 = &((*input_)[pair.index_match]);
662 const PointSource* pt2 = &((*input_)[pair.index_query]);
665 for (
int i = 0; i < 2; i++, it_pt++) {
666 it_pt->x = pt1->x + ratio[i] * (pt2->x - pt1->x);
667 it_pt->y = pt1->y + ratio[i] * (pt2->y - pt1->y);
668 it_pt->z = pt1->z + ratio[i] * (pt2->z - pt1->z);
674 tree_e->setInputCloud(cloud_e);
677 std::vector<float> dists_sqr;
680 for (
const auto& pair : pairs_b) {
681 const PointTarget* pt1 = &((*input_)[pair.index_match]);
682 const PointTarget* pt2 = &((*input_)[pair.index_query]);
685 for (
const float& r : ratio) {
687 pt_e.x = pt1->x + r * (pt2->x - pt1->x);
688 pt_e.y = pt1->y + r * (pt2->y - pt1->y);
689 pt_e.z = pt1->z + r * (pt2->z - pt1->z);
692 tree_e->radiusSearch(pt_e, coincidation_limit_, ids, dists_sqr);
693 for (
const auto&
id : ids) {
697 pairs_a[
static_cast<int>(std::floor((
id / 2.f)))].index_match;
699 pairs_a[
static_cast<int>(std::floor((
id / 2.f)))].index_query;
700 match_indices[2] = pair.index_match;
701 match_indices[3] = pair.index_query;
702 if (match_indices[0] == match_indices[2] ||
703 match_indices[0] == match_indices[3] ||
704 match_indices[1] == match_indices[2] ||
705 match_indices[1] == match_indices[3])
709 if (checkBaseMatch(match_indices, dist_base) < 0)
712 matches.push_back(match_indices);
718 if (matches.empty()) {
719 PCL_DEBUG(
"[%s::determineBaseMatches] no matches found\n", reg_name_.c_str());
723 PCL_DEBUG(
"[%s::determineBaseMatches] found %zu matches\n",
731 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
746 return (std::abs(d0 - dist_ref[0]) < max_edge_diff_ &&
747 std::abs(d1 - dist_ref[1]) < max_edge_diff_ &&
748 std::abs(d2 - dist_ref[2]) < max_edge_diff_ &&
749 std::abs(d3 - dist_ref[3]) < max_edge_diff_)
755 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
759 std::vector<pcl::Indices>& matches,
762 candidates.resize(1);
763 float fitness_score = std::numeric_limits<float>::max();
766 for (
auto& match : matches) {
767 Eigen::Matrix4f transformation_temp;
769 correspondences_temp.emplace_back(match[0], base_indices[0], 0.0);
770 correspondences_temp.emplace_back(match[1], base_indices[1], 0.0);
771 correspondences_temp.emplace_back(match[2], base_indices[2], 0.0);
772 correspondences_temp.emplace_back(match[3], base_indices[3], 0.0);
775 if (validateMatch(base_indices, match, correspondences_temp, transformation_temp) <
781 if (validateTransformation(transformation_temp, fitness_score) < 0)
785 candidates[0].fitness_score = fitness_score;
786 candidates[0].transformation = transformation_temp;
787 correspondences_temp.erase(correspondences_temp.end() - 1);
788 candidates[0].correspondences = correspondences_temp;
793 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
801 Eigen::Vector4f centre_base{0, 0, 0, 0}, centre_match{0, 0, 0, 0};
805 PointTarget centre_pt_base;
806 centre_pt_base.x = centre_base[0];
807 centre_pt_base.y = centre_base[1];
808 centre_pt_base.z = centre_base[2];
810 PointSource centre_pt_match;
811 centre_pt_match.x = centre_match[0];
812 centre_pt_match.y = centre_match[1];
813 centre_pt_match.z = centre_match[2];
818 auto it_match_orig = match_indices.begin();
819 for (
auto it_base = base_indices.cbegin(), it_base_e = base_indices.cend();
820 it_base != it_base_e;
821 it_base++, it_match_orig++) {
824 float best_diff_sqr = std::numeric_limits<float>::max();
827 for (
const auto& match_index : copy) {
831 float diff_sqr = std::abs(dist_sqr_1 - dist_sqr_2);
833 if (diff_sqr < best_diff_sqr) {
834 best_diff_sqr = diff_sqr;
835 best_index = match_index;
840 correspondences.emplace_back(best_index, *it_base, best_diff_sqr);
841 *it_match_orig = best_index;
846 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
852 Eigen::Matrix4f& transformation)
856 correspondences_temp.erase(correspondences_temp.end() - 1);
859 transformation_estimation_->estimateRigidTransformation(
860 *input_, *target_, correspondences_temp, transformation);
863 PointCloudSource match_transformed;
867 std::size_t nr_points = correspondences_temp.size();
869 for (std::size_t i = 0; i < nr_points; i++)
871 target_->points[base_indices[i]]);
874 return (mse < max_mse_ ? 0 : -1);
878 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
884 PointCloudSource source_transformed;
886 *input_, *source_indices_, source_transformed, transformation);
888 std::size_t nr_points = source_transformed.size();
889 std::size_t terminate_value =
890 fitness_score > 1 ? 0
891 :
static_cast<std::size_t
>((1.f - fitness_score) * nr_points);
893 float inlier_score_temp = 0;
895 std::vector<float> dists_sqr(1);
896 auto it = source_transformed.begin();
898 for (std::size_t i = 0; i < nr_points; it++, i++) {
900 tree_->nearestKSearch(*it, 1, ids, dists_sqr);
901 inlier_score_temp += (dists_sqr[0] < max_inlier_dist_sqr_ ? 1 : 0);
904 if (nr_points - i + inlier_score_temp < terminate_value)
909 inlier_score_temp /=
static_cast<float>(nr_points);
910 float fitness_score_temp = 1.f - inlier_score_temp;
912 if (fitness_score_temp > fitness_score)
915 fitness_score = fitness_score_temp;
920 template <
typename Po
intSource,
typename Po
intTarget,
typename NormalT,
typename Scalar>
923 finalCompute(
const std::vector<MatchingCandidates>& candidates)
926 int nr_candidates =
static_cast<int>(candidates.size());
928 float best_score = std::numeric_limits<float>::max();
929 for (
int i = 0; i < nr_candidates; i++) {
930 const float& fitness_score = candidates[i][0].fitness_score;
931 if (fitness_score < best_score) {
932 best_score = fitness_score;
937 "[%s::finalCompute] best score is %g (iteration %i), out of %zu iterations.\n",
944 if (!(best_index < 0)) {
945 fitness_score_ = candidates[best_index][0].fitness_score;
946 final_transformation_ = candidates[best_index][0].transformation;
947 *correspondences_ = candidates[best_index][0].correspondences;
950 converged_ = fitness_score_ < score_threshold_;
void filter(Indices &indices)
Calls the filtering method and returns the filtered point cloud indices.
virtual void setInputCloud(const PointCloudConstPtr &cloud)
Provide a pointer to the input dataset.
virtual void setIndices(const IndicesPtr &indices)
Provide a pointer to the vector of indices that represents the input data.
shared_ptr< const PointCloud< PointT > > ConstPtr
RandomSample applies a random sampling with uniform probability.
void setSeed(unsigned int seed)
Set seed of random function.
void setSample(unsigned int sample)
Set number of indices to be sampled.
typename KdTreeReciprocal::Ptr KdTreeReciprocalPtr
std::string reg_name_
The registration method name.
typename PointCloudSource::Ptr PointCloudSourcePtr
int ransac_iterations_
The number of iterations RANSAC should run for.
TransformationEstimationPtr transformation_estimation_
A TransformationEstimation object, used to calculate the 4x4 rigid transformation.
int max_iterations_
The maximum number of iterations the internal optimization should run for.
void setIndices(const IndicesPtr &indices)
Provide a pointer to the vector of indices that represents the input data.
SampleConsensusModelPlane defines a model for 3D plane segmentation.
bool computeModelCoefficients(const Indices &samples, Eigen::VectorXf &model_coefficients) const override
Check whether the given index samples can form a valid plane model, compute the model coefficients fr...
double getTimeSeconds() const
Retrieve the time in seconds spent since the last call to reset().
virtual void finalCompute(const std::vector< MatchingCandidates > &candidates)
Final computation of best match out of vector of best matches.
void setupBase(pcl::Indices &base_indices, float(&ratio)[2])
Setup the base (four coplanar points) by ordering the points and computing intersection ratios and se...
int selectBaseTriangle(pcl::Indices &base_indices)
Select randomly a triplet of points with large point-to-point distances.
virtual int bruteForceCorrespondences(int idx1, int idx2, pcl::Correspondences &pairs)
Search for corresponding point pairs given the distance between two base points.
int selectBase(pcl::Indices &base_indices, float(&ratio)[2])
Select an approximately coplanar set of four points from the source cloud.
virtual int validateTransformation(Eigen::Matrix4f &transformation, float &fitness_score)
Validate the transformation by calculating the number of inliers after transforming the source cloud.
virtual int determineBaseMatches(const pcl::Indices &base_indices, std::vector< pcl::Indices > &matches, const pcl::Correspondences &pairs_a, const pcl::Correspondences &pairs_b, const float(&ratio)[2])
Determine base matches by combining the point pair candidate and search for coinciding intersection p...
virtual void linkMatchWithBase(const pcl::Indices &base_indices, pcl::Indices &match_indices, pcl::Correspondences &correspondences)
Sets the correspondences between the base B and the match M by using the distance of each point to th...
virtual void handleMatches(const pcl::Indices &base_indices, std::vector< pcl::Indices > &matches, MatchingCandidates &candidates)
Method to handle current candidate matches.
int checkBaseMatch(const pcl::Indices &match_indices, const float(&ds)[4])
Check if outer rectangle distance of matched points fit with the base rectangle.
float segmentToSegmentDist(const pcl::Indices &base_indices, float(&ratio)[2])
Calculate intersection ratios and segment to segment distances of base diagonals.
virtual int validateMatch(const pcl::Indices &base_indices, const pcl::Indices &match_indices, const pcl::Correspondences &correspondences, Eigen::Matrix4f &transformation)
Validate the matching by computing the transformation between the source and target based on the four...
FPCSInitialAlignment()
Constructor.
void computeTransformation(PointCloudSource &output, const Eigen::Matrix4f &guess) override
Rigid transformation computation method.
virtual bool initCompute()
Internal computation initialization.
search::KdTree is a wrapper class which inherits the pcl::KdTree class for performing search function...
int nearestKSearch(const PointT &point, int k, Indices &k_indices, std::vector< float > &k_sqr_distances) const override
Search for the k-nearest neighbors for the given query point.
bool setInputCloud(const PointCloudConstPtr &cloud, const IndicesConstPtr &indices=IndicesConstPtr()) override
Provide a pointer to the input dataset.
Define standard C methods to do distance calculations.
Define methods for measuring time spent in code blocks.
void getMinMax3D(const pcl::PointCloud< PointT > &cloud, PointT &min_pt, PointT &max_pt)
Get the minimum and maximum values on each of the 3 (x-y-z) dimensions in a given pointcloud.
void transformPointCloud(const pcl::PointCloud< PointT > &cloud_in, pcl::PointCloud< PointT > &cloud_out, const Eigen::Matrix< Scalar, 4, 4 > &transform, bool copy_all_fields)
Apply a rigid transform defined by a 4x4 matrix.
unsigned int compute3DCentroid(ConstCloudIterator< PointT > &cloud_iterator, Eigen::Matrix< Scalar, 4, 1 > ¢roid)
Compute the 3D (X-Y-Z) centroid of a set of points and return it as a 3D vector.
double pointToPlaneDistance(const Point &p, double a, double b, double c, double d)
Get the distance from a point to a plane (unsigned) defined by ax+by+cz+d=0.
std::vector< MatchingCandidate, Eigen::aligned_allocator< MatchingCandidate > > MatchingCandidates
void ignore(const T &...)
Utility function to eliminate unused variable warnings.
float squaredEuclideanDistance(const PointType1 &p1, const PointType2 &p2)
Calculate the squared euclidean distance between the two given points.
float getMeanPointDensity(const typename pcl::PointCloud< PointT >::ConstPtr &cloud, float max_dist, int nr_threads=1)
Compute the mean point density of a given point cloud.
std::vector< pcl::Correspondence, Eigen::aligned_allocator< pcl::Correspondence > > Correspondences
float euclideanDistance(const PointType1 &p1, const PointType2 &p2)
Calculate the euclidean distance between the two given points.
IndicesAllocator<> Indices
Type used for indices in PCL.
shared_ptr< Indices > IndicesPtr
A point structure representing normal coordinates and the surface curvature estimate.