grab_point_rs.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #pragma once
  2. #include <pcl\point_types.h>
  3. #include <pcl\point_cloud.h>
  4. #include <pcl\segmentation\sac_segmentation.h>
  5. #include "data_def_api.h"
  6. #include "data_def.h"
  7. #include "logger.h"
  8. #include "imstorage_manager.h"
  9. namespace graft_cv {
  10. class CRootStockGrabPoint {
  11. public:
  12. CRootStockGrabPoint(ConfigParam&c, CGcvLogger*pLog = 0);
  13. ~CRootStockGrabPoint();
  14. int grab_point_detect(PositionInfo& posinfo);
  15. float* get_raw_point_cloud(int &data_size);
  16. int load_data(float*pPoint, int pixel_size, int pt_size, int dtype, const char* fn = 0);
  17. void set_image_saver(CImStoreManager** ppis) { m_ppImgSaver = ppis; }
  18. private:
  19. //global configure parameters
  20. ConfigParam& m_cparam;
  21. CGcvLogger * m_pLogger;
  22. int m_dtype;
  23. string m_pcdId;
  24. //用于记录第一排z均值,用来辅助判别1、2排的苗
  25. float m_1st_row_zmean_rs = -1.0;
  26. float m_1st_row_zmean_sc = -1.0;
  27. CImStoreManager** m_ppImgSaver;
  28. //返回图片,用于调试
  29. ImgInfo* m_pImginfoResult;//识别结果的图片
  30. pcl::PointCloud<pcl::PointXYZ>::Ptr m_raw_cloud;
  31. double m_cloud_mean_dist;
  32. void clear_imginfo();
  33. int read_ply_file(const char* fn);
  34. double compute_nearest_neighbor_distance(pcl::PointCloud<pcl::PointXYZ>::Ptr);
  35. /////////////////////////////////////////////////////////////////////////////////////
  36. //叶子剔除,通过欧式距离聚类,然后判断每一个类别是否是叶子
  37. // 出现的问题:叶子和茎通过叶柄连接时,会误判
  38. void leaf_filter(
  39. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  40. std::vector<int> &stem_cloud_idx
  41. );
  42. //////////////////////////////////////////////////////////////////////////////////////
  43. //叶子剔除,通过形态学方法,腐蚀,得到叶子区域,将叶子区域内地的点云去掉
  44. void leaf_filter_morph(
  45. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  46. std::vector<int> &stem_cloud_idx //output, 过滤后的点云序号
  47. );
  48. //生成3d图像
  49. void gen_3d_image(
  50. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  51. pcl::PointXYZ& min_pt, //input 图像下限
  52. pcl::PointXYZ& max_pt, //input 图像上限
  53. float step, //input 图像像素间隔
  54. cv::Mat& bin_3d_img, //output, 生成的binary图像
  55. std::map<int, std::vector<int>>& id2pcdidx
  56. );
  57. //3*3*3内的膨胀
  58. void dilation_3d(
  59. cv::Mat& in_3d_img, //input, 输入图像
  60. cv::Mat& out_3d_img //output, 生成的图像
  61. );
  62. //3*3*3内的腐蚀
  63. void erosion_3d(
  64. cv::Mat& in_3d_img,
  65. int th, //阈值
  66. cv::Mat& out_3d_img
  67. );
  68. //得到点云序号
  69. void get_mass_obj_idx(
  70. cv::Mat& open_3d_img, //input, 开运算后的图像
  71. std::map<int, std::vector<int>>& id2pcdidx, //input 保存的像素位置的点云序号
  72. std::vector<int>& mass_idx //output,
  73. );
  74. //////////////////////////////////////////////////////////////////////////////////////
  75. //叶子剔除,通过形态学方法
  76. void leaf_filter_ror(
  77. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  78. std::vector<int> &stem_cloud_idx //output, 过滤后的点云序号
  79. );
  80. void cloud_mean_dist(
  81. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  82. double& mean_dist
  83. );
  84. //////////////////////////////////////////////////////////////////////////////////////
  85. void CRootStockGrabPoint::find_fork(
  86. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud,
  87. float& max_dist,//最大距离
  88. int& max_idx //距离边界最大的点序号
  89. );
  90. void gen_all_seedling_positions(
  91. pcl::PointXYZ&key_center, //输入,已知的苗的坐标
  92. std::vector<pcl::PointXYZ>&candidate_center //输出,有倾斜苗的坐标
  93. );
  94. bool find_seedling_position_key(
  95. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  96. std::vector<int> &first_seedling_cloud_idx,
  97. pcl::PointXYZ&xz_center,
  98. pcl::ModelCoefficients::Ptr& lmodel,
  99. int& first_row_size //返回第一排植株的数量
  100. );
  101. // 邻域最小抑制
  102. void nms_box(
  103. std::vector<pcl::PointXYZ>&clt_root_v, //目标点的可能位置
  104. std::vector<float>&valid_box_cc_dist, //目标点的重心到中心的距离,距离越近越好
  105. float hole_step_radius, //目标点搜索半径
  106. std::vector<pcl::PointXYZ>& target_toot // 返回值
  107. );
  108. //通过指定位置内,取部分点云分析是否存在真正的茎,真茎位置保存到target_filtered
  109. void line_filter(pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //输入点云
  110. float radius, //输入,取点半径
  111. float ymin, //输入,y最小值
  112. float ymax, //输入,y最大值
  113. std::vector<pcl::PointXYZ>&target_root, //输入,位置
  114. std::vector<pcl::PointXYZ>&target_filtered, //输出,位置
  115. std::vector<pcl::PointXYZ>&target_filtered_root, //输出,茎上根部点坐标
  116. std::vector<std::vector<int>>&target_filtered_element, //输出,茎上点index
  117. std::vector<pcl::ModelCoefficients::Ptr>& target_filtered_models//输出,茎直线模型
  118. );
  119. void get_line_project_hist(
  120. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud, //input 茎上直线点云
  121. pcl::ModelCoefficients::Ptr line_model, //input
  122. std::vector<int>& count_hist // 返回有效直线1mm内点云数量
  123. );
  124. /////////////////////////////////////////////////
  125. void find_seedling_position(
  126. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  127. std::vector<int> &first_seedling_cloud_idx,
  128. pcl::PointXYZ&xz_center
  129. );
  130. void crop_nn_analysis(
  131. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  132. pcl::PointCloud<pcl::PointXYZ>::Ptr seed_cloud,
  133. double dist_mean,
  134. std::vector<double>&mass_indices,
  135. std::vector<int>& idx
  136. );
  137. void euclidean_clustering_ttsas(
  138. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  139. double d1, double d2,
  140. std::vector<pcl::PointXYZ>&cluster_center,
  141. std::vector<std::vector<int>> &clustr_member
  142. );
  143. void cal_obb_2d(
  144. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  145. int axis,
  146. double &dx_obb,
  147. double &dy_obb,
  148. double &angle_obb);
  149. //已知检测出苗的位置,找出有可能有苗的位置
  150. void tilted_seedling_discover(
  151. std::vector<pcl::PointXYZ>&cluster_center, //输入,已知的苗的坐标
  152. std::vector<pcl::PointXYZ>&tilted_center //输出,有倾斜苗的坐标
  153. );
  154. //通过比较直线点云和原始点云相同位置邻域内xz的范围,确定此点是否是无干扰点的茎
  155. void get_optimal_seed_compare(
  156. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input
  157. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud, //input
  158. pcl::ModelCoefficients::Ptr line_model, //input
  159. pcl::PointXYZ&pt, //输出,
  160. pcl::PointXYZ &pt_ref, //输出, 返回点茎节的参考点
  161. std::vector<int>& valid_line_index, //输出,
  162. float& stem_width_mu, //输出,茎宽度,直径
  163. float& stem_deflection //输出,茎的最大挠度,最大弯曲处的偏离直径轴心的距离,毫米
  164. );
  165. //计算茎的弯曲度:点云到直线距离的95分位
  166. double calculate_deflection(
  167. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud, //input
  168. pcl::ModelCoefficients::Ptr line_model //input
  169. );
  170. //在一株苗的空间范围内找出直线(茎,假设茎是直线分布的),并返回直线上的点index
  171. void find_line_in_cube(
  172. pcl::PointCloud<pcl::PointXYZ>::Ptr, //输入,整体点云
  173. pcl::PointXYZ& min_pt_ex, //输入,
  174. pcl::PointXYZ& max_pt_ex, //输入,
  175. std::vector<int>& out_idx //输出,直线上点的index, 基于输入整体点云
  176. );
  177. //生成结果图片
  178. void gen_result_img(
  179. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud_raw,//输入,未经过滤的整体点云in_cloud_raw,
  180. pcl::PointCloud<pcl::PointXYZ>::Ptr, //输入,整体点云cloud_dowm_sampled,
  181. pcl::PointXYZ& selected_pt, //输入,selected_pt,
  182. pcl::PointXYZ& selected_pt_ref, //输入,selected_pt_ref,
  183. cv::Mat& rst_img //输出,图片, 640*1280
  184. );
  185. void viewer_cloud(pcl::PointCloud<pcl::PointXYZ>::Ptr, std::string&winname);
  186. void viewer_cloud(pcl::PointCloud<pcl::PointXYZRGB>::Ptr, std::string&winname);
  187. void viewer_cloud_debug(
  188. pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud,
  189. pcl::PointXYZ&p,//抓取点
  190. pcl::PointXYZ &p_ref,//分叉点
  191. pcl::PointXYZ &root_pt,
  192. std::string&winname);
  193. void viewer_cloud_cluster(pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud, std::vector<pcl::PointXYZ>cluster_center, std::string&winname);
  194. void viewer_cloud_cluster_box(
  195. pcl::PointCloud<pcl::PointXYZ>::Ptr cloud,
  196. std::vector<pcl::PointXYZ>&cluster_center,
  197. std::vector<pcl::PointXYZ>&cluster_box,
  198. std::vector<std::vector<int> >& clt_line_idx,
  199. std::string&winname);
  200. };
  201. };