grab_point_rs.h 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  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. pcl::PointCloud<pcl::PointXYZ>::Ptr m_raw_cloud;
  29. double m_cloud_mean_dist;
  30. //历史茎节位置,用于计算平均茎节位置
  31. std::vector<int>m_stem_fork_ys;
  32. int m_stem_fork_ys_size;
  33. int read_ply_file(const char* fn);
  34. double compute_nearest_neighbor_distance(pcl::PointCloud<pcl::PointXYZ>::Ptr);
  35. ////////////////////////////////////////////////
  36. //---> 2023-8-8优化,seedling order识别错误问题
  37. /*void CRootStockGrabPoint::find_tray_top_edge(
  38. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  39. float & tray_top_edge_y
  40. );
  41. void CRootStockGrabPoint::find_seedling_position_line_based(
  42. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  43. std::vector<int> &first_seedling_cloud_idx,
  44. pcl::PointXYZ&xz_center
  45. );
  46. void segement_line(
  47. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud
  48. );*/
  49. //<---------
  50. /////////////////////////////////////////////////////////////////////////////////////
  51. //叶子剔除,通过欧式距离聚类,然后判断每一个类别是否是叶子
  52. // 出现的问题:叶子和茎通过叶柄连接时,会误判
  53. void leaf_filter(
  54. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  55. std::vector<int> &stem_cloud_idx
  56. );
  57. //////////////////////////////////////////////////////////////////////////////////////
  58. //叶子剔除,通过形态学方法,腐蚀,得到叶子区域,将叶子区域内地的点云去掉
  59. void leaf_filter_morph(
  60. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  61. std::vector<int> &stem_cloud_idx //output, 过滤后的点云序号
  62. );
  63. //生成3d图像
  64. void gen_3d_image(
  65. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  66. pcl::PointXYZ& min_pt, //input 图像下限
  67. pcl::PointXYZ& max_pt, //input 图像上限
  68. float step, //input 图像像素间隔
  69. cv::Mat& bin_3d_img, //output, 生成的binary图像
  70. std::map<int, std::vector<int>>& id2pcdidx
  71. );
  72. //3*3*3内的膨胀
  73. void dilation_3d(
  74. cv::Mat& in_3d_img, //input, 输入图像
  75. cv::Mat& out_3d_img //output, 生成的图像
  76. );
  77. //3*3*3内的腐蚀
  78. void erosion_3d(
  79. cv::Mat& in_3d_img,
  80. int th, //阈值
  81. cv::Mat& out_3d_img
  82. );
  83. //得到点云序号
  84. void get_mass_obj_idx(
  85. cv::Mat& open_3d_img, //input, 开运算后的图像
  86. std::map<int, std::vector<int>>& id2pcdidx, //input 保存的像素位置的点云序号
  87. std::vector<int>& mass_idx //output,
  88. );
  89. //////////////////////////////////////////////////////////////////////////////////////
  90. //叶子剔除,通过形态学方法
  91. void leaf_filter_ror(
  92. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  93. std::vector<int> &stem_cloud_idx //output, 过滤后的点云序号
  94. );
  95. void cloud_mean_dist(
  96. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input 输入点云数据
  97. double& mean_dist
  98. );
  99. //////////////////////////////////////////////////////////////////////////////////////
  100. void CRootStockGrabPoint::find_fork(
  101. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud,
  102. float& max_dist,//最大距离
  103. int& max_idx //距离边界最大的点序号
  104. );
  105. void gen_all_seedling_positions(
  106. pcl::PointXYZ&key_center, //输入,已知的苗的坐标
  107. std::vector<pcl::PointXYZ>&candidate_center //输出,有倾斜苗的坐标
  108. );
  109. bool find_seedling_position_key(
  110. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  111. std::vector<int> &first_seedling_cloud_idx,
  112. pcl::PointXYZ&xz_center,
  113. pcl::ModelCoefficients::Ptr& lmodel
  114. );
  115. // 邻域最小抑制
  116. void nms_box(
  117. std::vector<pcl::PointXYZ>&clt_root_v, //目标点的可能位置
  118. std::vector<float>&valid_box_cc_dist, //目标点的重心到中心的距离,距离越近越好
  119. float hole_step_radius, //目标点搜索半径
  120. std::vector<pcl::PointXYZ>& target_toot // 返回值
  121. );
  122. //通过指定位置内,取部分点云分析是否存在真正的茎,真茎位置保存到target_filtered
  123. void line_filter(pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //输入点云
  124. float radius, //输入,取点半径
  125. float ymin, //输入,y最小值
  126. float ymax, //输入,y最大值
  127. std::vector<pcl::PointXYZ>&target_root, //输入,位置
  128. std::vector<pcl::PointXYZ>&target_filtered, //输出,位置
  129. std::vector<pcl::PointXYZ>&target_filtered_root, //输出,茎上根部点坐标
  130. std::vector<std::vector<int>>&target_filtered_element, //输出,茎上点index
  131. std::vector<pcl::ModelCoefficients::Ptr>& target_filtered_models//输出,茎直线模型
  132. );
  133. void get_line_project_hist(
  134. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud, //input 茎上直线点云
  135. pcl::ModelCoefficients::Ptr line_model, //input
  136. std::vector<int>& count_hist // 返回有效直线1mm内点云数量
  137. );
  138. /////////////////////////////////////////////////
  139. void find_seedling_position(
  140. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  141. std::vector<int> &first_seedling_cloud_idx,
  142. pcl::PointXYZ&xz_center
  143. );
  144. void crop_nn_analysis(
  145. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  146. pcl::PointCloud<pcl::PointXYZ>::Ptr seed_cloud,
  147. double dist_mean,
  148. std::vector<double>&mass_indices,
  149. std::vector<int>& idx
  150. );
  151. void euclidean_clustering_ttsas(
  152. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  153. double d1, double d2,
  154. std::vector<pcl::PointXYZ>&cluster_center,
  155. std::vector<std::vector<int>> &clustr_member
  156. );
  157. void cal_obb_2d(
  158. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  159. int axis,
  160. double &dx_obb,
  161. double &dy_obb,
  162. double &angle_obb);
  163. void get_optimal_seed(
  164. pcl::PointCloud<pcl::PointXYZ>::Ptr,
  165. pcl::PointXYZ&pt,
  166. int &pt_idx);
  167. //已知检测出苗的位置,找出有可能有苗的位置
  168. void tilted_seedling_discover(
  169. std::vector<pcl::PointXYZ>&cluster_center, //输入,已知的苗的坐标
  170. std::vector<pcl::PointXYZ>&tilted_center //输出,有倾斜苗的坐标
  171. );
  172. //通过比较直线点云和原始点云相同位置邻域内xz的范围,确定此点是否是无干扰点的茎
  173. /*void get_optimal_seed_compare(
  174. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud,
  175. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud,
  176. pcl::PointXYZ&pt,
  177. int &pt_idx,
  178. std::vector<int>& valid_line_index
  179. );*/
  180. //通过比较直线点云和原始点云相同位置邻域内xz的范围,确定此点是否是无干扰点的茎
  181. void get_optimal_seed_compare(
  182. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud, //input
  183. pcl::PointCloud<pcl::PointXYZ>::Ptr in_line_cloud, //input
  184. pcl::ModelCoefficients::Ptr line_model, //input
  185. pcl::PointXYZ&pt, //输出,
  186. pcl::PointXYZ &pt_ref, //输出, 返回点茎节的参考点
  187. std::vector<int>& valid_line_index //输出,
  188. );
  189. //在一株苗的空间范围内找出直线(茎,假设茎是直线分布的),并返回直线上的点index
  190. void find_line_in_cube(
  191. pcl::PointCloud<pcl::PointXYZ>::Ptr, //输入,整体点云
  192. pcl::PointXYZ& min_pt_ex, //输入,
  193. pcl::PointXYZ& max_pt_ex, //输入,
  194. std::vector<int>& out_idx //输出,直线上点的index, 基于输入整体点云
  195. );
  196. //生成结果图片
  197. void gen_result_img(
  198. pcl::PointCloud<pcl::PointXYZ>::Ptr in_cloud_raw,//输入,未经过滤的整体点云in_cloud_raw,
  199. pcl::PointCloud<pcl::PointXYZ>::Ptr, //输入,整体点云cloud_dowm_sampled,
  200. pcl::PointXYZ& selected_pt, //输入,selected_pt,
  201. pcl::PointXYZ& selected_pt_ref, //输入,selected_pt_ref,
  202. cv::Mat& rst_img //输出,图片, 640*1280
  203. );
  204. void viewer_cloud(pcl::PointCloud<pcl::PointXYZ>::Ptr, std::string&winname);
  205. void viewer_cloud(pcl::PointCloud<pcl::PointXYZRGB>::Ptr, std::string&winname);
  206. void viewer_cloud_debug(
  207. pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud,
  208. pcl::PointXYZ&p,//抓取点
  209. pcl::PointXYZ &p_ref,//分叉点
  210. pcl::PointXYZ &root_pt,
  211. std::string&winname);
  212. void viewer_cloud_cluster(pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud, std::vector<pcl::PointXYZ>cluster_center, std::string&winname);
  213. void viewer_cloud_cluster_box(
  214. pcl::PointCloud<pcl::PointXYZ>::Ptr cloud,
  215. std::vector<pcl::PointXYZ>&cluster_center,
  216. std::vector<pcl::PointXYZ>&cluster_box,
  217. std::vector<std::vector<int> >& clt_line_idx,
  218. std::string&winname);
  219. };
  220. };