// Boost.Geometry // Copyright (c) 2017-2021, 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 // Licensed under the Boost Software License version 1.0. // http://www.boost.org/users/license.html #ifndef BOOST_GEOMETRY_STRATEGIES_SPHERICAL_DENSIFY_HPP #define BOOST_GEOMETRY_STRATEGIES_SPHERICAL_DENSIFY_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace boost { namespace geometry { namespace strategy { namespace densify { /*! \brief Densification of spherical segment. \ingroup strategies \tparam RadiusTypeOrSphere \tparam_radius_or_sphere \tparam CalculationType \tparam_calculation \qbk{ [heading See also] [link geometry.reference.algorithms.densify.densify_4_with_strategy densify (with strategy)] } */ template < typename RadiusTypeOrSphere = double, typename CalculationType = void > class spherical { public: typedef typename strategy_detail::get_radius < RadiusTypeOrSphere >::type radius_type; // For consistency with area strategy the radius is set to 1 inline spherical() : m_radius(1.0) {} template explicit inline spherical(RadiusOrSphere const& radius_or_sphere) : m_radius(strategy_detail::get_radius < RadiusOrSphere >::apply(radius_or_sphere)) {} template inline void apply(Point const& p0, Point const& p1, AssignPolicy & policy, T const& length_threshold) const { typedef typename AssignPolicy::point_type out_point_t; typedef typename select_most_precise < typename coordinate_type::type, typename coordinate_type::type, CalculationType >::type calc_t; calc_t angle01; formula::interpolate_point_spherical formula; formula.compute_angle(p0, p1, angle01); BOOST_GEOMETRY_ASSERT(length_threshold > T(0)); signed_size_type n = signed_size_type(angle01 * m_radius / length_threshold); if (n <= 0) return; formula.compute_axis(p0, angle01); calc_t step = angle01 / (n + 1); calc_t a = step; for (signed_size_type i = 0 ; i < n ; ++i, a += step) { out_point_t p; formula.compute_point(a, p); geometry::detail::conversion::point_to_point < Point, out_point_t, 2, dimension::value >::apply(p0, p); policy.apply(p); } } inline radius_type radius() const { return m_radius; } private: radius_type m_radius; }; #ifndef DOXYGEN_NO_STRATEGY_SPECIALIZATIONS namespace services { template <> struct default_strategy { typedef strategy::densify::spherical<> type; }; } // namespace services #endif // DOXYGEN_NO_STRATEGY_SPECIALIZATIONS }} // namespace strategy::densify }} // namespace boost::geometry #endif // BOOST_GEOMETRY_ALGORITHMS_DENSIFY_HPP