goodfeature_track_nodelet.cpp
Go to the documentation of this file.
1 /*********************************************************************
2 * Software License Agreement (BSD License)
3 *
4 * Copyright (c) 2014, Kei Okada.
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 *
11 * * Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * * Redistributions in binary form must reproduce the above
14 * copyright notice, this list of conditions and the following
15 * disclaimer in the documentation and/or other materials provided
16 * with the distribution.
17 * * Neither the name of the Kei Okada nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32 * POSSIBILITY OF SUCH DAMAGE.
33 *********************************************************************/
34 
35 // http://github.com/Itseez/opencv/blob/master/samples/cpp/tutorial_code/TrackingMotion/goodFeaturesToTrack_Demo.cpp
42 #include <ros/ros.h>
43 #include "opencv_apps/nodelet.h"
45 #include <cv_bridge/cv_bridge.h>
47 
48 #include <opencv2/highgui/highgui.hpp>
49 #include <opencv2/imgproc/imgproc.hpp>
50 
51 #include <dynamic_reconfigure/server.h>
52 #include "opencv_apps/GoodfeatureTrackConfig.h"
53 #include "opencv_apps/Point2D.h"
54 #include "opencv_apps/Point2DArrayStamped.h"
55 
56 namespace opencv_apps
57 {
58 class GoodfeatureTrackNodelet : public opencv_apps::Nodelet
59 {
64 
66 
67  typedef opencv_apps::GoodfeatureTrackConfig Config;
68  typedef dynamic_reconfigure::Server<Config> ReconfigureServer;
71 
72  int queue_size_;
73  bool debug_view_;
75 
76  std::string window_name_;
77  static bool need_config_update_;
78 
79  int max_corners_;
80 
81  void reconfigureCallback(Config& new_config, uint32_t level)
82  {
83  config_ = new_config;
84  max_corners_ = config_.max_corners;
85  }
86 
87  const std::string& frameWithDefault(const std::string& frame, const std::string& image_frame)
88  {
89  if (frame.empty())
90  return image_frame;
91  return frame;
92  }
93 
94  void imageCallbackWithInfo(const sensor_msgs::ImageConstPtr& msg, const sensor_msgs::CameraInfoConstPtr& cam_info)
95  {
96  doWork(msg, cam_info->header.frame_id);
97  }
98 
99  void imageCallback(const sensor_msgs::ImageConstPtr& msg)
100  {
101  doWork(msg, msg->header.frame_id);
102  }
103 
104  static void trackbarCallback(int /*unused*/, void* /*unused*/)
105  {
106  need_config_update_ = true;
107  }
108 
109  void doWork(const sensor_msgs::ImageConstPtr& msg, const std::string& input_frame_from_msg)
110  {
111  // Work on the image.
112  try
113  {
114  // Convert the image into something opencv can handle.
115  cv::Mat frame = cv_bridge::toCvShare(msg, sensor_msgs::image_encodings::BGR8)->image;
116 
117  // Messages
118  opencv_apps::Point2DArrayStamped corners_msg;
119  corners_msg.header = msg->header;
120 
121  // Do the work
122  cv::Mat src_gray;
123  int max_trackbar = 100;
124 
125  if (frame.channels() > 1)
126  {
127  cv::cvtColor(frame, src_gray, cv::COLOR_BGR2GRAY);
128  }
129  else
130  {
131  src_gray = frame;
132  cv::cvtColor(src_gray, frame, cv::COLOR_GRAY2BGR);
133  }
134 
135  if (debug_view_)
136  {
138  cv::namedWindow(window_name_, cv::WINDOW_AUTOSIZE);
139  cv::createTrackbar("Max corners", window_name_, &max_corners_, max_trackbar, trackbarCallback);
141  {
142  config_.max_corners = max_corners_;
143  reconfigure_server_->updateConfig(config_);
144  need_config_update_ = false;
145  }
146  }
147 
149  if (max_corners_ < 1)
150  {
152  }
153 
155  std::vector<cv::Point2f> corners;
156  double quality_level = 0.01;
157  double min_distance = 10;
158  int block_size = 3;
159  bool use_harris_detector = false;
160  double k = 0.04;
161 
162  cv::RNG rng(12345);
163 
165  cv::goodFeaturesToTrack(src_gray, corners, max_corners_, quality_level, min_distance, cv::Mat(), block_size,
166  use_harris_detector, k);
167 
169  NODELET_INFO_STREAM("** Number of corners detected: " << corners.size());
170  int r = 4;
171  for (const cv::Point2f& corner : corners)
172  {
173  cv::circle(frame, corner, r, cv::Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), -1, 8,
174  0);
175  }
176 
177  //-- Show what you got
178  if (debug_view_)
179  {
180  cv::imshow(window_name_, frame);
181  int c = cv::waitKey(1);
182  }
183 
184  // Create msgs
185  for (const cv::Point2f& i : corners)
186  {
187  opencv_apps::Point2D corner;
188  corner.x = i.x;
189  corner.y = i.y;
190  corners_msg.points.push_back(corner);
191  }
192  // Publish the image.
193  sensor_msgs::Image::Ptr out_img = cv_bridge::CvImage(msg->header, "bgr8", frame).toImageMsg();
194  img_pub_.publish(out_img);
195  msg_pub_.publish(corners_msg);
196  }
197  catch (cv::Exception& e)
198  {
199  NODELET_ERROR("Image processing error: %s %s %s %i", e.err.c_str(), e.func.c_str(), e.file.c_str(), e.line);
200  }
201 
202  prev_stamp_ = msg->header.stamp;
203  }
204 
205  void subscribe() // NOLINT(modernize-use-override)
206  {
207  NODELET_DEBUG("Subscribing to image topic.");
208  if (config_.use_camera_info)
209  cam_sub_ = it_->subscribeCamera("image", queue_size_, &GoodfeatureTrackNodelet::imageCallbackWithInfo, this);
210  else
211  img_sub_ = it_->subscribe("image", queue_size_, &GoodfeatureTrackNodelet::imageCallback, this);
212  }
213 
214  void unsubscribe() // NOLINT(modernize-use-override)
215  {
216  NODELET_DEBUG("Unsubscribing from image topic.");
217  img_sub_.shutdown();
218  cam_sub_.shutdown();
219  }
220 
221 public:
222  virtual void onInit() // NOLINT(modernize-use-override)
223  {
224  Nodelet::onInit();
226 
227  pnh_->param("queue_size", queue_size_, 3);
228  pnh_->param("debug_view", debug_view_, false);
229  if (debug_view_)
230  {
231  always_subscribe_ = true;
232  }
233  prev_stamp_ = ros::Time(0, 0);
234 
235  window_name_ = "Image";
236  max_corners_ = 23;
237 
238  reconfigure_server_ = boost::make_shared<dynamic_reconfigure::Server<Config> >(*pnh_);
239  dynamic_reconfigure::Server<Config>::CallbackType f = boost::bind(
241  reconfigure_server_->setCallback(f);
242 
243  img_pub_ = advertiseImage(*pnh_, "image", 1);
244  msg_pub_ = advertise<opencv_apps::Point2DArrayStamped>(*pnh_, "corners", 1);
245 
247  }
248 };
250 } // namespace opencv_apps
251 
253 {
255 {
256 public:
257  virtual void onInit() // NOLINT(modernize-use-override)
258  {
259  ROS_WARN("DeprecationWarning: Nodelet goodfeature_track/goodfeature_track is deprecated, "
260  "and renamed to opencv_apps/goodfeature_track.");
262  }
263 };
264 } // namespace goodfeature_track
265 
266 #ifdef USE_PLUGINLIB_CLASS_LIST_MACROS_H
268 #else
270 #endif
nodelet.h
opencv_apps::GoodfeatureTrackNodelet::cam_sub_
image_transport::CameraSubscriber cam_sub_
Definition: goodfeature_track_nodelet.cpp:126
NODELET_ERROR
#define NODELET_ERROR(...)
opencv_apps::Nodelet::onInitPostProcess
virtual void onInitPostProcess()
Post processing of initialization of nodelet. You need to call this method in order to use always_sub...
Definition: nodelet.cpp:77
opencv_apps::GoodfeatureTrackNodelet::Config
opencv_apps::GoodfeatureTrackConfig Config
Definition: goodfeature_track_nodelet.cpp:131
ros::Publisher
opencv_apps::GoodfeatureTrackNodelet::max_corners_
int max_corners_
Definition: goodfeature_track_nodelet.cpp:143
image_encodings.h
image_transport::ImageTransport
boost::shared_ptr< image_transport::ImageTransport >
cv_bridge::CvImage::toImageMsg
sensor_msgs::ImagePtr toImageMsg() const
opencv_apps::GoodfeatureTrackNodelet::img_sub_
image_transport::Subscriber img_sub_
Definition: goodfeature_track_nodelet.cpp:125
ros.h
boost::placeholders::_1
boost::arg< 1 > _1
Definition: nodelet.cpp:44
opencv_apps::Nodelet
Nodelet to automatically subscribe/unsubscribe topics according to subscription of advertised topics.
Definition: nodelet.h:90
opencv_apps::GoodfeatureTrackNodelet::imageCallback
void imageCallback(const sensor_msgs::ImageConstPtr &msg)
Definition: goodfeature_track_nodelet.cpp:163
opencv_apps::GoodfeatureTrackNodelet::subscribe
void subscribe()
This method is called when publisher is subscribed by other nodes. Set up subscribers in this method.
Definition: goodfeature_track_nodelet.cpp:269
opencv_apps::GoodfeatureTrackNodelet::trackbarCallback
static void trackbarCallback(int, void *)
Definition: goodfeature_track_nodelet.cpp:168
opencv_apps::GoodfeatureTrackNodelet::img_pub_
image_transport::Publisher img_pub_
Definition: goodfeature_track_nodelet.cpp:124
goodfeature_track
Definition: goodfeature_track_nodelet.cpp:252
ros::Publisher::publish
void publish(const boost::shared_ptr< M > &message) const
f
f
image_transport::Subscriber
opencv_apps::Nodelet::advertiseImage
image_transport::Publisher advertiseImage(ros::NodeHandle &nh, const std::string &topic, int queue_size)
Advertise an image topic and watch the publisher. Publishers which are created by this method....
Definition: nodelet.h:201
class_list_macros.h
opencv_apps::GoodfeatureTrackNodelet::window_name_
std::string window_name_
Definition: goodfeature_track_nodelet.cpp:140
opencv_apps::GoodfeatureTrackNodelet::frameWithDefault
const std::string & frameWithDefault(const std::string &frame, const std::string &image_frame)
Definition: goodfeature_track_nodelet.cpp:151
opencv_apps::GoodfeatureTrackNodelet
Definition: goodfeature_track_nodelet.cpp:90
image_transport::CameraSubscriber
opencv_apps::GoodfeatureTrackNodelet::reconfigure_server_
boost::shared_ptr< ReconfigureServer > reconfigure_server_
Definition: goodfeature_track_nodelet.cpp:134
image_transport::CameraSubscriber::shutdown
void shutdown()
PLUGINLIB_EXPORT_CLASS
PLUGINLIB_EXPORT_CLASS(opencv_apps::GoodfeatureTrackNodelet, nodelet::Nodelet)
opencv_apps::GoodfeatureTrackNodelet::imageCallbackWithInfo
void imageCallbackWithInfo(const sensor_msgs::ImageConstPtr &msg, const sensor_msgs::CameraInfoConstPtr &cam_info)
Definition: goodfeature_track_nodelet.cpp:158
boost::placeholders::_2
boost::arg< 2 > _2
Definition: nodelet.cpp:45
goodfeature_track::GoodfeatureTrackNodelet::onInit
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method.
Definition: goodfeature_track_nodelet.cpp:257
opencv_apps::Nodelet::always_subscribe_
bool always_subscribe_
A flag to disable watching mechanism and always subscribe input topics. It can be specified via ~alwa...
Definition: nodelet.h:300
ROS_WARN
#define ROS_WARN(...)
opencv_apps::GoodfeatureTrackNodelet::doWork
void doWork(const sensor_msgs::ImageConstPtr &msg, const std::string &input_frame_from_msg)
Definition: goodfeature_track_nodelet.cpp:173
image_transport::Publisher::publish
void publish(const sensor_msgs::Image &message) const
opencv_apps
Demo code to calculate moments.
Definition: nodelet.h:68
opencv_apps::GoodfeatureTrackNodelet::config_
Config config_
Definition: goodfeature_track_nodelet.cpp:133
image_transport.h
opencv_apps::GoodfeatureTrackNodelet::ReconfigureServer
dynamic_reconfigure::Server< Config > ReconfigureServer
Definition: goodfeature_track_nodelet.cpp:132
opencv_apps::GoodfeatureTrackNodelet::msg_pub_
ros::Publisher msg_pub_
Definition: goodfeature_track_nodelet.cpp:127
opencv_apps::GoodfeatureTrackNodelet::it_
boost::shared_ptr< image_transport::ImageTransport > it_
Definition: goodfeature_track_nodelet.cpp:129
nodelet::Nodelet
ros::Time
opencv_apps::GoodfeatureTrackNodelet::unsubscribe
void unsubscribe()
This method is called when publisher is unsubscribed by other nodes. Shut down subscribers in this me...
Definition: goodfeature_track_nodelet.cpp:278
image_transport::Publisher
opencv_apps::GoodfeatureTrackNodelet::prev_stamp_
ros::Time prev_stamp_
Definition: goodfeature_track_nodelet.cpp:138
cv_bridge.h
cv_bridge::CvImage
class_list_macros.hpp
opencv_apps::GoodfeatureTrackNodelet::debug_view_
bool debug_view_
Definition: goodfeature_track_nodelet.cpp:137
sensor_msgs::image_encodings::BGR8
const std::string BGR8
opencv_apps::GoodfeatureTrackNodelet::reconfigureCallback
void reconfigureCallback(Config &new_config, uint32_t level)
Definition: goodfeature_track_nodelet.cpp:145
cv_bridge::toCvShare
CvImageConstPtr toCvShare(const sensor_msgs::Image &source, const boost::shared_ptr< void const > &tracked_object, const std::string &encoding=std::string())
NODELET_INFO_STREAM
#define NODELET_INFO_STREAM(...)
opencv_apps::GoodfeatureTrackNodelet::queue_size_
int queue_size_
Definition: goodfeature_track_nodelet.cpp:136
opencv_apps::Nodelet::nh_
boost::shared_ptr< ros::NodeHandle > nh_
Shared pointer to nodehandle.
Definition: nodelet.h:272
opencv_apps::GoodfeatureTrackNodelet::onInit
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method.
Definition: goodfeature_track_nodelet.cpp:286
opencv_apps::Nodelet::onInit
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method.
Definition: nodelet.cpp:60
opencv_apps::Nodelet::pnh_
boost::shared_ptr< ros::NodeHandle > pnh_
Shared pointer to private nodehandle.
Definition: nodelet.h:277
NODELET_DEBUG
#define NODELET_DEBUG(...)
opencv_apps::GoodfeatureTrackNodelet::need_config_update_
static bool need_config_update_
Definition: goodfeature_track_nodelet.cpp:141
image_transport::Subscriber::shutdown
void shutdown()
goodfeature_track::GoodfeatureTrackNodelet
Definition: goodfeature_track_nodelet.cpp:254


opencv_apps
Author(s): Kei Okada
autogenerated on Fri May 23 2025 02:49:49