[line_interpolate] Add support for umbrella strategies.

This commit is contained in:
Adam Wulkiewicz
2021-04-02 03:04:57 +02:00
parent 5f262f8d19
commit 8a80effc0d
7 changed files with 397 additions and 35 deletions

View File

@@ -1,6 +1,6 @@
// Boost.Geometry (aka GGL, Generic Geometry Library)
// Copyright (c) 2018-2020 Oracle and/or its affiliates.
// Copyright (c) 2018-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
@@ -20,6 +20,9 @@
#include <boost/range/iterator.hpp>
#include <boost/range/value_type.hpp>
#include <boost/geometry/algorithms/detail/convert_point_to_point.hpp>
#include <boost/geometry/algorithms/detail/dummy_geometries.hpp>
#include <boost/geometry/core/cs.hpp>
#include <boost/geometry/core/closure.hpp>
#include <boost/geometry/core/static_assert.hpp>
@@ -27,10 +30,9 @@
#include <boost/geometry/geometries/concepts/check.hpp>
#include <boost/geometry/algorithms/assign.hpp>
#include <boost/geometry/algorithms/length.hpp>
#include <boost/geometry/strategies/default_strategy.hpp>
#include <boost/geometry/strategies/line_interpolate.hpp>
#include <boost/geometry/strategies/line_interpolate/cartesian.hpp>
#include <boost/geometry/strategies/line_interpolate/geographic.hpp>
#include <boost/geometry/strategies/line_interpolate/spherical.hpp>
namespace boost { namespace geometry
{
@@ -74,12 +76,12 @@ struct interpolate_range
typename Range,
typename Distance,
typename PointLike,
typename Strategy
typename Strategies
>
static inline void apply(Range const& range,
Distance const& max_distance,
PointLike & pointlike,
Strategy const& strategy)
Strategies const& strategies)
{
Policy policy;
@@ -100,6 +102,9 @@ struct interpolate_range
return;
}
auto const pp_strategy = strategies.distance(dummy_point(), dummy_point());
auto const strategy = strategies.line_interpolate(range);
iterator_t prev = it++;
Distance repeated_distance = max_distance;
Distance prev_distance = 0;
@@ -108,7 +113,7 @@ struct interpolate_range
for ( ; it != end ; ++it)
{
Distance dist = strategy.get_distance_pp_strategy().apply(*prev, *it);
Distance dist = pp_strategy.apply(*prev, *it);
current_distance = prev_distance + dist;
while (current_distance >= repeated_distance)
@@ -219,41 +224,63 @@ struct line_interpolate<Geometry, Pointlike, segment_tag, multi_point_tag>
namespace resolve_strategy {
template
<
typename Strategies,
bool IsUmbrella = strategies::detail::is_umbrella_strategy<Strategies>::value
>
struct line_interpolate
{
template
<
typename Geometry,
typename Distance,
typename Pointlike,
typename Strategy
>
template <typename Geometry, typename Distance, typename Pointlike>
static inline void apply(Geometry const& geometry,
Distance const& max_distance,
Pointlike & pointlike,
Strategies const& strategies)
{
dispatch::line_interpolate
<
Geometry, Pointlike
>::apply(geometry, max_distance, pointlike, strategies);
}
};
template <typename Strategy>
struct line_interpolate<Strategy, false>
{
template <typename Geometry, typename Distance, typename Pointlike>
static inline void apply(Geometry const& geometry,
Distance const& max_distance,
Pointlike & pointlike,
Strategy const& strategy)
{
dispatch::line_interpolate<Geometry, Pointlike>::apply(geometry,
max_distance,
pointlike,
strategy);
}
{
using strategies::line_interpolate::services::strategy_converter;
dispatch::line_interpolate
<
Geometry, Pointlike
>::apply(geometry, max_distance, pointlike,
strategy_converter<Strategy>::get(strategy));
}
};
template <>
struct line_interpolate<default_strategy, false>
{
template <typename Geometry, typename Distance, typename Pointlike>
static inline void apply(Geometry const& geometry,
Distance const& max_distance,
Pointlike & pointlike,
default_strategy)
{
typedef typename strategy::line_interpolate::services::default_strategy
typedef typename strategies::line_interpolate::services::default_strategy
<
typename cs_tag<Geometry>::type
Geometry
>::type strategy_type;
dispatch::line_interpolate<Geometry, Pointlike>::apply(geometry,
max_distance,
pointlike,
strategy_type());
dispatch::line_interpolate
<
Geometry, Pointlike
>::apply(geometry, max_distance, pointlike, strategy_type());
}
};
@@ -271,10 +298,10 @@ struct line_interpolate
Pointlike & pointlike,
Strategy const& strategy)
{
return resolve_strategy::line_interpolate::apply(geometry,
max_distance,
pointlike,
strategy);
return resolve_strategy::line_interpolate
<
Strategy
>::apply(geometry, max_distance, pointlike, strategy);
}
};
@@ -303,7 +330,7 @@ struct line_interpolate<boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> >
template <typename Distance, typename Pointlike, typename Strategy>
static inline void
apply(boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> const& geometry,
double const& max_distance,
Distance const& max_distance,
Pointlike & pointlike,
Strategy const& strategy)
{
@@ -318,7 +345,7 @@ struct line_interpolate<boost::variant<BOOST_VARIANT_ENUM_PARAMS(T)> >
} // namespace resolve_variant
/*!
\brief Returns one or more points interpolated along a LineString \brief_strategy
\brief Returns one or more points interpolated along a LineString \brief_strategy
\ingroup line_interpolate
\tparam Geometry Any type fulfilling a LineString concept
\tparam Distance A numerical distance measure
@@ -372,7 +399,7 @@ inline void line_interpolate(Geometry const& geometry,
/*!
\brief Returns one or more points interpolated along a LineString.
\brief Returns one or more points interpolated along a LineString.
\ingroup line_interpolate
\tparam Geometry Any type fulfilling a LineString concept
\tparam Distance A numerical distance measure

View File

@@ -101,6 +101,11 @@ public:
set_from_radian<1>(p, dir_r.lat2);
}
inline Spheroid model() const
{
return m_spheroid;
}
private:
Spheroid m_spheroid;
};

View File

@@ -0,0 +1,78 @@
// Boost.Geometry
// Copyright (c) 2021, Oracle and/or its affiliates.
// 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_LINE_INTERPOLATE_CARTESIAN_HPP
#define BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_CARTESIAN_HPP
#include <boost/geometry/strategies/cartesian/distance_pythagoras.hpp>
#include <boost/geometry/strategies/cartesian/line_interpolate.hpp>
#include <boost/geometry/strategies/detail.hpp>
#include <boost/geometry/strategies/line_interpolate/services.hpp>
#include <boost/geometry/util/type_traits.hpp>
namespace boost { namespace geometry
{
namespace strategies { namespace line_interpolate
{
template <typename CalculationType = void>
struct cartesian
: public strategies::detail::cartesian_base
{
template <typename Geometry1, typename Geometry2>
static auto distance(Geometry1 const&, Geometry2 const&,
std::enable_if_t
<
util::is_pointlike<Geometry1>::value
&& util::is_pointlike<Geometry2>::value
> * = nullptr)
{
return strategy::distance::pythagoras<CalculationType>();
}
template <typename Geometry>
static auto line_interpolate(Geometry const&)
{
return strategy::line_interpolate::cartesian<CalculationType>();
}
};
namespace services
{
template <typename Geometry>
struct default_strategy<Geometry, cartesian_tag>
{
using type = strategies::line_interpolate::cartesian<>;
};
template <typename CT, typename DS>
struct strategy_converter<strategy::line_interpolate::cartesian<CT, DS> >
{
static auto get(strategy::line_interpolate::cartesian<CT, DS> const&)
{
return strategies::line_interpolate::cartesian<CT>();
}
};
} // namespace services
}} // namespace strategies::line_interpolate
}} // namespace boost::geometry
#endif // BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_CARTESIAN_HPP

View File

@@ -0,0 +1,97 @@
// Boost.Geometry
// Copyright (c) 2021, Oracle and/or its affiliates.
// 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_LINE_INTERPOLATE_GEOGRAPHIC_HPP
#define BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_GEOGRAPHIC_HPP
#include <boost/geometry/strategies/detail.hpp>
#include <boost/geometry/strategies/geographic/distance.hpp>
#include <boost/geometry/strategies/geographic/line_interpolate.hpp>
#include <boost/geometry/strategies/line_interpolate/services.hpp>
namespace boost { namespace geometry
{
namespace strategies { namespace line_interpolate
{
template
<
typename FormulaPolicy = strategy::andoyer,
typename Spheroid = srs::spheroid<double>,
typename CalculationType = void
>
class geographic
: public strategies::detail::geographic_base<Spheroid>
{
using base_t = strategies::detail::geographic_base<Spheroid>;
public:
geographic() = default;
explicit geographic(Spheroid const& spheroid)
: base_t(spheroid)
{}
template <typename Geometry1, typename Geometry2>
auto distance(Geometry1 const&, Geometry2 const&,
std::enable_if_t
<
util::is_pointlike<Geometry1>::value
&& util::is_pointlike<Geometry2>::value
> * = nullptr) const
{
return strategy::distance::geographic
<
FormulaPolicy, Spheroid, CalculationType
>(base_t::m_spheroid);
}
template <typename Geometry>
auto line_interpolate(Geometry const&) const
{
return strategy::line_interpolate::geographic
<
FormulaPolicy, Spheroid, CalculationType
>(base_t::m_spheroid);
}
};
namespace services
{
template <typename Geometry>
struct default_strategy<Geometry, geographic_tag>
{
using type = strategies::line_interpolate::geographic<>;
};
template <typename FP, typename S, typename CT>
struct strategy_converter<strategy::line_interpolate::geographic<FP, S, CT> >
{
static auto get(strategy::line_interpolate::geographic<FP, S, CT> const& s)
{
return strategies::line_interpolate::geographic<FP, S, CT>(s.model());
}
};
} // namespace services
}} // namespace strategies::line_interpolate
}} // namespace boost::geometry
#endif // BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_GEOGRAPHIC_HPP

View File

@@ -0,0 +1,54 @@
// Boost.Geometry
// Copyright (c) 2021, Oracle and/or its affiliates.
// 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_LINE_INTERPOLATE_SERVICES_HPP
#define BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_SERVICES_HPP
#include <boost/geometry/core/cs.hpp>
#include <boost/geometry/core/static_assert.hpp>
namespace boost { namespace geometry
{
namespace strategies { namespace line_interpolate
{
namespace services
{
template
<
typename Geometry,
typename CSTag = typename geometry::cs_tag<Geometry>::type
>
struct default_strategy
{
BOOST_GEOMETRY_STATIC_ASSERT_FALSE(
"Not implemented for this Geometry's coordinate systems.",
Geometry, CSTag);
};
template <typename Strategy>
struct strategy_converter
{
BOOST_GEOMETRY_STATIC_ASSERT_FALSE(
"Not implemented for this Strategy.",
Strategy);
};
} // namespace services
}} // namespace strategies::line_interpolate
}} // namespace boost::geometry
#endif // BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_SERVICES_HPP

View File

@@ -0,0 +1,94 @@
// Boost.Geometry
// Copyright (c) 2021, Oracle and/or its affiliates.
// 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_LINE_INTERPOLATE_SPHERICAL_HPP
#define BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_SPHERICAL_HPP
#include <boost/geometry/strategies/detail.hpp>
#include <boost/geometry/strategies/line_interpolate/services.hpp>
#include <boost/geometry/strategies/spherical/distance_haversine.hpp>
#include <boost/geometry/strategies/spherical/line_interpolate.hpp>
namespace boost { namespace geometry
{
namespace strategies { namespace line_interpolate
{
template
<
typename RadiusTypeOrSphere = double,
typename CalculationType = void
>
class spherical
: public strategies::detail::spherical_base<RadiusTypeOrSphere>
{
using base_t = strategies::detail::spherical_base<RadiusTypeOrSphere>;
public:
spherical() = default;
template <typename RadiusOrSphere>
explicit spherical(RadiusOrSphere const& radius_or_sphere)
: base_t(radius_or_sphere)
{}
template <typename Geometry1, typename Geometry2>
auto distance(Geometry1 const&, Geometry2 const&,
std::enable_if_t
<
util::is_pointlike<Geometry1>::value
&& util::is_pointlike<Geometry2>::value
> * = nullptr) const
{
return strategy::distance::haversine
<
typename base_t::radius_type, CalculationType
>(base_t::radius());
}
template <typename Geometry>
auto line_interpolate(Geometry const&) const
{
// NOTE: radius is ignored, but pass it just in case
return strategy::line_interpolate::spherical<CalculationType>(base_t::radius());
}
};
namespace services
{
template <typename Geometry>
struct default_strategy<Geometry, spherical_equatorial_tag>
{
using type = strategies::line_interpolate::spherical<>;
};
template <typename CT, typename DS>
struct strategy_converter<strategy::line_interpolate::spherical<CT, DS> >
{
static auto get(strategy::line_interpolate::spherical<CT, DS> const& s)
{
typedef typename strategy::line_interpolate::spherical<CT, DS>::radius_type radius_type;
return strategies::line_interpolate::spherical<radius_type, CT>(s.radius());
}
};
} // namespace services
}} // namespace strategies::line_interpolate
}} // namespace boost::geometry
#endif // BOOST_GEOMETRY_STRATEGIES_LINE_INTERPOLATE_SPHERICAL_HPP

View File

@@ -1,8 +1,9 @@
// Boost.Geometry
// Copyright (c) 2018, Oracle and/or its affiliates.
// Copyright (c) 2018-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
@@ -95,6 +96,12 @@ public:
calc_t a = angle01 * fraction;
formula.compute_point(a, p);
}
inline radius_type radius() const
{
return m_strategy.radius();
}
private :
DistanceStrategy m_strategy;
};