// Boost.Geometry (aka GGL, Generic Geometry Library) // Copyright (c) 2007-2012 Barend Gehrels, Amsterdam, the Netherlands. // This file was modified by Oracle on 2017. // Modifications copyright (c) 2017, Oracle and/or its affiliates. // 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_STRATEGIES_SPHERICAL_COMPARE_HPP #define BOOST_GEOMETRY_STRATEGIES_SPHERICAL_COMPARE_HPP #include #include #include #include #include #include #include #include #include #include #include #include namespace boost { namespace geometry { namespace strategy { namespace compare { #ifndef DOXYGEN_NO_DETAIL namespace detail { template static inline typename geometry::coordinate_type

::type get(P const& p, boost::true_type /*same units*/) { return geometry::get(p); } template static inline typename geometry::coordinate_type

::type get(P const& p, boost::false_type /*different units*/) { return geometry::get_as_radian(p); } template < typename ComparePolicy, typename Point1, typename Point2, std::size_t DimensionCount > struct spherical_latitude { typedef typename geometry::coordinate_type::type coordinate1_type; typedef typename geometry::coordinate_system::type::units units1_type; typedef typename geometry::coordinate_type::type coordinate2_type; typedef typename geometry::coordinate_system::type::units units2_type; typedef typename boost::is_same::type same_units_type; template static inline bool apply(Point1 const& left, Point2 const& right, T1 const& l1, T2 const& r1) { // latitudes equal if (math::equals(l1, r1)) { return compare::detail::compare_loop < ComparePolicy, 2, DimensionCount >::apply(left, right); } else { return ComparePolicy::apply(l1, r1); } } static inline bool apply(Point1 const& left, Point2 const& right) { coordinate1_type const& l1 = compare::detail::get<1>(left, same_units_type()); coordinate2_type const& r1 = compare::detail::get<1>(right, same_units_type()); return apply(left, right, l1, r1); } }; template < typename ComparePolicy, typename Point1, typename Point2 > struct spherical_latitude { template static inline bool apply(Point1 const& left, Point2 const& right, T1 const& , T2 const& ) { return apply(left, right); } static inline bool apply(Point1 const& left, Point2 const& right) { return compare::detail::compare_loop < ComparePolicy, 1, 1 >::apply(left, right); } }; template < typename ComparePolicy, typename Point1, typename Point2, std::size_t DimensionCount > struct spherical_longitude { typedef typename geometry::coordinate_type::type coordinate1_type; typedef typename geometry::coordinate_system::type::units units1_type; typedef typename geometry::coordinate_type::type coordinate2_type; typedef typename geometry::coordinate_system::type::units units2_type; typedef typename boost::is_same::type same_units_type; typedef typename boost::mpl::if_::type units_type; static const bool is_equatorial = ! boost::is_same < typename geometry::cs_tag::type, geometry::spherical_polar_tag >::value; static inline bool are_both_at_antimeridian(coordinate1_type const& l0, coordinate2_type const& r0, bool & is_left_at, bool & is_right_at) { is_left_at = math::is_longitude_antimeridian(l0); is_right_at = math::is_longitude_antimeridian(r0); return is_left_at && is_right_at; } static inline bool apply(Point1 const& left, Point2 const& right) { // if units are different the coordinates are in radians coordinate1_type const& l0 = compare::detail::get<0>(left, same_units_type()); coordinate2_type const& r0 = compare::detail::get<0>(right, same_units_type()); coordinate1_type const& l1 = compare::detail::get<1>(left, same_units_type()); coordinate2_type const& r1 = compare::detail::get<1>(right, same_units_type()); bool is_left_at_antimeridian = false; bool is_right_at_antimeridian = false; // longitudes equal if (math::equals(l0, r0) // both at antimeridian || are_both_at_antimeridian(l0, r0, is_left_at_antimeridian, is_right_at_antimeridian) // both at pole || (math::equals(l1, r1) && math::is_latitude_pole(l1))) { return spherical_latitude < ComparePolicy, Point1, Point2, DimensionCount >::apply(left, right, l1, r1); } // if left is at antimeridian and right is not at antimeridian // then left is greater than right else if (is_left_at_antimeridian) { // less/equal_to -> false, greater -> true return ComparePolicy::apply(1, 0); } // if right is at antimeridian and left is not at antimeridian // then left is lesser than right else if (is_right_at_antimeridian) { // less -> true, equal_to/greater -> false return ComparePolicy::apply(0, 1); } else { return ComparePolicy::apply(l0, r0); } } }; } // namespace detail #endif // DOXYGEN_NO_DETAIL /*! \brief Compare strategy for spherical coordinates \ingroup strategies \tparam Point point-type \tparam Dimension dimension */ template < typename ComparePolicy, int Dimension = -1 > struct spherical : cartesian {}; #ifndef DOXYGEN_NO_STRATEGY_SPECIALIZATIONS // all dimensions starting from longitude template struct spherical { template static inline bool apply(Point1 const& left, Point2 const& right) { return compare::detail::spherical_longitude < ComparePolicy, Point1, Point2, boost::mpl::min < geometry::dimension, geometry::dimension >::type::value >::apply(left, right); } }; // only longitudes (and latitudes to check poles) template struct spherical { template static inline bool apply(Point1 const& left, Point2 const& right) { return compare::detail::spherical_longitude < ComparePolicy, Point1, Point2, 1 >::apply(left, right); } }; // only latitudes template struct spherical { template static inline bool apply(Point1 const& left, Point2 const& right) { return compare::detail::spherical_latitude < ComparePolicy, Point1, Point2, 2 >::apply(left, right); } }; #endif // DOXYGEN_NO_STRATEGY_SPECIALIZATIONS namespace services { template struct default_strategy < ComparePolicy, Point1, Point2, Dimension, spherical_polar_tag, spherical_polar_tag > { typedef compare::spherical type; }; template struct default_strategy < ComparePolicy, Point1, Point2, Dimension, spherical_equatorial_tag, spherical_equatorial_tag > { typedef compare::spherical type; }; template struct default_strategy < ComparePolicy, Point1, Point2, Dimension, geographic_tag, geographic_tag > { typedef compare::spherical type; }; } // namespace services }} // namespace strategy::compare }} // namespace boost::geometry #endif // BOOST_GEOMETRY_STRATEGIES_SPHERICAL_COMPARE_HPP