// Boost.Geometry (aka GGL, Generic Geometry Library) // Copyright (c) 2014-2020, Oracle and/or its affiliates. // Contributed and/or modified by Menelaos Karavelas, on behalf of Oracle // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle // Licensed under the Boost Software License version 1.0. // http://www.boost.org/users/license.html #ifndef BOOST_GEOMETRY_STRATEGIES_COMPARABLE_DISTANCE_RESULT_HPP #define BOOST_GEOMETRY_STRATEGIES_COMPARABLE_DISTANCE_RESULT_HPP #include #include #include #include #include #include #include #include namespace boost { namespace geometry { namespace resolve_strategy { template < typename Geometry1, typename Geometry2, typename Strategy, bool AreGeometries = (util::is_geometry::value && util::is_geometry::value) > struct comparable_distance_result : strategy::distance::services::return_type < typename strategy::distance::services::comparable_type < Strategy >::type, typename point_type::type, typename point_type::type > {}; template struct comparable_distance_result : comparable_distance_result < Geometry1, Geometry2, typename detail::distance::default_strategy < Geometry1, Geometry2 >::type > {}; // Workaround for VS2015 #if defined(_MSC_VER) && (_MSC_VER < 1910) template struct comparable_distance_result { typedef int type; }; template struct comparable_distance_result { typedef int type; }; #endif } // namespace resolve_strategy #ifndef DOXYGEN_NO_DETAIL namespace detail { namespace distance { template struct more_precise_comparable_distance_result { template struct predicate : std::is_same < typename resolve_strategy::comparable_distance_result < typename util::sequence_element<0, Curr>::type, typename util::sequence_element<1, Curr>::type, Strategy >::type, typename geometry::select_most_precise < typename resolve_strategy::comparable_distance_result < typename util::sequence_element<0, Curr>::type, typename util::sequence_element<1, Curr>::type, Strategy >::type, typename resolve_strategy::comparable_distance_result < typename util::sequence_element<0, Next>::type, typename util::sequence_element<1, Next>::type, Strategy >::type >::type > {}; }; }} // namespace detail::distance #endif //DOXYGEN_NO_DETAIL namespace resolve_variant { template struct comparable_distance_result : resolve_strategy::comparable_distance_result < Geometry1, Geometry2, Strategy > {}; template < typename Geometry1, BOOST_VARIANT_ENUM_PARAMS(typename T), typename Strategy > struct comparable_distance_result < Geometry1, boost::variant, Strategy > { // Select the most precise distance strategy result type // for all variant type combinations. // TODO: We should ignore the combinations that are not valid // but is_implemented is not ready for prime time. typedef typename util::select_combination_element < util::type_sequence, util::type_sequence, detail::distance::more_precise_comparable_distance_result < Strategy >::template predicate >::type elements; typedef typename resolve_strategy::comparable_distance_result < typename util::sequence_element<0, elements>::type, typename util::sequence_element<1, elements>::type, Strategy >::type type; }; // Distance arguments are commutative template < BOOST_VARIANT_ENUM_PARAMS(typename T), typename Geometry2, typename Strategy > struct comparable_distance_result < boost::variant, Geometry2, Strategy > : public comparable_distance_result < Geometry2, boost::variant, Strategy > {}; template < BOOST_VARIANT_ENUM_PARAMS(typename T), BOOST_VARIANT_ENUM_PARAMS(typename U), typename Strategy > struct comparable_distance_result < boost::variant, boost::variant, Strategy > { // Select the most precise distance strategy result type // for all variant type combinations. // TODO: We should ignore the combinations that are not valid // but is_implemented is not ready for prime time. typedef typename util::select_combination_element < util::type_sequence, util::type_sequence, detail::distance::more_precise_comparable_distance_result < Strategy >::template predicate >::type elements; typedef typename resolve_strategy::comparable_distance_result < typename util::sequence_element<0, elements>::type, typename util::sequence_element<1, elements>::type, Strategy >::type type; }; } // namespace resolve_variant /*! \brief Meta-function defining return type of comparable_distance function \ingroup distance */ template < typename Geometry1, typename Geometry2 = Geometry1, typename Strategy = void > struct comparable_distance_result : resolve_variant::comparable_distance_result < Geometry1, Geometry2, Strategy > {}; template struct comparable_distance_result : comparable_distance_result {}; }} // namespace boost::geometry #endif // BOOST_GEOMETRY_STRATEGIES_COMPARABLE_DISTANCE_RESULT_HPP