49 #include <opencv2/highgui/highgui.hpp>
50 #include <opencv2/imgproc/imgproc.hpp>
52 #include <dynamic_reconfigure/server.h>
53 #include "opencv_apps/GeneralContoursConfig.h"
54 #include "opencv_apps/RotatedRect.h"
55 #include "opencv_apps/RotatedRectArray.h"
56 #include "opencv_apps/RotatedRectArrayStamped.h"
69 typedef opencv_apps::GeneralContoursConfig
Config;
89 const std::string&
frameWithDefault(
const std::string& frame,
const std::string& image_frame)
96 void imageCallbackWithInfo(
const sensor_msgs::ImageConstPtr& msg,
const sensor_msgs::CameraInfoConstPtr& cam_info)
98 doWork(msg, cam_info->header.frame_id);
103 doWork(msg, msg->header.frame_id);
111 void doWork(
const sensor_msgs::ImageConstPtr& msg,
const std::string& input_frame_from_msg)
120 opencv_apps::RotatedRectArrayStamped rects_msg, ellipses_msg;
121 rects_msg.header = msg->header;
122 ellipses_msg.header = msg->header;
128 if (frame.channels() > 1)
130 cv::cvtColor(frame, src_gray, cv::COLOR_RGB2GRAY);
136 cv::blur(src_gray, src_gray, cv::Size(3, 3));
144 cv::Mat threshold_output;
145 int max_thresh = 255;
146 std::vector<std::vector<cv::Point> > contours;
147 std::vector<cv::Vec4i> hierarchy;
151 cv::threshold(src_gray, threshold_output,
threshold_, 255, cv::THRESH_BINARY);
154 cv::findContours(threshold_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, cv::Point(0, 0));
157 std::vector<cv::RotatedRect> min_rect(contours.size());
158 std::vector<cv::RotatedRect> min_ellipse(contours.size());
160 for (
size_t i = 0; i < contours.size(); i++)
162 min_rect[i] = cv::minAreaRect(cv::Mat(contours[i]));
163 if (contours[i].size() > 5)
165 min_ellipse[i] = cv::fitEllipse(cv::Mat(contours[i]));
170 cv::Mat drawing = cv::Mat::zeros(threshold_output.size(), CV_8UC3);
171 for (
size_t i = 0; i < contours.size(); i++)
173 cv::Scalar color = cv::Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255));
175 cv::drawContours(drawing, contours, (
int)i, color, 1, 8, std::vector<cv::Vec4i>(), 0, cv::Point());
177 cv::ellipse(drawing, min_ellipse[i], color, 2, 8);
179 cv::Point2f rect_points[4];
180 min_rect[i].points(rect_points);
181 for (
int j = 0; j < 4; j++)
182 cv::line(drawing, rect_points[j], rect_points[(j + 1) % 4], color, 1, 8);
184 opencv_apps::RotatedRect rect_msg;
185 opencv_apps::Point2D rect_center;
186 opencv_apps::Size rect_size;
187 rect_center.x = min_rect[i].center.x;
188 rect_center.y = min_rect[i].center.y;
189 rect_size.width = min_rect[i].size.width;
190 rect_size.height = min_rect[i].size.height;
191 rect_msg.center = rect_center;
192 rect_msg.size = rect_size;
193 rect_msg.angle = min_rect[i].angle;
194 opencv_apps::RotatedRect ellipse_msg;
195 opencv_apps::Point2D ellipse_center;
196 opencv_apps::Size ellipse_size;
197 ellipse_center.x = min_ellipse[i].center.x;
198 ellipse_center.y = min_ellipse[i].center.y;
199 ellipse_size.width = min_ellipse[i].size.width;
200 ellipse_size.height = min_ellipse[i].size.height;
201 ellipse_msg.center = ellipse_center;
202 ellipse_msg.size = ellipse_size;
203 ellipse_msg.angle = min_ellipse[i].angle;
205 rects_msg.rects.push_back(rect_msg);
206 ellipses_msg.rects.push_back(ellipse_msg);
221 int c = cv::waitKey(1);
225 sensor_msgs::Image::Ptr out_img =
231 catch (cv::Exception& e)
233 NODELET_ERROR(
"Image processing error: %s %s %s %i", e.err.c_str(), e.func.c_str(), e.file.c_str(), e.line);
273 dynamic_reconfigure::Server<Config>::CallbackType
f = boost::bind(
278 rects_pub_ = advertise<opencv_apps::RotatedRectArrayStamped>(*
pnh_,
"rectangles", 1);
279 ellipses_pub_ = advertise<opencv_apps::RotatedRectArrayStamped>(*
pnh_,
"ellipses", 1);
294 ROS_WARN(
"DeprecationWarning: Nodelet general_contours/general_contours is deprecated, "
295 "and renamed to opencv_apps/general_contours.");
301 #ifdef USE_PLUGINLIB_CLASS_LIST_MACROS_H