// Boost.Geometry // Copyright (c) 2017 Barend Gehrels, Amsterdam, the Netherlands. // This file was modified by Oracle on 2020-2023. // Modifications copyright (c) 2020-2023 Oracle and/or its affiliates. // Contributed and/or modified by Vissarion Fysikopoulos, on behalf of Oracle // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle // Use, modification and distribution is subject to the Boost Software License, // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_GEOMETRY_ALGORITHMS_CORRECT_CLOSURE_HPP #define BOOST_GEOMETRY_ALGORITHMS_CORRECT_CLOSURE_HPP #include #include #include #include #include #include #include #include #include #include namespace boost { namespace geometry { // Silence warning C4127: conditional expression is constant #if defined(_MSC_VER) #pragma warning(push) #pragma warning(disable : 4127) #endif #ifndef DOXYGEN_NO_DETAIL namespace detail { namespace correct_closure { struct nop { template static inline void apply(Geometry& ) {} }; // Close a ring, if not closed, or open it struct close_or_open_ring { template static inline void apply(Ring& r) { auto size = boost::size(r); if (size <= 2) { return; } // TODO: This requires relate(pt, pt) strategy bool const disjoint = geometry::disjoint(*boost::begin(r), *(boost::end(r) - 1)); closure_selector const closure = geometry::closure::value; if (disjoint && closure == closed) { // Close it by adding first point geometry::append(r, *boost::begin(r)); } else if (! disjoint && closure == open) { // Open it by removing last point range::resize(r, size - 1); } } }; // Close/open exterior ring and all its interior rings struct close_or_open_polygon { template static inline void apply(Polygon& poly) { close_or_open_ring::apply(exterior_ring(poly)); auto&& rings = interior_rings(poly); auto const end = boost::end(rings); for (auto it = boost::begin(rings); it != end; ++it) { close_or_open_ring::apply(*it); } } }; }} // namespace detail::correct_closure #endif // DOXYGEN_NO_DETAIL #ifndef DOXYGEN_NO_DISPATCH namespace dispatch { template ::type> struct correct_closure: not_implemented {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::correct_closure::close_or_open_ring {}; template struct correct_closure : detail::correct_closure::close_or_open_polygon {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::correct_closure::nop {}; template struct correct_closure : detail::multi_modify < detail::correct_closure::close_or_open_polygon > {}; } // namespace dispatch #endif // DOXYGEN_NO_DISPATCH namespace resolve_variant { template ::type> struct correct_closure { static inline void apply(Geometry& geometry) { concepts::check(); dispatch::correct_closure::apply(geometry); } }; template struct correct_closure { static void apply(Geometry& geometry) { traits::visit::apply([](auto & g) { correct_closure>::apply(g); }, geometry); } }; template struct correct_closure { static void apply(Geometry& geometry) { detail::visit_breadth_first([](auto & g) { correct_closure>::apply(g); return true; }, geometry); } }; } // namespace resolve_variant // TODO: This algorithm should use relate(pt, pt) strategy /*! \brief Closes or opens a geometry, according to its type \details Corrects a geometry w.r.t. closure points to all rings which do not have a closing point and are typed as they should have one, the first point is appended. \ingroup correct_closure \tparam Geometry \tparam_geometry \param geometry \param_geometry which will be corrected if necessary */ template inline void correct_closure(Geometry& geometry) { resolve_variant::correct_closure::apply(geometry); } #if defined(_MSC_VER) #pragma warning(pop) #endif }} // namespace boost::geometry #endif // BOOST_GEOMETRY_ALGORITHMS_CORRECT_CLOSURE_HPP