contrast_stretch_nodelet.cpp
Go to the documentation of this file.
00001 // *****************************************************************************
00002 //
00003 // Copyright (c) 2014, Southwest Research Institute® (SwRI®)
00004 // All rights reserved.
00005 //
00006 // Redistribution and use in source and binary forms, with or without
00007 // modification, are permitted provided that the following conditions are met:
00008 //     * Redistributions of source code must retain the above copyright
00009 //       notice, this list of conditions and the following disclaimer.
00010 //     * Redistributions in binary form must reproduce the above copyright
00011 //       notice, this list of conditions and the following disclaimer in the
00012 //       documentation and/or other materials provided with the distribution.
00013 //     * Neither the name of Southwest Research Institute® (SwRI®) nor the
00014 //       names of its contributors may be used to endorse or promote products
00015 //       derived from this software without specific prior written permission.
00016 //
00017 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
00018 // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
00019 // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
00020 // ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
00021 // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
00022 // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
00023 // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
00024 // ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
00025 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
00026 // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
00027 //
00028 // *****************************************************************************
00029 
00030 #include <string>
00031 
00032 #include <opencv2/core/core.hpp>
00033 #include <opencv2/highgui/highgui.hpp>
00034 #include <opencv2/imgproc/imgproc.hpp>
00035 
00036 #include <ros/ros.h>
00037 #include <nodelet/nodelet.h>
00038 #include <image_transport/image_transport.h>
00039 #include <sensor_msgs/image_encodings.h>
00040 #include <sensor_msgs/Image.h>
00041 #include <cv_bridge/cv_bridge.h>
00042 #include <swri_image_util/image_normalization.h>
00043 
00044 #include <swri_math_util/math_util.h>
00045 
00046 namespace swri_image_util
00047 {
00048   class ContrastStretchNodelet : public nodelet::Nodelet
00049   {
00050   public:
00051     ContrastStretchNodelet() :
00052       bins_(8),
00053       max_min_(0.0),
00054       min_max_(0.0),
00055       over_exposure_threshold_(255.0),
00056       over_exposure_dilation_(3)
00057     {
00058     }
00059 
00060     ~ContrastStretchNodelet()
00061     {
00062     }
00063 
00064     void onInit()
00065     {
00066       ros::NodeHandle &node = getNodeHandle();
00067       ros::NodeHandle &priv = getPrivateNodeHandle();
00068 
00069       priv.param("bins", bins_, bins_);
00070       priv.param("max_min", max_min_, max_min_);
00071       priv.param("min_max", min_max_, min_max_);
00072       priv.param("over_exposure_threshold", over_exposure_threshold_, over_exposure_threshold_);
00073       priv.param("over_exposure_dilation", over_exposure_dilation_, over_exposure_dilation_);
00074       
00075       std::string mask;
00076       priv.param("mask", mask, std::string(""));
00077       if (!mask.empty())
00078       {
00079         mask_ = cv::imread(mask, 0);
00080       }
00081 
00082       image_transport::ImageTransport it(node);
00083       image_pub_ = it.advertise("normalized_image", 1);
00084       image_sub_ = it.subscribe("image", 1, &ContrastStretchNodelet::ImageCallback, this);
00085     }
00086 
00087     void ImageCallback(const sensor_msgs::ImageConstPtr& image)
00088     {
00089       cv_bridge::CvImagePtr cv_image = cv_bridge::toCvCopy(image);
00090 
00091       if (mask_.empty())
00092       {
00093         mask_ = cv::Mat::ones(cv_image->image.size(), CV_8U);
00094       }
00095       else if (mask_.rows != cv_image->image.rows || mask_.cols != cv_image->image.cols)
00096       {
00097         cv::resize(mask_, mask_, cv_image->image.size(), 1.0, 1.0, cv::INTER_NEAREST);
00098       }
00099 
00100       cv::Mat mask;
00101 
00102       if (over_exposure_threshold_ < 255 && over_exposure_threshold_ > 0)
00103       {
00104         cv::Mat over_exposed = cv_image->image > over_exposure_threshold_;
00105         cv::Mat element = cv::getStructuringElement(
00106           cv::MORPH_ELLIPSE,
00107           cv::Size(2 * over_exposure_dilation_ + 1, 2 * over_exposure_dilation_ + 1 ),
00108           cv::Point(over_exposure_dilation_, over_exposure_dilation_ ));
00109         cv::dilate(over_exposed, over_exposed, element);
00110         
00111         mask = mask_.clone();
00112         mask.setTo(0, over_exposed);
00113       }
00114       else
00115       {
00116         mask = mask_;
00117       }
00118 
00119       swri_image_util::ContrastStretch(bins_, cv_image->image, cv_image->image, mask, max_min_, min_max_);
00120 
00121       image_pub_.publish(cv_image->toImageMsg());
00122     }
00123 
00124   private:
00125     int32_t bins_;
00126     double max_min_;
00127     double min_max_;
00128     double over_exposure_threshold_;
00129     int32_t over_exposure_dilation_;
00130     
00131     cv::Mat mask_;
00132 
00133     image_transport::Subscriber image_sub_;
00134     image_transport::Publisher image_pub_;
00135 
00136   };
00137 }
00138 
00139 // Register nodelet plugin
00140 #include <swri_nodelet/class_list_macros.h>
00141 SWRI_NODELET_EXPORT_CLASS(swri_image_util, ContrastStretchNodelet)


swri_image_util
Author(s): Kris Kozak
autogenerated on Thu Jun 6 2019 20:34:52