area.hpp 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. // Boost.Geometry (aka GGL, Generic Geometry Library)
  2. // Copyright (c) 2007-2012 Barend Gehrels, Amsterdam, the Netherlands.
  3. // Copyright (c) 2008-2012 Bruno Lalande, Paris, France.
  4. // Copyright (c) 2009-2012 Mateusz Loskot, London, UK.
  5. // Parts of Boost.Geometry are redesigned from Geodan's Geographic Library
  6. // (geolib/GGL), copyright (c) 1995-2010 Geodan, Amsterdam, the Netherlands.
  7. // Use, modification and distribution is subject to the Boost Software License,
  8. // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  9. // http://www.boost.org/LICENSE_1_0.txt)
  10. #ifndef BOOST_GEOMETRY_ALGORITHMS_AREA_HPP
  11. #define BOOST_GEOMETRY_ALGORITHMS_AREA_HPP
  12. #include <boost/concept_check.hpp>
  13. #include <boost/mpl/if.hpp>
  14. #include <boost/range/functions.hpp>
  15. #include <boost/range/metafunctions.hpp>
  16. #include <boost/variant/apply_visitor.hpp>
  17. #include <boost/variant/static_visitor.hpp>
  18. #include <boost/variant/variant_fwd.hpp>
  19. #include <boost/geometry/core/closure.hpp>
  20. #include <boost/geometry/core/exterior_ring.hpp>
  21. #include <boost/geometry/core/interior_rings.hpp>
  22. #include <boost/geometry/core/point_order.hpp>
  23. #include <boost/geometry/core/point_type.hpp>
  24. #include <boost/geometry/core/ring_type.hpp>
  25. #include <boost/geometry/core/tags.hpp>
  26. #include <boost/geometry/geometries/concepts/check.hpp>
  27. #include <boost/geometry/algorithms/detail/calculate_null.hpp>
  28. #include <boost/geometry/algorithms/detail/calculate_sum.hpp>
  29. // #include <boost/geometry/algorithms/detail/throw_on_empty_input.hpp>
  30. #include <boost/geometry/algorithms/detail/multi_sum.hpp>
  31. #include <boost/geometry/strategies/area.hpp>
  32. #include <boost/geometry/strategies/default_area_result.hpp>
  33. #include <boost/geometry/strategies/concepts/area_concept.hpp>
  34. #include <boost/geometry/util/math.hpp>
  35. #include <boost/geometry/util/order_as_direction.hpp>
  36. #include <boost/geometry/views/closeable_view.hpp>
  37. #include <boost/geometry/views/reversible_view.hpp>
  38. namespace boost { namespace geometry
  39. {
  40. #ifndef DOXYGEN_NO_DETAIL
  41. namespace detail { namespace area
  42. {
  43. struct box_area
  44. {
  45. template <typename Box, typename Strategy>
  46. static inline typename coordinate_type<Box>::type
  47. apply(Box const& box, Strategy const&)
  48. {
  49. // Currently only works for 2D Cartesian boxes
  50. assert_dimension<Box, 2>();
  51. return (get<max_corner, 0>(box) - get<min_corner, 0>(box))
  52. * (get<max_corner, 1>(box) - get<min_corner, 1>(box));
  53. }
  54. };
  55. template
  56. <
  57. iterate_direction Direction,
  58. closure_selector Closure
  59. >
  60. struct ring_area
  61. {
  62. template <typename Ring, typename Strategy>
  63. static inline typename Strategy::return_type
  64. apply(Ring const& ring, Strategy const& strategy)
  65. {
  66. BOOST_CONCEPT_ASSERT( (geometry::concept::AreaStrategy<Strategy>) );
  67. assert_dimension<Ring, 2>();
  68. // Ignore warning (because using static method sometimes) on strategy
  69. boost::ignore_unused_variable_warning(strategy);
  70. // An open ring has at least three points,
  71. // A closed ring has at least four points,
  72. // if not, there is no (zero) area
  73. if (boost::size(ring)
  74. < core_detail::closure::minimum_ring_size<Closure>::value)
  75. {
  76. return typename Strategy::return_type();
  77. }
  78. typedef typename reversible_view<Ring const, Direction>::type rview_type;
  79. typedef typename closeable_view
  80. <
  81. rview_type const, Closure
  82. >::type view_type;
  83. typedef typename boost::range_iterator<view_type const>::type iterator_type;
  84. rview_type rview(ring);
  85. view_type view(rview);
  86. typename Strategy::state_type state;
  87. iterator_type it = boost::begin(view);
  88. iterator_type end = boost::end(view);
  89. for (iterator_type previous = it++;
  90. it != end;
  91. ++previous, ++it)
  92. {
  93. strategy.apply(*previous, *it, state);
  94. }
  95. return strategy.result(state);
  96. }
  97. };
  98. }} // namespace detail::area
  99. #endif // DOXYGEN_NO_DETAIL
  100. #ifndef DOXYGEN_NO_DISPATCH
  101. namespace dispatch
  102. {
  103. template
  104. <
  105. typename Geometry,
  106. typename Tag = typename tag<Geometry>::type
  107. >
  108. struct area : detail::calculate_null
  109. {
  110. template <typename Strategy>
  111. static inline typename Strategy::return_type apply(Geometry const& geometry, Strategy const& strategy)
  112. {
  113. return calculate_null::apply<typename Strategy::return_type>(geometry, strategy);
  114. }
  115. };
  116. template <typename Geometry>
  117. struct area<Geometry, box_tag> : detail::area::box_area
  118. {};
  119. template <typename Ring>
  120. struct area<Ring, ring_tag>
  121. : detail::area::ring_area
  122. <
  123. order_as_direction<geometry::point_order<Ring>::value>::value,
  124. geometry::closure<Ring>::value
  125. >
  126. {};
  127. template <typename Polygon>
  128. struct area<Polygon, polygon_tag> : detail::calculate_polygon_sum
  129. {
  130. template <typename Strategy>
  131. static inline typename Strategy::return_type apply(Polygon const& polygon, Strategy const& strategy)
  132. {
  133. return calculate_polygon_sum::apply<
  134. typename Strategy::return_type,
  135. detail::area::ring_area
  136. <
  137. order_as_direction<geometry::point_order<Polygon>::value>::value,
  138. geometry::closure<Polygon>::value
  139. >
  140. >(polygon, strategy);
  141. }
  142. };
  143. template <typename MultiGeometry>
  144. struct area<MultiGeometry, multi_polygon_tag> : detail::multi_sum
  145. {
  146. template <typename Strategy>
  147. static inline typename Strategy::return_type
  148. apply(MultiGeometry const& multi, Strategy const& strategy)
  149. {
  150. return multi_sum::apply
  151. <
  152. typename Strategy::return_type,
  153. area<typename boost::range_value<MultiGeometry>::type>
  154. >(multi, strategy);
  155. }
  156. };
  157. } // namespace dispatch
  158. #endif // DOXYGEN_NO_DISPATCH
  159. namespace resolve_variant {
  160. template <typename Geometry>
  161. struct area
  162. {
  163. template <typename Strategy>
  164. static inline typename Strategy::return_type apply(Geometry const& geometry,
  165. Strategy const& strategy)
  166. {
  167. return dispatch::area<Geometry>::apply(geometry, strategy);
  168. }
  169. };
  170. template <BOOST_VARIANT_ENUM_PARAMS(typename T)>
  171. struct area<boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> >
  172. {
  173. template <typename Strategy>
  174. struct visitor: boost::static_visitor<typename Strategy::return_type>
  175. {
  176. Strategy const& m_strategy;
  177. visitor(Strategy const& strategy): m_strategy(strategy) {}
  178. template <typename Geometry>
  179. typename Strategy::return_type operator()(Geometry const& geometry) const
  180. {
  181. return area<Geometry>::apply(geometry, m_strategy);
  182. }
  183. };
  184. template <typename Strategy>
  185. static inline typename Strategy::return_type
  186. apply(boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> const& geometry,
  187. Strategy const& strategy)
  188. {
  189. return boost::apply_visitor(visitor<Strategy>(strategy), geometry);
  190. }
  191. };
  192. } // namespace resolve_variant
  193. /*!
  194. \brief \brief_calc{area}
  195. \ingroup area
  196. \details \details_calc{area}. \details_default_strategy
  197. The area algorithm calculates the surface area of all geometries having a surface, namely
  198. box, polygon, ring, multipolygon. The units are the square of the units used for the points
  199. defining the surface. If subject geometry is defined in meters, then area is calculated
  200. in square meters.
  201. The area calculation can be done in all three common coordinate systems, Cartesian, Spherical
  202. and Geographic as well.
  203. \tparam Geometry \tparam_geometry
  204. \param geometry \param_geometry
  205. \return \return_calc{area}
  206. \qbk{[include reference/algorithms/area.qbk]}
  207. \qbk{[heading Examples]}
  208. \qbk{[area] [area_output]}
  209. */
  210. template <typename Geometry>
  211. inline typename default_area_result<Geometry>::type area(Geometry const& geometry)
  212. {
  213. concept::check<Geometry const>();
  214. // TODO put this into a resolve_strategy stage
  215. // (and take the return type from resolve_variant)
  216. typedef typename point_type<Geometry>::type point_type;
  217. typedef typename strategy::area::services::default_strategy
  218. <
  219. typename cs_tag<point_type>::type,
  220. point_type
  221. >::type strategy_type;
  222. // detail::throw_on_empty_input(geometry);
  223. return resolve_variant::area<Geometry>::apply(geometry, strategy_type());
  224. }
  225. /*!
  226. \brief \brief_calc{area} \brief_strategy
  227. \ingroup area
  228. \details \details_calc{area} \brief_strategy. \details_strategy_reasons
  229. \tparam Geometry \tparam_geometry
  230. \tparam Strategy \tparam_strategy{Area}
  231. \param geometry \param_geometry
  232. \param strategy \param_strategy{area}
  233. \return \return_calc{area}
  234. \qbk{distinguish,with strategy}
  235. \qbk{
  236. [include reference/algorithms/area.qbk]
  237. [heading Example]
  238. [area_with_strategy]
  239. [area_with_strategy_output]
  240. [heading Available Strategies]
  241. \* [link geometry.reference.strategies.strategy_area_surveyor Surveyor (cartesian)]
  242. \* [link geometry.reference.strategies.strategy_area_huiller Huiller (spherical)]
  243. }
  244. */
  245. template <typename Geometry, typename Strategy>
  246. inline typename Strategy::return_type area(
  247. Geometry const& geometry, Strategy const& strategy)
  248. {
  249. concept::check<Geometry const>();
  250. // detail::throw_on_empty_input(geometry);
  251. return resolve_variant::area<Geometry>::apply(geometry, strategy);
  252. }
  253. }} // namespace boost::geometry
  254. #endif // BOOST_GEOMETRY_ALGORITHMS_AREA_HPP