123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297 |
- // Boost.Geometry (aka GGL, Generic Geometry Library)
- // Copyright (c) 2007-2012 Barend Gehrels, Amsterdam, the Netherlands.
- // Copyright (c) 2013 Adam Wulkiewicz, Lodz, Poland.
- // This file was modified by Oracle on 2013-2021.
- // Modifications copyright (c) 2013-2021 Oracle and/or its affiliates.
- // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle
- // Parts of Boost.Geometry are redesigned from Geodan's Geographic Library
- // (geolib/GGL), copyright (c) 1995-2010 Geodan, Amsterdam, the Netherlands.
- // 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_STRATEGY_CARTESIAN_POINT_IN_POLY_WINDING_HPP
- #define BOOST_GEOMETRY_STRATEGY_CARTESIAN_POINT_IN_POLY_WINDING_HPP
- #include <boost/geometry/core/tags.hpp>
- #include <boost/geometry/util/math.hpp>
- #include <boost/geometry/util/select_calculation_type.hpp>
- #include <boost/geometry/strategy/cartesian/expand_point.hpp>
- #include <boost/geometry/strategies/side.hpp>
- #include <boost/geometry/strategies/cartesian/point_in_box.hpp>
- #include <boost/geometry/strategies/cartesian/disjoint_box_box.hpp>
- #include <boost/geometry/strategies/covered_by.hpp>
- #include <boost/geometry/strategies/within.hpp>
- namespace boost { namespace geometry
- {
- namespace strategy { namespace within
- {
- #ifndef DOXYGEN_NO_DETAIL
- namespace detail
- {
- /*!
- \brief Within detection using winding rule in cartesian coordinate system.
- \ingroup strategies
- \tparam SideStrategy A strategy defining creation along sides
- \tparam CalculationType \tparam_calculation
- */
- template <typename SideStrategy, typename CalculationType>
- class cartesian_winding_base
- {
- template <typename Point, typename PointOfSegment>
- struct calculation_type
- : select_calculation_type
- <
- Point,
- PointOfSegment,
- CalculationType
- >
- {};
- /*! subclass to keep state */
- class counter
- {
- int m_count;
- bool m_touches;
- inline int code() const
- {
- return m_touches ? 0 : m_count == 0 ? -1 : 1;
- }
- public :
- friend class cartesian_winding_base;
- inline counter()
- : m_count(0)
- , m_touches(false)
- {}
- };
- public:
- typedef cartesian_tag cs_tag;
- // Typedefs and static methods to fulfill the concept
- typedef counter state_type;
- template <typename Point, typename PointOfSegment>
- static inline bool apply(Point const& point,
- PointOfSegment const& s1, PointOfSegment const& s2,
- counter& state)
- {
- bool eq1 = false;
- bool eq2 = false;
- int count = check_segment(point, s1, s2, state, eq1, eq2);
- if (count != 0)
- {
- int side = 0;
- if (count == 1 || count == -1)
- {
- side = side_equal(point, eq1 ? s1 : s2, count);
- }
- else // count == 2 || count == -2
- {
- // 1 left, -1 right
- side = SideStrategy::apply(s1, s2, point);
- }
- if (side == 0)
- {
- // Point is lying on segment
- state.m_touches = true;
- state.m_count = 0;
- return false;
- }
- // Side is NEG for right, POS for left.
- // The count is -2 for left, 2 for right (or -1/1)
- // Side positive thus means RIGHT and LEFTSIDE or LEFT and RIGHTSIDE
- // See accompagnying figure (TODO)
- if (side * count > 0)
- {
- state.m_count += count;
- }
- }
- return ! state.m_touches;
- }
- static inline int result(counter const& state)
- {
- return state.code();
- }
- private:
- template <typename Point, typename PointOfSegment>
- static inline int check_segment(Point const& point,
- PointOfSegment const& seg1,
- PointOfSegment const& seg2,
- counter& state,
- bool& eq1, bool& eq2)
- {
- if (check_touch(point, seg1, seg2, state, eq1, eq2))
- {
- return 0;
- }
- return calculate_count(point, seg1, seg2, eq1, eq2);
- }
- template <typename Point, typename PointOfSegment>
- static inline bool check_touch(Point const& point,
- PointOfSegment const& seg1,
- PointOfSegment const& seg2,
- counter& state,
- bool& eq1, bool& eq2)
- {
- typedef typename calculation_type<Point, PointOfSegment>::type calc_t;
- calc_t const px = get<0>(point);
- calc_t const s1x = get<0>(seg1);
- calc_t const s2x = get<0>(seg2);
- eq1 = math::equals(s1x, px);
- eq2 = math::equals(s2x, px);
- // Both equal p -> segment vertical
- // The only thing which has to be done is check if point is ON segment
- if (eq1 && eq2)
- {
- calc_t const py = get<1>(point);
- calc_t const s1y = get<1>(seg1);
- calc_t const s2y = get<1>(seg2);
- if ((s1y <= py && s2y >= py) || (s2y <= py && s1y >= py))
- {
- state.m_touches = true;
- }
- return true;
- }
- return false;
- }
- template <typename Point, typename PointOfSegment>
- static inline int calculate_count(Point const& point,
- PointOfSegment const& seg1,
- PointOfSegment const& seg2,
- bool eq1, bool eq2)
- {
- typedef typename calculation_type<Point, PointOfSegment>::type calc_t;
- calc_t const p = get<0>(point);
- calc_t const s1 = get<0>(seg1);
- calc_t const s2 = get<0>(seg2);
- return eq1 ? (s2 > p ? 1 : -1) // Point on level s1, E/W depending on s2
- : eq2 ? (s1 > p ? -1 : 1) // idem
- : s1 < p && s2 > p ? 2 // Point between s1 -> s2 --> E
- : s2 < p && s1 > p ? -2 // Point between s2 -> s1 --> W
- : 0;
- }
- template <typename Point, typename PointOfSegment>
- static inline int side_equal(Point const& point,
- PointOfSegment const& se,
- int count)
- {
- // NOTE: for D=0 the signs would be reversed
- return math::equals(get<1>(point), get<1>(se)) ?
- 0 :
- get<1>(point) < get<1>(se) ?
- // assuming count is equal to 1 or -1
- -count : // ( count > 0 ? -1 : 1) :
- count; // ( count > 0 ? 1 : -1) ;
- }
- };
- } // namespace detail
- #endif // DOXYGEN_NO_DETAIL
- /*!
- \brief Within detection using winding rule in cartesian coordinate system.
- \ingroup strategies
- \tparam Point_ \tparam_point
- \tparam PointOfSegment_ \tparam_segment_point
- \tparam CalculationType \tparam_calculation
- \qbk{
- [heading See also]
- [link geometry.reference.algorithms.within.within_3_with_strategy within (with strategy)]
- }
- */
- template
- <
- typename Point_ = void, // for backward compatibility
- typename PointOfSegment_ = Point_, // for backward compatibility
- typename CalculationType = void
- >
- class cartesian_winding
- : public detail::cartesian_winding_base
- <
- typename side::services::default_strategy<cartesian_tag, CalculationType>::type,
- CalculationType
- >
- {};
- #ifndef DOXYGEN_NO_STRATEGY_SPECIALIZATIONS
- namespace services
- {
- template <typename PointLike, typename Geometry, typename AnyTag1, typename AnyTag2>
- struct default_strategy<PointLike, Geometry, AnyTag1, AnyTag2, pointlike_tag, polygonal_tag, cartesian_tag, cartesian_tag>
- {
- using type = cartesian_winding<>;
- };
- template <typename PointLike, typename Geometry, typename AnyTag1, typename AnyTag2>
- struct default_strategy<PointLike, Geometry, AnyTag1, AnyTag2, pointlike_tag, linear_tag, cartesian_tag, cartesian_tag>
- {
- using type = cartesian_winding<>;
- };
- } // namespace services
- #endif
- }} // namespace strategy::within
- #ifndef DOXYGEN_NO_STRATEGY_SPECIALIZATIONS
- namespace strategy { namespace covered_by { namespace services
- {
- template <typename PointLike, typename Geometry, typename AnyTag1, typename AnyTag2>
- struct default_strategy<PointLike, Geometry, AnyTag1, AnyTag2, pointlike_tag, polygonal_tag, cartesian_tag, cartesian_tag>
- {
- using type = within::cartesian_winding<>;
- };
- template <typename PointLike, typename Geometry, typename AnyTag1, typename AnyTag2>
- struct default_strategy<PointLike, Geometry, AnyTag1, AnyTag2, pointlike_tag, linear_tag, cartesian_tag, cartesian_tag>
- {
- using type = within::cartesian_winding<>;
- };
- }}} // namespace strategy::covered_by::services
- #endif
- }} // namespace boost::geometry
- #endif // BOOST_GEOMETRY_STRATEGY_CARTESIAN_POINT_IN_POLY_WINDING_HPP
|