union.hpp 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. // Boost.Geometry (aka GGL, Generic Geometry Library)
  2. // Copyright (c) 2007-2014 Barend Gehrels, Amsterdam, the Netherlands.
  3. // This file was modified by Oracle on 2014.
  4. // Modifications copyright (c) 2014 Oracle and/or its affiliates.
  5. // Contributed and/or modified by Menelaos Karavelas, on behalf of Oracle
  6. // Use, modification and distribution is subject to the Boost Software License,
  7. // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  8. // http://www.boost.org/LICENSE_1_0.txt)
  9. #ifndef BOOST_GEOMETRY_ALGORITHMS_UNION_HPP
  10. #define BOOST_GEOMETRY_ALGORITHMS_UNION_HPP
  11. #include <boost/range/metafunctions.hpp>
  12. #include <boost/geometry/core/is_areal.hpp>
  13. #include <boost/geometry/core/point_order.hpp>
  14. #include <boost/geometry/core/reverse_dispatch.hpp>
  15. #include <boost/geometry/geometries/concepts/check.hpp>
  16. #include <boost/geometry/algorithms/not_implemented.hpp>
  17. #include <boost/geometry/algorithms/detail/overlay/overlay.hpp>
  18. #include <boost/geometry/policies/robustness/get_rescale_policy.hpp>
  19. #include <boost/geometry/algorithms/detail/overlay/linear_linear.hpp>
  20. #include <boost/geometry/algorithms/detail/overlay/pointlike_pointlike.hpp>
  21. namespace boost { namespace geometry
  22. {
  23. #ifndef DOXYGEN_NO_DISPATCH
  24. namespace dispatch
  25. {
  26. template
  27. <
  28. typename Geometry1, typename Geometry2, typename GeometryOut,
  29. typename TagIn1 = typename tag<Geometry1>::type,
  30. typename TagIn2 = typename tag<Geometry2>::type,
  31. typename TagOut = typename tag<GeometryOut>::type,
  32. bool Areal1 = geometry::is_areal<Geometry1>::value,
  33. bool Areal2 = geometry::is_areal<Geometry2>::value,
  34. bool ArealOut = geometry::is_areal<GeometryOut>::value,
  35. bool Reverse1 = detail::overlay::do_reverse<geometry::point_order<Geometry1>::value>::value,
  36. bool Reverse2 = detail::overlay::do_reverse<geometry::point_order<Geometry2>::value>::value,
  37. bool ReverseOut = detail::overlay::do_reverse<geometry::point_order<GeometryOut>::value>::value,
  38. bool Reverse = geometry::reverse_dispatch<Geometry1, Geometry2>::type::value
  39. >
  40. struct union_insert: not_implemented<TagIn1, TagIn2, TagOut>
  41. {};
  42. // If reversal is needed, perform it first
  43. template
  44. <
  45. typename Geometry1, typename Geometry2, typename GeometryOut,
  46. typename TagIn1, typename TagIn2, typename TagOut,
  47. bool Areal1, bool Areal2, bool ArealOut,
  48. bool Reverse1, bool Reverse2, bool ReverseOut
  49. >
  50. struct union_insert
  51. <
  52. Geometry1, Geometry2, GeometryOut,
  53. TagIn1, TagIn2, TagOut,
  54. Areal1, Areal2, ArealOut,
  55. Reverse1, Reverse2, ReverseOut,
  56. true
  57. >: union_insert<Geometry2, Geometry1, GeometryOut>
  58. {
  59. template <typename RobustPolicy, typename OutputIterator, typename Strategy>
  60. static inline OutputIterator apply(Geometry1 const& g1,
  61. Geometry2 const& g2,
  62. RobustPolicy const& robust_policy,
  63. OutputIterator out,
  64. Strategy const& strategy)
  65. {
  66. return union_insert
  67. <
  68. Geometry2, Geometry1, GeometryOut
  69. >::apply(g2, g1, robust_policy, out, strategy);
  70. }
  71. };
  72. template
  73. <
  74. typename Geometry1, typename Geometry2, typename GeometryOut,
  75. typename TagIn1, typename TagIn2, typename TagOut,
  76. bool Reverse1, bool Reverse2, bool ReverseOut
  77. >
  78. struct union_insert
  79. <
  80. Geometry1, Geometry2, GeometryOut,
  81. TagIn1, TagIn2, TagOut,
  82. true, true, true,
  83. Reverse1, Reverse2, ReverseOut,
  84. false
  85. > : detail::overlay::overlay
  86. <Geometry1, Geometry2, Reverse1, Reverse2, ReverseOut, GeometryOut, overlay_union>
  87. {};
  88. // dispatch for union of non-areal geometries
  89. template
  90. <
  91. typename Geometry1, typename Geometry2, typename GeometryOut,
  92. typename TagIn1, typename TagIn2, typename TagOut,
  93. bool Reverse1, bool Reverse2, bool ReverseOut
  94. >
  95. struct union_insert
  96. <
  97. Geometry1, Geometry2, GeometryOut,
  98. TagIn1, TagIn2, TagOut,
  99. false, false, false,
  100. Reverse1, Reverse2, ReverseOut,
  101. false
  102. > : union_insert
  103. <
  104. Geometry1, Geometry2, GeometryOut,
  105. typename tag_cast<TagIn1, pointlike_tag, linear_tag>::type,
  106. typename tag_cast<TagIn2, pointlike_tag, linear_tag>::type,
  107. TagOut,
  108. false, false, false,
  109. Reverse1, Reverse2, ReverseOut,
  110. false
  111. >
  112. {};
  113. // dispatch for union of linear geometries
  114. template
  115. <
  116. typename Linear1, typename Linear2, typename LineStringOut,
  117. bool Reverse1, bool Reverse2, bool ReverseOut
  118. >
  119. struct union_insert
  120. <
  121. Linear1, Linear2, LineStringOut,
  122. linear_tag, linear_tag, linestring_tag,
  123. false, false, false,
  124. Reverse1, Reverse2, ReverseOut,
  125. false
  126. > : detail::overlay::linear_linear_linestring
  127. <
  128. Linear1, Linear2, LineStringOut, overlay_union
  129. >
  130. {};
  131. // dispatch for point-like geometries
  132. template
  133. <
  134. typename PointLike1, typename PointLike2, typename PointOut,
  135. bool Reverse1, bool Reverse2, bool ReverseOut
  136. >
  137. struct union_insert
  138. <
  139. PointLike1, PointLike2, PointOut,
  140. pointlike_tag, pointlike_tag, point_tag,
  141. false, false, false,
  142. Reverse1, Reverse2, ReverseOut,
  143. false
  144. > : detail::overlay::union_pointlike_pointlike_point
  145. <
  146. PointLike1, PointLike2, PointOut
  147. >
  148. {};
  149. } // namespace dispatch
  150. #endif // DOXYGEN_NO_DISPATCH
  151. #ifndef DOXYGEN_NO_DETAIL
  152. namespace detail { namespace union_
  153. {
  154. /*!
  155. \brief_calc2{union}
  156. \ingroup union
  157. \details \details_calc2{union_insert, spatial set theoretic union}.
  158. \details_insert{union}
  159. \tparam GeometryOut output geometry type, must be specified
  160. \tparam Geometry1 \tparam_geometry
  161. \tparam Geometry2 \tparam_geometry
  162. \tparam OutputIterator output iterator
  163. \param geometry1 \param_geometry
  164. \param geometry2 \param_geometry
  165. \param out \param_out{union}
  166. \return \return_out
  167. */
  168. template
  169. <
  170. typename GeometryOut,
  171. typename Geometry1,
  172. typename Geometry2,
  173. typename OutputIterator
  174. >
  175. inline OutputIterator union_insert(Geometry1 const& geometry1,
  176. Geometry2 const& geometry2,
  177. OutputIterator out)
  178. {
  179. concept::check<Geometry1 const>();
  180. concept::check<Geometry2 const>();
  181. concept::check<GeometryOut>();
  182. typedef typename geometry::rescale_overlay_policy_type
  183. <
  184. Geometry1,
  185. Geometry2
  186. >::type rescale_policy_type;
  187. typedef strategy_intersection
  188. <
  189. typename cs_tag<GeometryOut>::type,
  190. Geometry1,
  191. Geometry2,
  192. typename geometry::point_type<GeometryOut>::type,
  193. rescale_policy_type
  194. > strategy;
  195. rescale_policy_type robust_policy
  196. = geometry::get_rescale_policy<rescale_policy_type>(geometry1, geometry2);
  197. return dispatch::union_insert
  198. <
  199. Geometry1, Geometry2, GeometryOut
  200. >::apply(geometry1, geometry2, robust_policy, out, strategy());
  201. }
  202. }} // namespace detail::union_
  203. #endif // DOXYGEN_NO_DETAIL
  204. /*!
  205. \brief Combines two geometries which each other
  206. \ingroup union
  207. \details \details_calc2{union, spatial set theoretic union}.
  208. \tparam Geometry1 \tparam_geometry
  209. \tparam Geometry2 \tparam_geometry
  210. \tparam Collection output collection, either a multi-geometry,
  211. or a std::vector<Geometry> / std::deque<Geometry> etc
  212. \param geometry1 \param_geometry
  213. \param geometry2 \param_geometry
  214. \param output_collection the output collection
  215. \note Called union_ because union is a reserved word.
  216. \qbk{[include reference/algorithms/union.qbk]}
  217. */
  218. template
  219. <
  220. typename Geometry1,
  221. typename Geometry2,
  222. typename Collection
  223. >
  224. inline void union_(Geometry1 const& geometry1,
  225. Geometry2 const& geometry2,
  226. Collection& output_collection)
  227. {
  228. concept::check<Geometry1 const>();
  229. concept::check<Geometry2 const>();
  230. typedef typename boost::range_value<Collection>::type geometry_out;
  231. concept::check<geometry_out>();
  232. detail::union_::union_insert<geometry_out>(geometry1, geometry2,
  233. range::back_inserter(output_collection));
  234. }
  235. }} // namespace boost::geometry
  236. #endif // BOOST_GEOMETRY_ALGORITHMS_UNION_HPP