smoothing_nodelet.cpp
Go to the documentation of this file.
1 // -*- mode: c++ -*-
2 /*********************************************************************
3  * Software License Agreement (BSD License)
4  *
5  * Copyright (c) 2016, JSK Lab
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  *
12  * * Redistributions of source code must retain the above copyright
13  * notice, this list of conditions and the following disclaimer.
14  * * Redistributions in binary form must reproduce the above
15  * copyright notice, this list of conditions and the following
16  * disclaimer in the documentation and/or other materials provided
17  * with the distribution.
18  * * Neither the name of the JSK Lab nor the names of its
19  * contributors may be used to endorse or promote products derived
20  * from this software without specific prior written permission.
21  *
22  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
23  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
24  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
25  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
26  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
27  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
28  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
30  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
32  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33  * POSSIBILITY OF SUCH DAMAGE.
34  *********************************************************************/
35 
36 // https://github.com/opencv/opencv/blob/2.4/samples/cpp/tutorial_code/ImgProc/Smoothing.cpp
43 #include <ros/ros.h>
44 #include "opencv_apps/nodelet.h"
47 #include <cv_bridge/cv_bridge.h>
48 
49 #include <iostream>
50 #include <vector>
51 
52 #include "opencv2/imgproc/imgproc.hpp"
53 #include "opencv2/highgui/highgui.hpp"
54 #include "opencv2/features2d/features2d.hpp"
55 
56 #include <dynamic_reconfigure/server.h>
57 #include "opencv_apps/SmoothingConfig.h"
58 
61 
62 namespace opencv_apps
63 {
65 {
70 
72 
73  typedef opencv_apps::SmoothingConfig Config;
74  typedef dynamic_reconfigure::Server<Config> ReconfigureServer;
75  Config config_;
77 
81 
82  std::string window_name_;
83  static bool need_config_update_;
84 
86 
87  void reconfigureCallback(Config& new_config, uint32_t level)
88  {
89  config_ = new_config;
90  kernel_size_ = (config_.kernel_size / 2) * 2 + 1; // kernel_size must be odd number
91  }
92 
93  const std::string& frameWithDefault(const std::string& frame, const std::string& image_frame)
94  {
95  if (frame.empty())
96  return image_frame;
97  return frame;
98  }
99 
100  void imageCallbackWithInfo(const sensor_msgs::ImageConstPtr& msg, const sensor_msgs::CameraInfoConstPtr& cam_info)
101  {
102  doWork(msg, cam_info->header.frame_id);
103  }
104 
105  void imageCallback(const sensor_msgs::ImageConstPtr& msg)
106  {
107  doWork(msg, msg->header.frame_id);
108  }
109 
110  static void trackbarCallback(int /*unused*/, void* /*unused*/)
111  {
112  need_config_update_ = true;
113  }
114 
115  void doWork(const sensor_msgs::ImageConstPtr& msg, const std::string& input_frame_from_msg)
116  {
117  // Work on the image.
118  try
119  {
120  // Convert the image into something opencv can handle.
121  cv::Mat in_image = cv_bridge::toCvShare(msg, sensor_msgs::image_encodings::BGR8)->image;
122 
123  if (debug_view_)
124  {
126  char kernel_label[] = "Kernel Size : ";
127 
128  cv::namedWindow(window_name_, cv::WINDOW_AUTOSIZE);
129  cv::createTrackbar(kernel_label, window_name_, &kernel_size_, MAX_KERNEL_LENGTH, trackbarCallback);
130  if (need_config_update_)
131  {
132  kernel_size_ = (kernel_size_ / 2) * 2 + 1; // kernel_size must be odd number
133  config_.kernel_size = kernel_size_;
134  reconfigure_server_->updateConfig(config_);
135  need_config_update_ = false;
136  }
137  }
138 
139  cv::Mat out_image = in_image.clone();
140  int i = kernel_size_;
141  switch (config_.filter_type)
142  {
143  case opencv_apps::Smoothing_Homogeneous_Blur:
144  {
146  ROS_DEBUG_STREAM("Applying Homogeneous blur with kernel size " << i);
147  cv::blur(in_image, out_image, cv::Size(i, i), cv::Point(-1, -1));
148  break;
149  }
150  case opencv_apps::Smoothing_Gaussian_Blur:
151  {
153  ROS_DEBUG_STREAM("Applying Gaussian blur with kernel size " << i);
154  cv::GaussianBlur(in_image, out_image, cv::Size(i, i), 0, 0);
155  break;
156  }
157  case opencv_apps::Smoothing_Median_Blur:
158  {
160  ROS_DEBUG_STREAM("Applying Median blur with kernel size " << i);
161  cv::medianBlur(in_image, out_image, i);
162  break;
163  }
164  case opencv_apps::Smoothing_Bilateral_Filter:
165  {
167  ROS_DEBUG_STREAM("Applying Bilateral blur with kernel size " << i);
168  cv::bilateralFilter(in_image, out_image, i, i * 2, i / 2);
169  break;
170  }
171  }
172 
173  //-- Show what you got
174  if (debug_view_)
175  {
176  cv::imshow(window_name_, out_image);
177  int c = cv::waitKey(1);
178  }
179 
180  // Publish the image.
181  sensor_msgs::Image::Ptr out_img = cv_bridge::CvImage(msg->header, "bgr8", out_image).toImageMsg();
182  img_pub_.publish(out_img);
183  }
184  catch (cv::Exception& e)
185  {
186  NODELET_ERROR("Image processing error: %s %s %s %i", e.err.c_str(), e.func.c_str(), e.file.c_str(), e.line);
187  }
188 
189  prev_stamp_ = msg->header.stamp;
190  }
191 
192  void subscribe() // NOLINT(modernize-use-override)
193  {
194  NODELET_DEBUG("Subscribing to image topic.");
195  if (config_.use_camera_info)
196  cam_sub_ = it_->subscribeCamera("image", queue_size_, &SmoothingNodelet::imageCallbackWithInfo, this);
197  else
198  img_sub_ = it_->subscribe("image", queue_size_, &SmoothingNodelet::imageCallback, this);
199  }
200 
201  void unsubscribe() // NOLINT(modernize-use-override)
202  {
203  NODELET_DEBUG("Unsubscribing from image topic.");
204  img_sub_.shutdown();
205  cam_sub_.shutdown();
206  }
207 
208 public:
209  virtual void onInit() // NOLINT(modernize-use-override)
210  {
211  Nodelet::onInit();
213 
214  pnh_->param("queue_size", queue_size_, 3);
215  pnh_->param("debug_view", debug_view_, false);
216  if (debug_view_)
217  {
218  always_subscribe_ = true;
219  }
220  prev_stamp_ = ros::Time(0, 0);
221 
222  window_name_ = "Smoothing Demo";
223  kernel_size_ = 7;
224 
225  reconfigure_server_ = boost::make_shared<dynamic_reconfigure::Server<Config> >(*pnh_);
226  dynamic_reconfigure::Server<Config>::CallbackType f =
227  boost::bind(&SmoothingNodelet::reconfigureCallback, this, _1, _2);
228  reconfigure_server_->setCallback(f);
229 
230  img_pub_ = advertiseImage(*pnh_, "image", 1);
231 
233  }
234 };
236 } // namespace opencv_apps
237 
238 namespace smoothing
239 {
241 {
242 public:
243  virtual void onInit() // NOLINT(modernize-use-override)
244  {
245  ROS_WARN("DeprecationWarning: Nodelet smoothing/smoothing is deprecated, "
246  "and renamed to opencv_apps/smoothing.");
248  }
249 };
250 } // namespace smoothing
251 
CvImageConstPtr toCvShare(const sensor_msgs::ImageConstPtr &source, const std::string &encoding=std::string())
int MAX_KERNEL_LENGTH
Global Variables.
#define NODELET_ERROR(...)
static void trackbarCallback(int, void *)
void imageCallback(const sensor_msgs::ImageConstPtr &msg)
f
dynamic_reconfigure::Server< Config > ReconfigureServer
Nodelet to automatically subscribe/unsubscribe topics according to subscription of advertised topics...
Definition: nodelet.h:70
void unsubscribe()
This method is called when publisher is unsubscribed by other nodes. Shut down subscribers in this me...
void reconfigureCallback(Config &new_config, uint32_t level)
Demo code to calculate moments.
Definition: nodelet.h:48
PLUGINLIB_EXPORT_CLASS(opencv_apps::SmoothingNodelet, nodelet::Nodelet)
const std::string & frameWithDefault(const std::string &frame, const std::string &image_frame)
boost::shared_ptr< ros::NodeHandle > pnh_
Shared pointer to private nodehandle.
Definition: nodelet.h:250
#define ROS_WARN(...)
opencv_apps::SmoothingConfig Config
bool always_subscribe_
A flag to disable watching mechanism and always subscribe input topics. It can be specified via ~alwa...
Definition: nodelet.h:273
virtual void onInitPostProcess()
Post processing of initialization of nodelet. You need to call this method in order to use always_sub...
Definition: nodelet.cpp:57
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method...
Definition: nodelet.cpp:40
void publish(const sensor_msgs::Image &message) const
void imageCallbackWithInfo(const sensor_msgs::ImageConstPtr &msg, const sensor_msgs::CameraInfoConstPtr &cam_info)
void subscribe()
This method is called when publisher is subscribed by other nodes. Set up subscribers in this method...
image_transport::Publisher img_pub_
boost::shared_ptr< ros::NodeHandle > nh_
Shared pointer to nodehandle.
Definition: nodelet.h:245
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method...
image_transport::CameraSubscriber cam_sub_
#define ROS_DEBUG_STREAM(args)
image_transport::Subscriber img_sub_
boost::shared_ptr< image_transport::ImageTransport > it_
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:180
void doWork(const sensor_msgs::ImageConstPtr &msg, const std::string &input_frame_from_msg)
virtual void onInit()
Initialize nodehandles nh_ and pnh_. Subclass should call this method in its onInit method...
#define NODELET_DEBUG(...)
sensor_msgs::ImagePtr toImageMsg() const
boost::shared_ptr< ReconfigureServer > reconfigure_server_


opencv_apps
Author(s): Kei Okada
autogenerated on Wed Apr 24 2019 03:00:17