diff --git a/CMakeLists.txt b/CMakeLists.txt index 2086e71a..5e197e8b 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -26,6 +26,7 @@ generate_dynamic_reconfigure_options( # cfg/CamShift.cfg cfg/FBackFlow.cfg + cfg/LineSegmentDetector.cfg cfg/LKFlow.cfg cfg/PeopleDetect.cfg cfg/PhaseCorr.cfg @@ -87,6 +88,11 @@ catkin_package(CATKIN_DEPENDS std_msgs include_directories(include ${catkin_INCLUDE_DIRS} ${OpenCV_INCLUDE_DIRS}) ## Declare a cpp library +if(OpenCV_VERSION VERSION_LESS "3.0") +else() + # 3.0 version + list(APPEND ${PROJECT_NAME}_EXTRA_FILES src/nodelet/line_segment_detector_nodelet.cpp) +endif() if(OPENCV_HAVE_OPTFLOW) list(APPEND ${PROJECT_NAME}_EXTRA_FILES src/nodelet/simple_flow_nodelet.cpp) endif() @@ -258,7 +264,10 @@ opencv_apps_add_nodelet(fback_flow fback_flow/fback_flow src/nodelet/fback_flow_ # ./letter_recog.cpp opencv_apps_add_nodelet(lk_flow lk_flow/lk_flow src/nodelet/lk_flow_nodelet.cpp) # ./lkdemo.cpp # ./logistic_regression.cpp -# ./lsd_lines.cpp +if(OpenCV_VERSION VERSION_LESS "3.0") +else() + opencv_apps_add_nodelet(line_segment_detector line_segment_detector/line_segment_detector src/nodelet/line_segment_detector_nodelet.cpp) # ./lsd_lines.cpp +endif() # ./mask_tmpl.cpp # ./matchmethod_orb_akaze_brisk.cpp # ./minarea.cpp diff --git a/cfg/LineSegmentDetector.cfg b/cfg/LineSegmentDetector.cfg new file mode 100755 index 00000000..847875ab --- /dev/null +++ b/cfg/LineSegmentDetector.cfg @@ -0,0 +1,55 @@ +#! /usr/bin/env python +# Software License Agreement (BSD License) +# +# Copyright (c) 2016, JSK Lab. +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions +# are met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following +# disclaimer in the documentation and/or other materials provided +# with the distribution. +# * Neither the name of Kei Okada nor the names of its +# contributors may be used to endorse or promote products derived +# from this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. + +PACKAGE='line_segment_detector' + +from dynamic_reconfigure.parameter_generator_catkin import * + +gen = ParameterGenerator() +gen.add("use_camera_info", bool_t, 0, "Indicates that the camera_info topic should be subscribed to to get the default input_frame_id. Otherwise the frame from the image message will be used.", True) + +lsd_refine_type = gen.enum([ gen.const("REFINE_NONE", int_t, 0, "No refinement applied"), + gen.const("REFINE_STD", int_t, 1, "Standard refinement is applied. E.g. breaking arches into smaller straighter line approximations."), + gen.const("REFINE_ADV", int_t, 2, "Advanced refinement. Number of false alarms is calculated, lines are refined through increase of precision, decrement in size, etc."), +], "An enum for Line Segment Detector Modes") +gen.add("lsd_refine_type", int_t, 0, "Line Segment Detector Modes", 0, 0, 2, edit_method=lsd_refine_type) +gen.add("lsd_scale", double_t, 0, "The scale of the image that will be used to find the lines. Range (0..1]", 0.8, 0.1, 1.0); +gen.add("lsd_sigma_scale", double_t, 0, "Sigma for Gaussian filter. It is computed as sigma = _sigma_scale/_scale", 0.6, 0.1, 10.0); +gen.add("lsd_quant", double_t, 0, "Bound to the quantization error on the gradient norm", 2.0, 0.0, 100.0); +gen.add("lsd_angle_threshold", double_t, 0, "Gradient angle tolerance in degrees", 22.5, 0.1, 179.0); +gen.add("lsd_log_eps", double_t, 0, "Detection threshold: -log10(NFA) > log_eps. Used only when advancent refinement is chosen", 0.0, 0.0, 360.0); +gen.add("lsd_density_threshold", double_t, 0, "Minimal density of aligned region points in the enclosing rectangle", 0.7, 0.0, 0.9); +gen.add("lsd_n_bins", int_t, 0, "Number of bins in pseudo-ordering of gradient modulus", 1024, 1, 10000) +gen.add("lsd_line_length_threshold", double_t, 0, "Threshold of line length.", 100.0, 0.0, 1000.0) + +exit(gen.generate(PACKAGE, "line_segment_detector", "LineSegmentDetector")) diff --git a/launch/line_segment_detector.launch b/launch/line_segment_detector.launch new file mode 100644 index 00000000..0d08d2de --- /dev/null +++ b/launch/line_segment_detector.launch @@ -0,0 +1,34 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/nodelet_plugins.xml b/nodelet_plugins.xml index a6ca40ca..db4e08e9 100644 --- a/nodelet_plugins.xml +++ b/nodelet_plugins.xml @@ -52,6 +52,10 @@ Nodelet to demonstrates dense optical flow algorithm by Gunnar Farneback + + Nodelet to detector line segments + + Nodelet to calculate Lukas-Kanade optical flow diff --git a/src/nodelet/line_segment_detector_nodelet.cpp b/src/nodelet/line_segment_detector_nodelet.cpp new file mode 100644 index 00000000..6cbbec0e --- /dev/null +++ b/src/nodelet/line_segment_detector_nodelet.cpp @@ -0,0 +1,258 @@ +// -*- coding:utf-8-unix; mode: c++; indent-tabs-mode: nil; c-basic-offset: 2; -*- +/********************************************************************* +* Software License Agreement (BSD License) +* +* Copyright (c) 2016, JSK Lab. +* All rights reserved. +* +* Redistribution and use in source and binary forms, with or without +* modification, are permitted provided that the following conditions +* are met: +* +* * Redistributions of source code must retain the above copyright +* notice, this list of conditions and the following disclaimer. +* * Redistributions in binary form must reproduce the above +* copyright notice, this list of conditions and the following +* disclaimer in the documentation and/or other materials provided +* with the distribution. +* * Neither the name of the Kei Okada nor the names of its +* contributors may be used to endorse or promote products derived +* from this software without specific prior written permission. +* +* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, +* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT +* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN +* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +* POSSIBILITY OF SUCH DAMAGE. +*********************************************************************/ + +/** + * @file https://github.com/opencv/opencv/blob/master/samples/cpp/lsd_lines.cpp + * @brief Sample code showing how to detect line segments using the LineSegmentDetector + * @author OpenCV team + */ + +#include +#include "opencv_apps/nodelet.h" +#include +#include +#include + +#include +#include + +#include +#include "opencv_apps/Line.h" +#include "opencv_apps/LineArrayStamped.h" +#include "opencv_apps/LineSegmentDetectorConfig.h" + + +namespace line_segment_detector { +class LineSegmentDetectorNodelet : public opencv_apps::Nodelet +{ + image_transport::Publisher img_pub_; + image_transport::Subscriber img_sub_; + image_transport::CameraSubscriber cam_sub_; + ros::Publisher msg_pub_; + + boost::shared_ptr it_; + + typedef line_segment_detector::LineSegmentDetectorConfig Config; + typedef dynamic_reconfigure::Server ReconfigureServer; + Config config_; + boost::shared_ptr reconfigure_server_; + + cv::Ptr lsd_; + + bool debug_view_; + + int lsd_refine_; + double lsd_scale_; + double lsd_sigma_scale_; + double lsd_quant_; + double lsd_angle_threshold_; + double lsd_log_eps_; + double lsd_density_threshold_; + int lsd_n_bins_; + double lsd_line_length_threshold_; + + std::string window_name_; + static bool need_config_update_; + + boost::mutex mutex_; + + void updateLSD() { + lsd_ = cv::createLineSegmentDetector(lsd_refine_, lsd_scale_, + lsd_sigma_scale_, lsd_quant_, + lsd_angle_threshold_, lsd_log_eps_, + lsd_density_threshold_, lsd_n_bins_); + } + + void reconfigureCallback(Config &new_config, uint32_t level) + { + boost::mutex::scoped_lock lock (mutex_); + config_ = new_config; + + lsd_refine_ = config_.lsd_refine_type; + lsd_scale_ = config_.lsd_scale; + lsd_sigma_scale_ = config_.lsd_sigma_scale; + lsd_angle_threshold_ = config_.lsd_angle_threshold; + lsd_log_eps_ = config_.lsd_log_eps; + lsd_density_threshold_ = config_.lsd_density_threshold; + lsd_n_bins_ = config_.lsd_n_bins; + lsd_line_length_threshold_ = config_.lsd_line_length_threshold; + + updateLSD(); + } + + const std::string &frameWithDefault(const std::string &frame, const std::string &image_frame) + { + if (frame.empty()) + return image_frame; + return frame; + } + + void imageCallbackWithInfo(const sensor_msgs::ImageConstPtr& msg, const sensor_msgs::CameraInfoConstPtr& cam_info) + { + do_work(msg, cam_info->header.frame_id); + } + + void imageCallback(const sensor_msgs::ImageConstPtr& msg) + { + do_work(msg, msg->header.frame_id); + } + + static void trackbarCallback( int, void* ) + { + need_config_update_ = true; + } + + void do_work(const sensor_msgs::ImageConstPtr& msg, const std::string input_frame_from_msg) + { + boost::mutex::scoped_lock lock (mutex_); + // Work on the image. + try + { + // Convert the image into something opencv can handle. + cv::Mat frame = cv_bridge::toCvShare(msg, sensor_msgs::image_encodings::BGR8)->image; + + // Do the work + cv::Mat src_gray; + /// Convert it to gray + if ( frame.channels() > 1 ) { + cv::cvtColor( frame, src_gray, cv::COLOR_RGB2GRAY ); + } else { + src_gray = frame; + } + + /// Create window + if( debug_view_) { + cv::namedWindow( window_name_, cv::WINDOW_AUTOSIZE ); + } + + cv::Mat line_image; + + std::vector lines; + std::vector widths; + std::vector precs; + std::vector nfas; + lsd_->detect(src_gray, lines, widths, precs, nfas); + line_image = cv::Mat::zeros(frame.rows, frame.cols, CV_8UC1); + + // Messages + opencv_apps::LineArrayStamped lines_msg; + lines_msg.header = msg->header; + + // draw lines + for(size_t i = 0; i < lines.size(); ++i ) { + int x1 = lines[i][0]; + int y1 = lines[i][1]; + int x2 = lines[i][2]; + int y2 = lines[i][3]; + double line_length = sqrt((x2 - x1) * (x2 - x1) + (y2 - y1) * (y2 - y1)); + + if (line_length >= lsd_line_length_threshold_) { + cv::line(line_image, cv::Point(x1, y1), cv::Point(x2, y2), 255); + } + + opencv_apps::Line line_msg; + line_msg.pt1.x = x1; + line_msg.pt1.y = y1; + line_msg.pt2.x = x2; + line_msg.pt2.y = y2; + lines_msg.lines.push_back(line_msg); + } + + if( debug_view_) { + cv::imshow( window_name_, line_image ); + int c = cv::waitKey(1); + } + + // Publish the image. + sensor_msgs::Image::Ptr out_img = cv_bridge::CvImage(msg->header, sensor_msgs::image_encodings::MONO8, line_image).toImageMsg(); + img_pub_.publish(out_img); + // Publish the detected lines. + msg_pub_.publish(lines_msg); + } + catch (cv::Exception &e) + { + NODELET_ERROR("Image processing error: %s %s %s %i", e.err.c_str(), e.func.c_str(), e.file.c_str(), e.line); + } + + } + + void subscribe() + { + NODELET_DEBUG("Subscribing to image topic."); + if (config_.use_camera_info) + cam_sub_ = it_->subscribeCamera("image", 3, &LineSegmentDetectorNodelet::imageCallbackWithInfo, this); + else + img_sub_ = it_->subscribe("image", 3, &LineSegmentDetectorNodelet::imageCallback, this); + } + + void unsubscribe() + { + NODELET_DEBUG("Unsubscribing from image topic."); + img_sub_.shutdown(); + cam_sub_.shutdown(); + } + +public: + virtual void onInit() + { + Nodelet::onInit(); + it_ = boost::shared_ptr(new image_transport::ImageTransport(*nh_)); + + pnh_->param("debug_view", debug_view_, false); + + if (debug_view_) { + always_subscribe_ = true; + } + + window_name_ = "Line Segment Detector Demo"; + + reconfigure_server_ = boost::make_shared >(*pnh_); + dynamic_reconfigure::Server::CallbackType f = + boost::bind(&LineSegmentDetectorNodelet::reconfigureCallback, this, _1, _2); + reconfigure_server_->setCallback(f); + + updateLSD(); + + img_pub_ = advertiseImage(*pnh_, "image", 1); + msg_pub_ = advertise(*pnh_, "lines", 1); + + onInitPostProcess(); + } +}; +bool LineSegmentDetectorNodelet::need_config_update_ = false; +} + +#include +PLUGINLIB_EXPORT_CLASS(line_segment_detector::LineSegmentDetectorNodelet, nodelet::Nodelet); diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index b73ce322..683507e7 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -35,6 +35,10 @@ add_rostest(test-goodfeature_track.test ARGS gui:=false) add_rostest(test-camshift.test ARGS gui:=false) add_rostest(test-fback_flow.test ARGS gui:=false) add_rostest(test-lk_flow.test ARGS gui:=false) +if(OpenCV_VERSION VERSION_LESS "3.0") +else() + add_rostest(test-line_segment_detector.test ARGS gui:=false) +endif() add_rostest(test-people_detect.test ARGS gui:=false) add_rostest(test-phase_corr.test ARGS gui:=false) add_rostest(test-segment_objects.test ARGS gui:=false) diff --git a/test/test-line_segment_detector.test b/test/test-line_segment_detector.test new file mode 100644 index 00000000..dab7007d --- /dev/null +++ b/test/test-line_segment_detector.test @@ -0,0 +1,35 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +