/* * Software License Agreement (BSD License) * * Point Cloud Library (PCL) - www.pointclouds.org * Copyright (c) 2010-2011, Willow Garage, Inc. * Copyright (c) 2012-, Open Perception, Inc. * * 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 copyright holder(s) 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. * * */ #pragma once #include #include #include #include #include #include #include #include namespace pcl { /** \brief Base class for Image file grabber. * \ingroup io */ class PCL_EXPORTS ImageGrabberBase : public Grabber { public: /** \brief Constructor taking a folder of depth+[rgb] images. * \param[in] directory Directory which contains an ordered set of images corresponding to an [RGB]D video, stored as TIFF, PNG, JPG, or PPM files. The naming convention is: frame_[timestamp]_["depth"/"rgb"].[extension] * \param[in] frames_per_second frames per second. If 0, start() functions like a trigger, publishing the next PCD in the list. * \param[in] repeat whether to play PCD file in an endless loop or not. * \param pclzf_mode */ ImageGrabberBase (const std::string& directory, float frames_per_second, bool repeat, bool pclzf_mode); ImageGrabberBase (const std::string& depth_directory, const std::string& rgb_directory, float frames_per_second, bool repeat); /** \brief Constructor taking a list of paths to PCD files, that are played in the order they appear in the list. * \param[in] depth_image_files Path to the depth image files files. * \param[in] frames_per_second frames per second. If 0, start() functions like a trigger, publishing the next PCD in the list. * \param[in] repeat whether to play PCD file in an endless loop or not. */ ImageGrabberBase (const std::vector& depth_image_files, float frames_per_second, bool repeat); /** \brief Virtual destructor. */ ~ImageGrabberBase () noexcept; /** \brief Starts playing the list of PCD files if frames_per_second is > 0. Otherwise it works as a trigger: publishes only the next PCD file in the list. */ void start () override; /** \brief Stops playing the list of PCD files if frames_per_second is > 0. Otherwise the method has no effect. */ void stop () override; /** \brief Triggers a callback with new data */ virtual void trigger (); /** \brief whether the grabber is started (publishing) or not. * \return true only if publishing. */ bool isRunning () const override; /** \return The name of the grabber */ std::string getName () const override; /** \brief Rewinds to the first PCD file in the list.*/ virtual void rewind (); /** \brief Returns the frames_per_second. 0 if grabber is trigger-based */ float getFramesPerSecond () const override; /** \brief Returns whether the repeat flag is on */ bool isRepeatOn () const; /** \brief Returns if the last frame is reached */ bool atLastFrame () const; /** \brief Returns the filename of the current indexed file */ std::string getCurrentDepthFileName () const; /** \brief Returns the filename of the previous indexed file * SDM: adding this back in, but is this useful, or confusing? */ std::string getPrevDepthFileName () const; /** \brief Get the depth filename at a particular index */ std::string getDepthFileNameAtIndex (std::size_t idx) const; /** \brief Query only the timestamp of an index, if it exists */ bool getTimestampAtIndex (std::size_t idx, std::uint64_t ×tamp) const; /** \brief Manually set RGB image files. * \param[in] rgb_image_files A vector of [tiff/png/jpg/ppm] files to use as input. There must be a 1-to-1 correspondence between these and the depth images you set */ void setRGBImageFiles (const std::vector& rgb_image_files); /** \brief Define custom focal length and center pixel. This will override ANY other setting of parameters for the duration of the grabber's life, whether by factory defaults or explicitly read from a frame_[timestamp].xml file. * \param[in] focal_length_x Horizontal focal length (fx) * \param[in] focal_length_y Vertical focal length (fy) * \param[in] principal_point_x Horizontal coordinates of the principal point (cx) * \param[in] principal_point_y Vertical coordinates of the principal point (cy) */ virtual void setCameraIntrinsics (const double focal_length_x, const double focal_length_y, const double principal_point_x, const double principal_point_y); /** \brief Get the current focal length and center pixel. If the intrinsics have been manually set with setCameraIntrinsics, this will return those values. Else, if start () has been called and the grabber has found a frame_[timestamp].xml file, this will return the most recent values read. Else, returns factory defaults. * \param[out] focal_length_x Horizontal focal length (fx) * \param[out] focal_length_y Vertical focal length (fy) * \param[out] principal_point_x Horizontal coordinates of the principal point (cx) * \param[out] principal_point_y Vertical coordinates of the principal point (cy) */ virtual void getCameraIntrinsics (double &focal_length_x, double &focal_length_y, double &principal_point_x, double &principal_point_y) const; /** \brief Define the units the depth data is stored in. * Defaults to mm (0.001), meaning a brightness of 1000 corresponds to 1 m*/ void setDepthImageUnits (float units); /** \brief Set the number of threads, if we wish to use OpenMP for quicker cloud population. * Note that for a standard (< 4 core) machine this is unlikely to yield a drastic speedup.*/ void setNumberOfThreads (unsigned int nr_threads = 0); protected: /** \brief Convenience function to see how many frames this consists of */ std::size_t numFrames () const; /** \brief Gets the cloud in ROS form at location idx */ bool getCloudAt (std::size_t idx, pcl::PCLPointCloud2 &blob, Eigen::Vector4f &origin, Eigen::Quaternionf &orientation) const; private: virtual void publish (const pcl::PCLPointCloud2& blob, const Eigen::Vector4f& origin, const Eigen::Quaternionf& orientation) const = 0; // to separate and hide the implementation from interface: PIMPL struct ImageGrabberImpl; ImageGrabberImpl* impl_; }; //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template class PointCloud; template class ImageGrabber : public ImageGrabberBase, public FileGrabber { public: using Ptr = shared_ptr; using ConstPtr = shared_ptr; ImageGrabber (const std::string& dir, float frames_per_second = 0, bool repeat = false, bool pclzf_mode = false); ImageGrabber (const std::string& depth_dir, const std::string& rgb_dir, float frames_per_second = 0, bool repeat = false); ImageGrabber (const std::vector& depth_image_files, float frames_per_second = 0, bool repeat = false); /** \brief Empty destructor */ ~ImageGrabber () noexcept {} // Inherited from FileGrabber const typename pcl::PointCloud::ConstPtr operator[] (std::size_t idx) const override; // Inherited from FileGrabber std::size_t size () const override; protected: void publish (const pcl::PCLPointCloud2& blob, const Eigen::Vector4f& origin, const Eigen::Quaternionf& orientation) const override; boost::signals2::signal::ConstPtr&)>* signal_; }; //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template ImageGrabber::ImageGrabber (const std::string& dir, float frames_per_second, bool repeat, bool pclzf_mode) : ImageGrabberBase (dir, frames_per_second, repeat, pclzf_mode) { signal_ = createSignal::ConstPtr&)>(); } //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template ImageGrabber::ImageGrabber (const std::string& depth_dir, const std::string& rgb_dir, float frames_per_second, bool repeat) : ImageGrabberBase (depth_dir, rgb_dir, frames_per_second, repeat) { signal_ = createSignal::ConstPtr&)>(); } //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template ImageGrabber::ImageGrabber (const std::vector& depth_image_files, float frames_per_second, bool repeat) : ImageGrabberBase (depth_image_files, frames_per_second, repeat), signal_ () { signal_ = createSignal::ConstPtr&)>(); } //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template const typename pcl::PointCloud::ConstPtr ImageGrabber::operator[] (std::size_t idx) const { pcl::PCLPointCloud2 blob; Eigen::Vector4f origin; Eigen::Quaternionf orientation; getCloudAt (idx, blob, origin, orientation); typename pcl::PointCloud::Ptr cloud (new pcl::PointCloud ()); pcl::fromPCLPointCloud2 (blob, *cloud); cloud->sensor_origin_ = origin; cloud->sensor_orientation_ = orientation; return (cloud); } /////////////////////////////////////////////////////////////////////////////////////////////////////////////// template std::size_t ImageGrabber::size () const { return (numFrames ()); } //////////////////////////////////////////////////////////////////////////////////////////////////////////////// template void ImageGrabber::publish (const pcl::PCLPointCloud2& blob, const Eigen::Vector4f& origin, const Eigen::Quaternionf& orientation) const { typename pcl::PointCloud::Ptr cloud (new pcl::PointCloud ()); pcl::fromPCLPointCloud2 (blob, *cloud); cloud->sensor_origin_ = origin; cloud->sensor_orientation_ = orientation; signal_->operator () (cloud); } }