centroid.hpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680
  1. // Boost.Geometry (aka GGL, Generic Geometry Library)
  2. // Copyright (c) 2007-2015 Barend Gehrels, Amsterdam, the Netherlands.
  3. // Copyright (c) 2008-2015 Bruno Lalande, Paris, France.
  4. // Copyright (c) 2009-2015 Mateusz Loskot, London, UK.
  5. // Copyright (c) 2014-2015 Adam Wulkiewicz, Lodz, Poland.
  6. // This file was modified by Oracle on 2014, 2015.
  7. // Modifications copyright (c) 2014-2015 Oracle and/or its affiliates.
  8. // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle
  9. // Contributed and/or modified by Menelaos Karavelas, on behalf of Oracle
  10. // Parts of Boost.Geometry are redesigned from Geodan's Geographic Library
  11. // (geolib/GGL), copyright (c) 1995-2010 Geodan, Amsterdam, the Netherlands.
  12. // Use, modification and distribution is subject to the Boost Software License,
  13. // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  14. // http://www.boost.org/LICENSE_1_0.txt)
  15. #ifndef BOOST_GEOMETRY_ALGORITHMS_CENTROID_HPP
  16. #define BOOST_GEOMETRY_ALGORITHMS_CENTROID_HPP
  17. #include <cstddef>
  18. #include <boost/core/ignore_unused.hpp>
  19. #include <boost/range.hpp>
  20. #include <boost/variant/apply_visitor.hpp>
  21. #include <boost/variant/static_visitor.hpp>
  22. #include <boost/variant/variant_fwd.hpp>
  23. #include <boost/geometry/core/closure.hpp>
  24. #include <boost/geometry/core/cs.hpp>
  25. #include <boost/geometry/core/coordinate_dimension.hpp>
  26. #include <boost/geometry/core/exception.hpp>
  27. #include <boost/geometry/core/exterior_ring.hpp>
  28. #include <boost/geometry/core/interior_rings.hpp>
  29. #include <boost/geometry/core/tag_cast.hpp>
  30. #include <boost/geometry/core/tags.hpp>
  31. #include <boost/geometry/core/point_type.hpp>
  32. #include <boost/geometry/geometries/concepts/check.hpp>
  33. #include <boost/geometry/algorithms/assign.hpp>
  34. #include <boost/geometry/algorithms/convert.hpp>
  35. #include <boost/geometry/algorithms/detail/interior_iterator.hpp>
  36. #include <boost/geometry/algorithms/detail/point_on_border.hpp>
  37. #include <boost/geometry/algorithms/not_implemented.hpp>
  38. #include <boost/geometry/strategies/centroid.hpp>
  39. #include <boost/geometry/strategies/concepts/centroid_concept.hpp>
  40. #include <boost/geometry/strategies/default_strategy.hpp>
  41. #include <boost/geometry/views/closeable_view.hpp>
  42. #include <boost/geometry/util/for_each_coordinate.hpp>
  43. #include <boost/geometry/util/select_coordinate_type.hpp>
  44. #include <boost/geometry/algorithms/is_empty.hpp>
  45. #include <boost/geometry/algorithms/detail/centroid/translating_transformer.hpp>
  46. namespace boost { namespace geometry
  47. {
  48. #if ! defined(BOOST_GEOMETRY_CENTROID_NO_THROW)
  49. /*!
  50. \brief Centroid Exception
  51. \ingroup centroid
  52. \details The centroid_exception is thrown if the free centroid function is called with
  53. geometries for which the centroid cannot be calculated. For example: a linestring
  54. without points, a polygon without points, an empty multi-geometry.
  55. \qbk{
  56. [heading See also]
  57. \* [link geometry.reference.algorithms.centroid the centroid function]
  58. }
  59. */
  60. class centroid_exception : public geometry::exception
  61. {
  62. public:
  63. /*!
  64. \brief The default constructor
  65. */
  66. inline centroid_exception() {}
  67. /*!
  68. \brief Returns the explanatory string.
  69. \return Pointer to a null-terminated string with explanatory information.
  70. */
  71. virtual char const* what() const throw()
  72. {
  73. return "Boost.Geometry Centroid calculation exception";
  74. }
  75. };
  76. #endif
  77. #ifndef DOXYGEN_NO_DETAIL
  78. namespace detail { namespace centroid
  79. {
  80. struct centroid_point
  81. {
  82. template<typename Point, typename PointCentroid, typename Strategy>
  83. static inline void apply(Point const& point, PointCentroid& centroid,
  84. Strategy const&)
  85. {
  86. geometry::convert(point, centroid);
  87. }
  88. };
  89. template
  90. <
  91. typename Indexed,
  92. typename Point,
  93. std::size_t Dimension = 0,
  94. std::size_t DimensionCount = dimension<Indexed>::type::value
  95. >
  96. struct centroid_indexed_calculator
  97. {
  98. typedef typename select_coordinate_type
  99. <
  100. Indexed, Point
  101. >::type coordinate_type;
  102. static inline void apply(Indexed const& indexed, Point& centroid)
  103. {
  104. coordinate_type const c1 = get<min_corner, Dimension>(indexed);
  105. coordinate_type const c2 = get<max_corner, Dimension>(indexed);
  106. coordinate_type m = c1 + c2;
  107. coordinate_type const two = 2;
  108. m /= two;
  109. set<Dimension>(centroid, m);
  110. centroid_indexed_calculator
  111. <
  112. Indexed, Point, Dimension + 1
  113. >::apply(indexed, centroid);
  114. }
  115. };
  116. template<typename Indexed, typename Point, std::size_t DimensionCount>
  117. struct centroid_indexed_calculator<Indexed, Point, DimensionCount, DimensionCount>
  118. {
  119. static inline void apply(Indexed const& , Point& )
  120. {
  121. }
  122. };
  123. struct centroid_indexed
  124. {
  125. template<typename Indexed, typename Point, typename Strategy>
  126. static inline void apply(Indexed const& indexed, Point& centroid,
  127. Strategy const&)
  128. {
  129. centroid_indexed_calculator
  130. <
  131. Indexed, Point
  132. >::apply(indexed, centroid);
  133. }
  134. };
  135. // There is one thing where centroid is different from e.g. within.
  136. // If the ring has only one point, it might make sense that
  137. // that point is the centroid.
  138. template<typename Point, typename Range>
  139. inline bool range_ok(Range const& range, Point& centroid)
  140. {
  141. std::size_t const n = boost::size(range);
  142. if (n > 1)
  143. {
  144. return true;
  145. }
  146. else if (n <= 0)
  147. {
  148. #if ! defined(BOOST_GEOMETRY_CENTROID_NO_THROW)
  149. throw centroid_exception();
  150. #else
  151. return false;
  152. #endif
  153. }
  154. else // if (n == 1)
  155. {
  156. // Take over the first point in a "coordinate neutral way"
  157. geometry::convert(*boost::begin(range), centroid);
  158. return false;
  159. }
  160. //return true; // unreachable
  161. }
  162. /*!
  163. \brief Calculate the centroid of a Ring or a Linestring.
  164. */
  165. template <closure_selector Closure>
  166. struct centroid_range_state
  167. {
  168. template<typename Ring, typename PointTransformer, typename Strategy>
  169. static inline void apply(Ring const& ring,
  170. PointTransformer const& transformer,
  171. Strategy const& strategy,
  172. typename Strategy::state_type& state)
  173. {
  174. boost::ignore_unused(strategy);
  175. typedef typename geometry::point_type<Ring const>::type point_type;
  176. typedef typename closeable_view<Ring const, Closure>::type view_type;
  177. typedef typename boost::range_iterator<view_type const>::type iterator_type;
  178. view_type view(ring);
  179. iterator_type it = boost::begin(view);
  180. iterator_type end = boost::end(view);
  181. if (it != end)
  182. {
  183. typename PointTransformer::result_type
  184. previous_pt = transformer.apply(*it);
  185. for ( ++it ; it != end ; ++it)
  186. {
  187. typename PointTransformer::result_type
  188. pt = transformer.apply(*it);
  189. strategy.apply(static_cast<point_type const&>(previous_pt),
  190. static_cast<point_type const&>(pt),
  191. state);
  192. previous_pt = pt;
  193. }
  194. }
  195. }
  196. };
  197. template <closure_selector Closure>
  198. struct centroid_range
  199. {
  200. template<typename Range, typename Point, typename Strategy>
  201. static inline bool apply(Range const& range, Point& centroid,
  202. Strategy const& strategy)
  203. {
  204. if (range_ok(range, centroid))
  205. {
  206. // prepare translation transformer
  207. translating_transformer<Range> transformer(*boost::begin(range));
  208. typename Strategy::state_type state;
  209. centroid_range_state<Closure>::apply(range, transformer,
  210. strategy, state);
  211. if ( strategy.result(state, centroid) )
  212. {
  213. // translate the result back
  214. transformer.apply_reverse(centroid);
  215. return true;
  216. }
  217. }
  218. return false;
  219. }
  220. };
  221. /*!
  222. \brief Centroid of a polygon.
  223. \note Because outer ring is clockwise, inners are counter clockwise,
  224. triangle approach is OK and works for polygons with rings.
  225. */
  226. struct centroid_polygon_state
  227. {
  228. template<typename Polygon, typename PointTransformer, typename Strategy>
  229. static inline void apply(Polygon const& poly,
  230. PointTransformer const& transformer,
  231. Strategy const& strategy,
  232. typename Strategy::state_type& state)
  233. {
  234. typedef typename ring_type<Polygon>::type ring_type;
  235. typedef centroid_range_state<geometry::closure<ring_type>::value> per_ring;
  236. per_ring::apply(exterior_ring(poly), transformer, strategy, state);
  237. typename interior_return_type<Polygon const>::type
  238. rings = interior_rings(poly);
  239. for (typename detail::interior_iterator<Polygon const>::type
  240. it = boost::begin(rings); it != boost::end(rings); ++it)
  241. {
  242. per_ring::apply(*it, transformer, strategy, state);
  243. }
  244. }
  245. };
  246. struct centroid_polygon
  247. {
  248. template<typename Polygon, typename Point, typename Strategy>
  249. static inline bool apply(Polygon const& poly, Point& centroid,
  250. Strategy const& strategy)
  251. {
  252. if (range_ok(exterior_ring(poly), centroid))
  253. {
  254. // prepare translation transformer
  255. translating_transformer<Polygon>
  256. transformer(*boost::begin(exterior_ring(poly)));
  257. typename Strategy::state_type state;
  258. centroid_polygon_state::apply(poly, transformer, strategy, state);
  259. if ( strategy.result(state, centroid) )
  260. {
  261. // translate the result back
  262. transformer.apply_reverse(centroid);
  263. return true;
  264. }
  265. }
  266. return false;
  267. }
  268. };
  269. /*!
  270. \brief Building block of a multi-point, to be used as Policy in the
  271. more generec centroid_multi
  272. */
  273. struct centroid_multi_point_state
  274. {
  275. template <typename Point, typename PointTransformer, typename Strategy>
  276. static inline void apply(Point const& point,
  277. PointTransformer const& transformer,
  278. Strategy const& strategy,
  279. typename Strategy::state_type& state)
  280. {
  281. boost::ignore_unused(strategy);
  282. strategy.apply(static_cast<Point const&>(transformer.apply(point)),
  283. state);
  284. }
  285. };
  286. /*!
  287. \brief Generic implementation which calls a policy to calculate the
  288. centroid of the total of its single-geometries
  289. \details The Policy is, in general, the single-version, with state. So
  290. detail::centroid::centroid_polygon_state is used as a policy for this
  291. detail::centroid::centroid_multi
  292. */
  293. template <typename Policy>
  294. struct centroid_multi
  295. {
  296. template <typename Multi, typename Point, typename Strategy>
  297. static inline bool apply(Multi const& multi,
  298. Point& centroid,
  299. Strategy const& strategy)
  300. {
  301. #if ! defined(BOOST_GEOMETRY_CENTROID_NO_THROW)
  302. // If there is nothing in any of the ranges, it is not possible
  303. // to calculate the centroid
  304. if (geometry::is_empty(multi))
  305. {
  306. throw centroid_exception();
  307. }
  308. #endif
  309. // prepare translation transformer
  310. translating_transformer<Multi> transformer(multi);
  311. typename Strategy::state_type state;
  312. for (typename boost::range_iterator<Multi const>::type
  313. it = boost::begin(multi);
  314. it != boost::end(multi);
  315. ++it)
  316. {
  317. Policy::apply(*it, transformer, strategy, state);
  318. }
  319. if ( strategy.result(state, centroid) )
  320. {
  321. // translate the result back
  322. transformer.apply_reverse(centroid);
  323. return true;
  324. }
  325. return false;
  326. }
  327. };
  328. template <typename Algorithm>
  329. struct centroid_linear_areal
  330. {
  331. template <typename Geometry, typename Point, typename Strategy>
  332. static inline void apply(Geometry const& geom,
  333. Point& centroid,
  334. Strategy const& strategy)
  335. {
  336. if ( ! Algorithm::apply(geom, centroid, strategy) )
  337. {
  338. geometry::point_on_border(centroid, geom);
  339. }
  340. }
  341. };
  342. }} // namespace detail::centroid
  343. #endif // DOXYGEN_NO_DETAIL
  344. #ifndef DOXYGEN_NO_DISPATCH
  345. namespace dispatch
  346. {
  347. template
  348. <
  349. typename Geometry,
  350. typename Tag = typename tag<Geometry>::type
  351. >
  352. struct centroid: not_implemented<Tag>
  353. {};
  354. template <typename Geometry>
  355. struct centroid<Geometry, point_tag>
  356. : detail::centroid::centroid_point
  357. {};
  358. template <typename Box>
  359. struct centroid<Box, box_tag>
  360. : detail::centroid::centroid_indexed
  361. {};
  362. template <typename Segment>
  363. struct centroid<Segment, segment_tag>
  364. : detail::centroid::centroid_indexed
  365. {};
  366. template <typename Ring>
  367. struct centroid<Ring, ring_tag>
  368. : detail::centroid::centroid_linear_areal
  369. <
  370. detail::centroid::centroid_range<geometry::closure<Ring>::value>
  371. >
  372. {};
  373. template <typename Linestring>
  374. struct centroid<Linestring, linestring_tag>
  375. : detail::centroid::centroid_linear_areal
  376. <
  377. detail::centroid::centroid_range<closed>
  378. >
  379. {};
  380. template <typename Polygon>
  381. struct centroid<Polygon, polygon_tag>
  382. : detail::centroid::centroid_linear_areal
  383. <
  384. detail::centroid::centroid_polygon
  385. >
  386. {};
  387. template <typename MultiLinestring>
  388. struct centroid<MultiLinestring, multi_linestring_tag>
  389. : detail::centroid::centroid_linear_areal
  390. <
  391. detail::centroid::centroid_multi
  392. <
  393. detail::centroid::centroid_range_state<closed>
  394. >
  395. >
  396. {};
  397. template <typename MultiPolygon>
  398. struct centroid<MultiPolygon, multi_polygon_tag>
  399. : detail::centroid::centroid_linear_areal
  400. <
  401. detail::centroid::centroid_multi
  402. <
  403. detail::centroid::centroid_polygon_state
  404. >
  405. >
  406. {};
  407. template <typename MultiPoint>
  408. struct centroid<MultiPoint, multi_point_tag>
  409. : detail::centroid::centroid_multi
  410. <
  411. detail::centroid::centroid_multi_point_state
  412. >
  413. {};
  414. } // namespace dispatch
  415. #endif // DOXYGEN_NO_DISPATCH
  416. namespace resolve_strategy {
  417. template <typename Geometry>
  418. struct centroid
  419. {
  420. template <typename Point, typename Strategy>
  421. static inline void apply(Geometry const& geometry, Point& out, Strategy const& strategy)
  422. {
  423. dispatch::centroid<Geometry>::apply(geometry, out, strategy);
  424. }
  425. template <typename Point>
  426. static inline void apply(Geometry const& geometry, Point& out, default_strategy)
  427. {
  428. typedef typename strategy::centroid::services::default_strategy
  429. <
  430. typename cs_tag<Geometry>::type,
  431. typename tag_cast
  432. <
  433. typename tag<Geometry>::type,
  434. pointlike_tag,
  435. linear_tag,
  436. areal_tag
  437. >::type,
  438. dimension<Geometry>::type::value,
  439. Point,
  440. Geometry
  441. >::type strategy_type;
  442. dispatch::centroid<Geometry>::apply(geometry, out, strategy_type());
  443. }
  444. };
  445. } // namespace resolve_strategy
  446. namespace resolve_variant {
  447. template <typename Geometry>
  448. struct centroid
  449. {
  450. template <typename Point, typename Strategy>
  451. static inline void apply(Geometry const& geometry, Point& out, Strategy const& strategy)
  452. {
  453. concept::check_concepts_and_equal_dimensions<Point, Geometry const>();
  454. resolve_strategy::centroid<Geometry>::apply(geometry, out, strategy);
  455. }
  456. };
  457. template <BOOST_VARIANT_ENUM_PARAMS(typename T)>
  458. struct centroid<boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> >
  459. {
  460. template <typename Point, typename Strategy>
  461. struct visitor: boost::static_visitor<void>
  462. {
  463. Point& m_out;
  464. Strategy const& m_strategy;
  465. visitor(Point& out, Strategy const& strategy)
  466. : m_out(out), m_strategy(strategy)
  467. {}
  468. template <typename Geometry>
  469. void operator()(Geometry const& geometry) const
  470. {
  471. centroid<Geometry>::apply(geometry, m_out, m_strategy);
  472. }
  473. };
  474. template <typename Point, typename Strategy>
  475. static inline void
  476. apply(boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> const& geometry,
  477. Point& out,
  478. Strategy const& strategy)
  479. {
  480. boost::apply_visitor(visitor<Point, Strategy>(out, strategy), geometry);
  481. }
  482. };
  483. } // namespace resolve_variant
  484. /*!
  485. \brief \brief_calc{centroid} \brief_strategy
  486. \ingroup centroid
  487. \details \details_calc{centroid,geometric center (or: center of mass)}. \details_strategy_reasons
  488. \tparam Geometry \tparam_geometry
  489. \tparam Point \tparam_point
  490. \tparam Strategy \tparam_strategy{Centroid}
  491. \param geometry \param_geometry
  492. \param c \param_point \param_set{centroid}
  493. \param strategy \param_strategy{centroid}
  494. \qbk{distinguish,with strategy}
  495. \qbk{[include reference/algorithms/centroid.qbk]}
  496. \qbk{[include reference/algorithms/centroid_strategies.qbk]}
  497. }
  498. */
  499. template<typename Geometry, typename Point, typename Strategy>
  500. inline void centroid(Geometry const& geometry, Point& c,
  501. Strategy const& strategy)
  502. {
  503. resolve_variant::centroid<Geometry>::apply(geometry, c, strategy);
  504. }
  505. /*!
  506. \brief \brief_calc{centroid}
  507. \ingroup centroid
  508. \details \details_calc{centroid,geometric center (or: center of mass)}. \details_default_strategy
  509. \tparam Geometry \tparam_geometry
  510. \tparam Point \tparam_point
  511. \param geometry \param_geometry
  512. \param c The calculated centroid will be assigned to this point reference
  513. \qbk{[include reference/algorithms/centroid.qbk]}
  514. \qbk{
  515. [heading Example]
  516. [centroid]
  517. [centroid_output]
  518. }
  519. */
  520. template<typename Geometry, typename Point>
  521. inline void centroid(Geometry const& geometry, Point& c)
  522. {
  523. geometry::centroid(geometry, c, default_strategy());
  524. }
  525. /*!
  526. \brief \brief_calc{centroid}
  527. \ingroup centroid
  528. \details \details_calc{centroid,geometric center (or: center of mass)}. \details_return{centroid}.
  529. \tparam Point \tparam_point
  530. \tparam Geometry \tparam_geometry
  531. \param geometry \param_geometry
  532. \return \return_calc{centroid}
  533. \qbk{[include reference/algorithms/centroid.qbk]}
  534. */
  535. template<typename Point, typename Geometry>
  536. inline Point return_centroid(Geometry const& geometry)
  537. {
  538. Point c;
  539. geometry::centroid(geometry, c);
  540. return c;
  541. }
  542. /*!
  543. \brief \brief_calc{centroid} \brief_strategy
  544. \ingroup centroid
  545. \details \details_calc{centroid,geometric center (or: center of mass)}. \details_return{centroid}. \details_strategy_reasons
  546. \tparam Point \tparam_point
  547. \tparam Geometry \tparam_geometry
  548. \tparam Strategy \tparam_strategy{centroid}
  549. \param geometry \param_geometry
  550. \param strategy \param_strategy{centroid}
  551. \return \return_calc{centroid}
  552. \qbk{distinguish,with strategy}
  553. \qbk{[include reference/algorithms/centroid.qbk]}
  554. \qbk{[include reference/algorithms/centroid_strategies.qbk]}
  555. */
  556. template<typename Point, typename Geometry, typename Strategy>
  557. inline Point return_centroid(Geometry const& geometry, Strategy const& strategy)
  558. {
  559. Point c;
  560. geometry::centroid(geometry, c, strategy);
  561. return c;
  562. }
  563. }} // namespace boost::geometry
  564. #endif // BOOST_GEOMETRY_ALGORITHMS_CENTROID_HPP