40 #ifndef PCL_FEATURES_IMPL_FPFH_H_
41 #define PCL_FEATURES_IMPL_FPFH_H_
47 template <
typename Po
intInT,
typename Po
intNT,
typename Po
intOutT>
bool
50 int p_idx,
int q_idx,
float &f1,
float &f2,
float &f3,
float &f4)
53 cloud.
points[q_idx].getVector4fMap (), normals.
points[q_idx].getNormalVector4fMap (),
59 template <
typename Po
intInT,
typename Po
intNT,
typename Po
intOutT>
void
62 int p_idx,
int row,
const std::vector<int> &indices,
63 Eigen::MatrixXf &hist_f1, Eigen::MatrixXf &hist_f2, Eigen::MatrixXf &hist_f3)
65 Eigen::Vector4f pfh_tuple;
67 int nr_bins_f1 =
static_cast<int> (hist_f1.cols ());
68 int nr_bins_f2 =
static_cast<int> (hist_f2.cols ());
69 int nr_bins_f3 =
static_cast<int> (hist_f3.cols ());
72 float hist_incr = 100.0f /
static_cast<float>(indices.size () - 1);
75 for (
size_t idx = 0; idx < indices.size (); ++idx)
78 if (p_idx == indices[idx])
82 if (!
computePairFeatures (cloud, normals, p_idx, indices[idx], pfh_tuple[0], pfh_tuple[1], pfh_tuple[2], pfh_tuple[3]))
86 int h_index =
static_cast<int> (floor (nr_bins_f1 * ((pfh_tuple[0] + M_PI) * d_pi_)));
87 if (h_index < 0) h_index = 0;
88 if (h_index >= nr_bins_f1) h_index = nr_bins_f1 - 1;
89 hist_f1 (row, h_index) += hist_incr;
91 h_index =
static_cast<int> (floor (nr_bins_f2 * ((pfh_tuple[1] + 1.0) * 0.5)));
92 if (h_index < 0) h_index = 0;
93 if (h_index >= nr_bins_f2) h_index = nr_bins_f2 - 1;
94 hist_f2 (row, h_index) += hist_incr;
96 h_index =
static_cast<int> (floor (nr_bins_f3 * ((pfh_tuple[2] + 1.0) * 0.5)));
97 if (h_index < 0) h_index = 0;
98 if (h_index >= nr_bins_f3) h_index = nr_bins_f3 - 1;
99 hist_f3 (row, h_index) += hist_incr;
104 template <
typename Po
intInT,
typename Po
intNT,
typename Po
intOutT>
void
106 const Eigen::MatrixXf &hist_f1,
const Eigen::MatrixXf &hist_f2,
const Eigen::MatrixXf &hist_f3,
107 const std::vector<int> &indices,
const std::vector<float> &dists, Eigen::VectorXf &fpfh_histogram)
109 assert (indices.size () == dists.size ());
110 double sum_f1 = 0.0, sum_f2 = 0.0, sum_f3 = 0.0;
111 float weight = 0.0, val_f1, val_f2, val_f3;
114 int nr_bins_f1 =
static_cast<int> (hist_f1.cols ());
115 int nr_bins_f2 =
static_cast<int> (hist_f2.cols ());
116 int nr_bins_f3 =
static_cast<int> (hist_f3.cols ());
117 int nr_bins_f12 = nr_bins_f1 + nr_bins_f2;
120 fpfh_histogram.setZero (nr_bins_f1 + nr_bins_f2 + nr_bins_f3);
123 for (
size_t idx = 0, data_size = indices.size (); idx < data_size; ++idx)
130 weight = 1.0f / dists[idx];
133 for (
int f1_i = 0; f1_i < nr_bins_f1; ++f1_i)
135 val_f1 = hist_f1 (indices[idx], f1_i) * weight;
137 fpfh_histogram[f1_i] += val_f1;
140 for (
int f2_i = 0; f2_i < nr_bins_f2; ++f2_i)
142 val_f2 = hist_f2 (indices[idx], f2_i) * weight;
144 fpfh_histogram[f2_i + nr_bins_f1] += val_f2;
147 for (
int f3_i = 0; f3_i < nr_bins_f3; ++f3_i)
149 val_f3 = hist_f3 (indices[idx], f3_i) * weight;
151 fpfh_histogram[f3_i + nr_bins_f12] += val_f3;
156 sum_f1 = 100.0 / sum_f1;
158 sum_f2 = 100.0 / sum_f2;
160 sum_f3 = 100.0 / sum_f3;
163 for (
int f1_i = 0; f1_i < nr_bins_f1; ++f1_i)
164 fpfh_histogram[f1_i] *= static_cast<float> (sum_f1);
165 for (
int f2_i = 0; f2_i < nr_bins_f2; ++f2_i)
166 fpfh_histogram[f2_i + nr_bins_f1] *= static_cast<float> (sum_f2);
167 for (
int f3_i = 0; f3_i < nr_bins_f3; ++f3_i)
168 fpfh_histogram[f3_i + nr_bins_f12] *= static_cast<float> (sum_f3);
172 template <
typename Po
intInT,
typename Po
intNT,
typename Po
intOutT>
void
174 Eigen::MatrixXf &hist_f1, Eigen::MatrixXf &hist_f2, Eigen::MatrixXf &hist_f3)
178 std::vector<int> nn_indices (k_);
179 std::vector<float> nn_dists (k_);
181 std::set<int> spfh_indices;
182 spfh_hist_lookup.resize (surface_->points.size ());
186 if (surface_ != input_ ||
187 indices_->size () != surface_->points.size ())
189 for (
size_t idx = 0; idx < indices_->size (); ++idx)
191 int p_idx = (*indices_)[idx];
192 if (this->searchForNeighbors (p_idx, search_parameter_, nn_indices, nn_dists) == 0)
195 spfh_indices.insert (nn_indices.begin (), nn_indices.end ());
201 for (
size_t idx = 0; idx < indices_->size (); ++idx)
202 spfh_indices.insert (static_cast<int> (idx));
206 size_t data_size = spfh_indices.size ();
207 hist_f1.setZero (data_size, nr_bins_f1_);
208 hist_f2.setZero (data_size, nr_bins_f2_);
209 hist_f3.setZero (data_size, nr_bins_f3_);
212 std::set<int>::iterator spfh_indices_itr = spfh_indices.begin ();
213 for (
int i = 0; i < static_cast<int> (spfh_indices.size ()); ++i)
216 int p_idx = *spfh_indices_itr;
220 if (this->searchForNeighbors (*surface_, p_idx, search_parameter_, nn_indices, nn_dists) == 0)
224 computePointSPFHSignature (*surface_, *normals_, p_idx, i, nn_indices, hist_f1, hist_f2, hist_f3);
227 spfh_hist_lookup[p_idx] = i;
232 template <
typename Po
intInT,
typename Po
intNT,
typename Po
intOutT>
void
237 std::vector<int> nn_indices (k_);
238 std::vector<float> nn_dists (k_);
240 std::vector<int> spfh_hist_lookup;
241 computeSPFHSignatures (spfh_hist_lookup, hist_f1_, hist_f2_, hist_f3_);
243 output.is_dense =
true;
245 if (input_->is_dense)
248 for (
size_t idx = 0; idx < indices_->size (); ++idx)
250 if (this->searchForNeighbors ((*indices_)[idx], search_parameter_, nn_indices, nn_dists) == 0)
252 for (
int d = 0; d < fpfh_histogram_.size (); ++d)
253 output.points[idx].histogram[d] = std::numeric_limits<float>::quiet_NaN ();
255 output.is_dense =
false;
261 for (
size_t i = 0; i < nn_indices.size (); ++i)
262 nn_indices[i] = spfh_hist_lookup[nn_indices[i]];
265 weightPointSPFHSignature (hist_f1_, hist_f2_, hist_f3_, nn_indices, nn_dists, fpfh_histogram_);
268 for (
int d = 0; d < fpfh_histogram_.size (); ++d)
269 output.points[idx].histogram[d] = fpfh_histogram_[d];
275 for (
size_t idx = 0; idx < indices_->size (); ++idx)
277 if (!
isFinite ((*input_)[(*indices_)[idx]]) ||
278 this->searchForNeighbors ((*indices_)[idx], search_parameter_, nn_indices, nn_dists) == 0)
280 for (
int d = 0; d < fpfh_histogram_.size (); ++d)
281 output.points[idx].histogram[d] = std::numeric_limits<float>::quiet_NaN ();
283 output.is_dense =
false;
289 for (
size_t i = 0; i < nn_indices.size (); ++i)
290 nn_indices[i] = spfh_hist_lookup[nn_indices[i]];
293 weightPointSPFHSignature (hist_f1_, hist_f2_, hist_f3_, nn_indices, nn_dists, fpfh_histogram_);
296 for (
int d = 0; d < fpfh_histogram_.size (); ++d)
297 output.points[idx].histogram[d] = fpfh_histogram_[d];
303 template <
typename Po
intInT,
typename Po
intNT>
void
307 output.
channels[
"fpfh"].name =
"fpfh";
315 std::vector<int> nn_indices (k_);
316 std::vector<float> nn_dists (k_);
318 std::vector<int> spfh_hist_lookup;
319 this->computeSPFHSignatures (spfh_hist_lookup, hist_f1_, hist_f2_, hist_f3_);
322 output.
points.resize (indices_->size (), nr_bins_f1_ + nr_bins_f2_ + nr_bins_f3_);
325 if (input_->is_dense)
328 for (
size_t idx = 0; idx < indices_->size (); ++idx)
330 if (!
isFinite ((*input_)[(*indices_)[idx]]) ||
331 this->searchForNeighbors ((*indices_)[idx], search_parameter_, nn_indices, nn_dists) == 0)
333 output.
points.row (idx).setConstant (std::numeric_limits<float>::quiet_NaN ());
340 for (
size_t i = 0; i < nn_indices.size (); ++i)
341 nn_indices[i] = spfh_hist_lookup[nn_indices[i]];
345 output.
points.row (idx) = fpfh_histogram_;
351 for (
size_t idx = 0; idx < indices_->size (); ++idx)
353 if (this->searchForNeighbors ((*indices_)[idx], search_parameter_, nn_indices, nn_dists) == 0)
355 output.
points.row (idx).setConstant (std::numeric_limits<float>::quiet_NaN ());
362 for (
size_t i = 0; i < nn_indices.size (); ++i)
363 nn_indices[i] = spfh_hist_lookup[nn_indices[i]];
367 output.
points.row (idx) = fpfh_histogram_;
373 #define PCL_INSTANTIATE_FPFHEstimation(T,NT,OutT) template class PCL_EXPORTS pcl::FPFHEstimation<T,NT,OutT>;
375 #endif // PCL_FEATURES_IMPL_FPFH_H_