38 #ifndef PCL_FEATURES_IMPL_NORMAL_BASED_SIGNATURE_H_
39 #define PCL_FEATURES_IMPL_NORMAL_BASED_SIGNATURE_H_
43 template <
typename Po
intT,
typename Po
intNT,
typename Po
intFeature>
void
48 PointFeature test_feature;
49 if (N_prime_ * M_prime_ !=
sizeof (test_feature.values) / sizeof (
float))
51 PCL_ERROR (
"NormalBasedSignatureEstimation: not using the proper signature size: %u vs %u\n", N_prime_ * M_prime_,
sizeof (test_feature.values) / sizeof (
float));
55 std::vector<int> k_indices;
56 std::vector<float> k_sqr_distances;
58 tree_->setInputCloud (input_);
59 output.points.resize (indices_->size ());
61 for (
size_t index_i = 0; index_i < indices_->size (); ++index_i)
63 size_t point_i = (*indices_)[index_i];
64 Eigen::MatrixXf s_matrix (N_, M_);
66 Eigen::Vector4f center_point = input_->points[point_i].getVector4fMap ();
68 for (
size_t k = 0; k < N_; ++k)
70 Eigen::VectorXf s_row (M_);
72 for (
size_t l = 0; l < M_; ++l)
74 Eigen::Vector4f normal = normals_->points[point_i].getNormalVector4fMap ();
75 Eigen::Vector4f normal_u = Eigen::Vector4f::Zero ();
76 Eigen::Vector4f normal_v = Eigen::Vector4f::Zero ();
78 if (fabs (normal.x ()) > 0.0001f)
80 normal_u.x () = - normal.y () / normal.x ();
83 normal_u.normalize ();
86 else if (fabs (normal.y ()) > 0.0001f)
89 normal_u.y () = - normal.x () / normal.y ();
91 normal_u.normalize ();
97 normal_u.z () = - normal.y () / normal.z ();
99 normal_v = normal.cross3 (normal_u);
101 Eigen::Vector4f zeta_point = 2.0f *
static_cast<float> (l + 1) * scale_h_ / static_cast<float> (M_) *
102 (cosf (2.0f * static_cast<float> (M_PI) * static_cast<float> ((k + 1) / N_)) * normal_u +
103 sinf (2.0f * static_cast<float> (M_PI) * static_cast<float> ((k + 1) / N_)) * normal_v);
106 Eigen::Vector4f zeta_point_plus_center = zeta_point + center_point;
108 zeta_point_pcl.x = zeta_point_plus_center.x (); zeta_point_pcl.y = zeta_point_plus_center.y (); zeta_point_pcl.z = zeta_point_plus_center.z ();
110 tree_->radiusSearch (zeta_point_pcl, search_radius_, k_indices, k_sqr_distances);
113 if (k_indices.size () == 0)
115 k_indices.resize (5);
116 k_sqr_distances.resize (5);
117 tree_->nearestKSearch (zeta_point_pcl, 5, k_indices, k_sqr_distances);
120 Eigen::Vector4f average_normal = Eigen::Vector4f::Zero ();
122 float average_normalization_factor = 0.0f;
125 for (
size_t nn_i = 0; nn_i < k_indices.size (); ++nn_i)
127 if (k_sqr_distances[nn_i] < 1e-7f)
129 average_normal = normals_->points[k_indices[nn_i]].getNormalVector4fMap ();
130 average_normalization_factor = 1.0f;
133 average_normal += normals_->points[k_indices[nn_i]].getNormalVector4fMap () / k_sqr_distances[nn_i];
134 average_normalization_factor += 1.0f / k_sqr_distances[nn_i];
136 average_normal /= average_normalization_factor;
137 float s = zeta_point.dot (average_normal) / zeta_point.norm ();
142 Eigen::VectorXf dct_row (M_);
143 for (
int m = 0; m < s_row.size (); ++m)
146 for (
int n = 0; n < s_row.size (); ++n)
147 Xk += static_cast<float> (s_row[n] * cos (M_PI / (static_cast<double> (M_ * n) + 0.5) * static_cast<double> (k)));
151 s_matrix.row (k) = dct_row;
155 Eigen::MatrixXf dft_matrix (N_, M_);
156 for (
size_t column_i = 0; column_i < M_; ++column_i)
158 Eigen::VectorXf dft_col (N_);
159 for (
size_t k = 0; k < N_; ++k)
161 float Xk_real = 0.0f, Xk_imag = 0.0f;
162 for (
size_t n = 0; n < N_; ++n)
164 Xk_real +=
static_cast<float> (s_matrix (n, column_i) * cos (2.0f * M_PI / static_cast<double> (N_ * k * n)));
165 Xk_imag +=
static_cast<float> (s_matrix (n, column_i) * sin (2.0f * M_PI / static_cast<double> (N_ * k * n)));
167 dft_col[k] =
sqrt (Xk_real*Xk_real + Xk_imag*Xk_imag);
169 dft_matrix.col (column_i) = dft_col;
172 Eigen::MatrixXf final_matrix = dft_matrix.block (0, 0, N_prime_, M_prime_);
174 PointFeature feature_point;
175 for (
size_t i = 0; i < N_prime_; ++i)
176 for (
size_t j = 0; j < M_prime_; ++j)
177 feature_point.values[i*M_prime_ + j] = final_matrix (i, j);
179 output.points[index_i] = feature_point;
185 #define PCL_INSTANTIATE_NormalBasedSignatureEstimation(T,NT,OutT) template class PCL_EXPORTS pcl::NormalBasedSignatureEstimation<T,NT,OutT>;