-
Notifications
You must be signed in to change notification settings - Fork 0
/
sidewalk_detect.cpp
154 lines (121 loc) · 5.23 KB
/
sidewalk_detect.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
// Progam to extract sidewalk from pointcloud
#include <ros/ros.h>
#include <sensor_msgs/PointCloud2.h>
#include <pcl_conversions/pcl_conversions.h>
#include <pcl_ros/point_cloud.h>
#include <pcl/point_cloud.h>
#include <pcl/point_types.h>
#include <pcl/Vertices.h>
#include <pcl/segmentation/sac_segmentation.h>
#include <pcl/segmentation/conditional_euclidean_clustering.h>
#include <pcl/segmentation/impl/conditional_euclidean_clustering.hpp>
#include <pcl/filters/extract_indices.h>
#include <pcl/sample_consensus/ransac.h>
#include <pcl/sample_consensus/sac_model_plane.h>
#include <pcl/sample_consensus/model_types.h>
#include <pcl/sample_consensus/method_types.h>
#include <pcl/ModelCoefficients.h>
#include <pcl/surface/mls.h>
#include <pcl/surface/concave_hull.h>
#include <pcl/filters/crop_box.h>
#include <pcl/filters/voxel_grid.h>
#include <pcl/filters/project_inliers.h>
#include <pcl/filters/statistical_outlier_removal.h>
#include <pcl/filters/passthrough.h>
#include <boost/thread/thread.hpp>
ros::Publisher pub;
ros::Publisher pub_out;
float deg2rad(float alpha)
{
return (alpha * 0.017453293f);
}
void ransac(pcl::PointCloud<pcl::PointXYZRGB>::Ptr input, pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_projected)
{
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZRGB>);
*cloud = *input;
pcl::ModelCoefficients::Ptr floor_coefficients(new pcl::ModelCoefficients());
pcl::PointIndices::Ptr floor_indices(new pcl::PointIndices());
pcl::SACSegmentation<pcl::PointXYZRGB> floor_finder;
floor_finder.setOptimizeCoefficients(true);
floor_finder.setModelType(pcl::SACMODEL_PARALLEL_PLANE);
//floor_finder.setModelType (SACMODEL_PLANE);
floor_finder.setMethodType(pcl::SAC_RANSAC);
floor_finder.setMaxIterations(50);
floor_finder.setAxis(Eigen::Vector3f(1, 0, 0));
floor_finder.setDistanceThreshold(1);
floor_finder.setEpsAngle(deg2rad(20));
floor_finder.setInputCloud(boost::make_shared<pcl::PointCloud<pcl::PointXYZRGB> >(*cloud));
floor_finder.segment(*floor_indices, *floor_coefficients);
if (floor_indices->indices.size() > 0)
{
// Extract the floor plane inliers
pcl::PointCloud<pcl::PointXYZRGB>::Ptr floor_points(new pcl::PointCloud<pcl::PointXYZRGB>);
pcl::ExtractIndices<pcl::PointXYZRGB> extractor;
extractor.setInputCloud(boost::make_shared<pcl::PointCloud<pcl::PointXYZRGB> >(*cloud));
extractor.setIndices(floor_indices);
extractor.filter(*floor_points);
//extractor.setNegative(true);
extractor.filter(*cloud);
// Project the floor inliers
pcl::ProjectInliers<pcl::PointXYZRGB> proj;
proj.setModelType(pcl::SACMODEL_PLANE);
proj.setInputCloud(floor_points);
proj.setModelCoefficients(floor_coefficients);
proj.filter(*cloud_projected);
floor_points->header.frame_id = "camera_depth_optical_frame";
floor_points->header.stamp = ros::Time::now().toNSec();
}
}
void passthrough_y(const sensor_msgs::PointCloud2ConstPtr& input, pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_passthrough)
{
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZRGB>);
pcl::fromROSMsg(*input, *cloud);
// Create the filtering object
pcl::PassThrough<pcl::PointXYZRGB> pass;
pass.setInputCloud (boost::make_shared<pcl::PointCloud<pcl::PointXYZRGB> >(*cloud));
pass.setFilterFieldName ("y");
pass.setFilterLimits (-20.0, 0.5);
//pass.setFilterLimitsNegative (true);
pass.filter (*cloud_passthrough);
}
void passthrough_oulier(const sensor_msgs::PointCloud2ConstPtr& input, pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_passthrough)
{
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZRGB>);
pcl::fromROSMsg(*input, *cloud);
// Create the filtering object
pcl::PassThrough<pcl::PointXYZRGB> pass;
pass.setInputCloud (boost::make_shared<pcl::PointCloud<pcl::PointXYZRGB> >(*cloud));
pass.setFilterFieldName ("y");
pass.setFilterLimits (-20.0, 0.5);
pass.setFilterLimitsNegative (true);
pass.filter (*cloud_passthrough);
}
void cloud_cb (const sensor_msgs::PointCloud2ConstPtr& input)
{
// Do data processing here...
// run pass through filter to shrink point cloud
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_passthrough(new pcl::PointCloud<pcl::PointXYZRGB>);
//passthrough_z(input, cloud_passthrough);
passthrough_y(input,cloud_passthrough);
//passthrough_x(cloud_passthrough);
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_outlier(new pcl::PointCloud<pcl::PointXYZRGB>);
passthrough_oulier(input,cloud_outlier);
pub_out.publish(*cloud_outlier);
// run ransac to find floor
pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_projected(new pcl::PointCloud<pcl::PointXYZRGB>);
ransac(cloud_passthrough, cloud_projected);
pub.publish(*cloud_projected);
}
int main (int argc, char** argv)
{
// Initialize ROS
ros::init (argc, argv, "pcl_realsense_pkg");
ros::NodeHandle nh;
// Create a ROS subscriber for the input point cloud
ros::Subscriber sub = nh.subscribe ("/camera/depth/points", 1, cloud_cb);
// Create a ROS publisher for the output point cloud
pub = nh.advertise<sensor_msgs::PointCloud2> ("/camera/depth/points_in", 1);
pub_out = nh.advertise<sensor_msgs::PointCloud2>("/camera/depth/points_out", 1);
// Spin
ros::spin ();
}