seam_finders.hpp 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. /*M///////////////////////////////////////////////////////////////////////////////////////
  2. //
  3. // IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
  4. //
  5. // By downloading, copying, installing or using the software you agree to this license.
  6. // If you do not agree to this license, do not download, install,
  7. // copy or use the software.
  8. //
  9. //
  10. // License Agreement
  11. // For Open Source Computer Vision Library
  12. //
  13. // Copyright (C) 2000-2008, Intel Corporation, all rights reserved.
  14. // Copyright (C) 2009, Willow Garage Inc., all rights reserved.
  15. // Third party copyrights are property of their respective owners.
  16. //
  17. // Redistribution and use in source and binary forms, with or without modification,
  18. // are permitted provided that the following conditions are met:
  19. //
  20. // * Redistribution's of source code must retain the above copyright notice,
  21. // this list of conditions and the following disclaimer.
  22. //
  23. // * Redistribution's in binary form must reproduce the above copyright notice,
  24. // this list of conditions and the following disclaimer in the documentation
  25. // and/or other materials provided with the distribution.
  26. //
  27. // * The name of the copyright holders may not be used to endorse or promote products
  28. // derived from this software without specific prior written permission.
  29. //
  30. // This software is provided by the copyright holders and contributors "as is" and
  31. // any express or implied warranties, including, but not limited to, the implied
  32. // warranties of merchantability and fitness for a particular purpose are disclaimed.
  33. // In no event shall the Intel Corporation or contributors be liable for any direct,
  34. // indirect, incidental, special, exemplary, or consequential damages
  35. // (including, but not limited to, procurement of substitute goods or services;
  36. // loss of use, data, or profits; or business interruption) however caused
  37. // and on any theory of liability, whether in contract, strict liability,
  38. // or tort (including negligence or otherwise) arising in any way out of
  39. // the use of this software, even if advised of the possibility of such damage.
  40. //
  41. //M*/
  42. #ifndef __OPENCV_STITCHING_SEAM_FINDERS_HPP__
  43. #define __OPENCV_STITCHING_SEAM_FINDERS_HPP__
  44. #include <set>
  45. #include "opencv2/core.hpp"
  46. #include "opencv2/opencv_modules.hpp"
  47. namespace cv {
  48. namespace detail {
  49. //! @addtogroup stitching_seam
  50. //! @{
  51. /** @brief Base class for a seam estimator.
  52. */
  53. class CV_EXPORTS SeamFinder
  54. {
  55. public:
  56. virtual ~SeamFinder() {}
  57. /** @brief Estimates seams.
  58. @param src Source images
  59. @param corners Source image top-left corners
  60. @param masks Source image masks to update
  61. */
  62. virtual void find(const std::vector<UMat> &src, const std::vector<Point> &corners,
  63. std::vector<UMat> &masks) = 0;
  64. };
  65. /** @brief Stub seam estimator which does nothing.
  66. */
  67. class CV_EXPORTS NoSeamFinder : public SeamFinder
  68. {
  69. public:
  70. void find(const std::vector<UMat>&, const std::vector<Point>&, std::vector<UMat>&) {}
  71. };
  72. /** @brief Base class for all pairwise seam estimators.
  73. */
  74. class CV_EXPORTS PairwiseSeamFinder : public SeamFinder
  75. {
  76. public:
  77. virtual void find(const std::vector<UMat> &src, const std::vector<Point> &corners,
  78. std::vector<UMat> &masks);
  79. protected:
  80. void run();
  81. /** @brief Resolves masks intersection of two specified images in the given ROI.
  82. @param first First image index
  83. @param second Second image index
  84. @param roi Region of interest
  85. */
  86. virtual void findInPair(size_t first, size_t second, Rect roi) = 0;
  87. std::vector<UMat> images_;
  88. std::vector<Size> sizes_;
  89. std::vector<Point> corners_;
  90. std::vector<UMat> masks_;
  91. };
  92. /** @brief Voronoi diagram-based seam estimator.
  93. */
  94. class CV_EXPORTS VoronoiSeamFinder : public PairwiseSeamFinder
  95. {
  96. public:
  97. virtual void find(const std::vector<UMat> &src, const std::vector<Point> &corners,
  98. std::vector<UMat> &masks);
  99. virtual void find(const std::vector<Size> &size, const std::vector<Point> &corners,
  100. std::vector<UMat> &masks);
  101. private:
  102. void findInPair(size_t first, size_t second, Rect roi);
  103. };
  104. class CV_EXPORTS DpSeamFinder : public SeamFinder
  105. {
  106. public:
  107. enum CostFunction { COLOR, COLOR_GRAD };
  108. DpSeamFinder(CostFunction costFunc = COLOR);
  109. CostFunction costFunction() const { return costFunc_; }
  110. void setCostFunction(CostFunction val) { costFunc_ = val; }
  111. virtual void find(const std::vector<UMat> &src, const std::vector<Point> &corners,
  112. std::vector<UMat> &masks);
  113. private:
  114. enum ComponentState
  115. {
  116. FIRST = 1, SECOND = 2, INTERS = 4,
  117. INTERS_FIRST = INTERS | FIRST,
  118. INTERS_SECOND = INTERS | SECOND
  119. };
  120. class ImagePairLess
  121. {
  122. public:
  123. ImagePairLess(const std::vector<Mat> &images, const std::vector<Point> &corners)
  124. : src_(&images[0]), corners_(&corners[0]) {}
  125. bool operator() (const std::pair<size_t, size_t> &l, const std::pair<size_t, size_t> &r) const
  126. {
  127. Point c1 = corners_[l.first] + Point(src_[l.first].cols / 2, src_[l.first].rows / 2);
  128. Point c2 = corners_[l.second] + Point(src_[l.second].cols / 2, src_[l.second].rows / 2);
  129. int d1 = (c1 - c2).dot(c1 - c2);
  130. c1 = corners_[r.first] + Point(src_[r.first].cols / 2, src_[r.first].rows / 2);
  131. c2 = corners_[r.second] + Point(src_[r.second].cols / 2, src_[r.second].rows / 2);
  132. int d2 = (c1 - c2).dot(c1 - c2);
  133. return d1 < d2;
  134. }
  135. private:
  136. const Mat *src_;
  137. const Point *corners_;
  138. };
  139. class ClosePoints
  140. {
  141. public:
  142. ClosePoints(int minDist) : minDist_(minDist) {}
  143. bool operator() (const Point &p1, const Point &p2) const
  144. {
  145. int dist2 = (p1.x-p2.x) * (p1.x-p2.x) + (p1.y-p2.y) * (p1.y-p2.y);
  146. return dist2 < minDist_ * minDist_;
  147. }
  148. private:
  149. int minDist_;
  150. };
  151. void process(
  152. const Mat &image1, const Mat &image2, Point tl1, Point tl2, Mat &mask1, Mat &mask2);
  153. void findComponents();
  154. void findEdges();
  155. void resolveConflicts(
  156. const Mat &image1, const Mat &image2, Point tl1, Point tl2, Mat &mask1, Mat &mask2);
  157. void computeGradients(const Mat &image1, const Mat &image2);
  158. bool hasOnlyOneNeighbor(int comp);
  159. bool closeToContour(int y, int x, const Mat_<uchar> &contourMask);
  160. bool getSeamTips(int comp1, int comp2, Point &p1, Point &p2);
  161. void computeCosts(
  162. const Mat &image1, const Mat &image2, Point tl1, Point tl2,
  163. int comp, Mat_<float> &costV, Mat_<float> &costH);
  164. bool estimateSeam(
  165. const Mat &image1, const Mat &image2, Point tl1, Point tl2, int comp,
  166. Point p1, Point p2, std::vector<Point> &seam, bool &isHorizontal);
  167. void updateLabelsUsingSeam(
  168. int comp1, int comp2, const std::vector<Point> &seam, bool isHorizontalSeam);
  169. CostFunction costFunc_;
  170. // processing images pair data
  171. Point unionTl_, unionBr_;
  172. Size unionSize_;
  173. Mat_<uchar> mask1_, mask2_;
  174. Mat_<uchar> contour1mask_, contour2mask_;
  175. Mat_<float> gradx1_, grady1_;
  176. Mat_<float> gradx2_, grady2_;
  177. // components data
  178. int ncomps_;
  179. Mat_<int> labels_;
  180. std::vector<ComponentState> states_;
  181. std::vector<Point> tls_, brs_;
  182. std::vector<std::vector<Point> > contours_;
  183. std::set<std::pair<int, int> > edges_;
  184. };
  185. /** @brief Base class for all minimum graph-cut-based seam estimators.
  186. */
  187. class CV_EXPORTS GraphCutSeamFinderBase
  188. {
  189. public:
  190. enum CostType { COST_COLOR, COST_COLOR_GRAD };
  191. };
  192. /** @brief Minimum graph cut-based seam estimator. See details in @cite V03 .
  193. */
  194. class CV_EXPORTS GraphCutSeamFinder : public GraphCutSeamFinderBase, public SeamFinder
  195. {
  196. public:
  197. GraphCutSeamFinder(int cost_type = COST_COLOR_GRAD, float terminal_cost = 10000.f,
  198. float bad_region_penalty = 1000.f);
  199. ~GraphCutSeamFinder();
  200. void find(const std::vector<UMat> &src, const std::vector<Point> &corners,
  201. std::vector<UMat> &masks);
  202. private:
  203. // To avoid GCGraph dependency
  204. class Impl;
  205. Ptr<PairwiseSeamFinder> impl_;
  206. };
  207. #ifdef HAVE_OPENCV_CUDALEGACY
  208. class CV_EXPORTS GraphCutSeamFinderGpu : public GraphCutSeamFinderBase, public PairwiseSeamFinder
  209. {
  210. public:
  211. GraphCutSeamFinderGpu(int cost_type = COST_COLOR_GRAD, float terminal_cost = 10000.f,
  212. float bad_region_penalty = 1000.f)
  213. : cost_type_(cost_type), terminal_cost_(terminal_cost),
  214. bad_region_penalty_(bad_region_penalty) {}
  215. void find(const std::vector<cv::UMat> &src, const std::vector<cv::Point> &corners,
  216. std::vector<cv::UMat> &masks);
  217. void findInPair(size_t first, size_t second, Rect roi);
  218. private:
  219. void setGraphWeightsColor(const cv::Mat &img1, const cv::Mat &img2, const cv::Mat &mask1, const cv::Mat &mask2,
  220. cv::Mat &terminals, cv::Mat &leftT, cv::Mat &rightT, cv::Mat &top, cv::Mat &bottom);
  221. void setGraphWeightsColorGrad(const cv::Mat &img1, const cv::Mat &img2, const cv::Mat &dx1, const cv::Mat &dx2,
  222. const cv::Mat &dy1, const cv::Mat &dy2, const cv::Mat &mask1, const cv::Mat &mask2,
  223. cv::Mat &terminals, cv::Mat &leftT, cv::Mat &rightT, cv::Mat &top, cv::Mat &bottom);
  224. std::vector<Mat> dx_, dy_;
  225. int cost_type_;
  226. float terminal_cost_;
  227. float bad_region_penalty_;
  228. };
  229. #endif
  230. //! @}
  231. } // namespace detail
  232. } // namespace cv
  233. #endif // __OPENCV_STITCHING_SEAM_FINDERS_HPP__