// Boost.Geometry
|
|
// Copyright (c) 2018 Yaghyavardhan Singh Khangarot, Hyderabad, India.
|
// Contributed and/or modified by Yaghyavardhan Singh Khangarot,
|
// as part of Google Summer of Code 2018 program.
|
|
// This file was modified by Oracle on 2018.
|
// Modifications copyright (c) 2018 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_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP
|
#define BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP
|
|
#include <algorithm>
|
|
#ifdef BOOST_GEOMETRY_DEBUG_FRECHET_DISTANCE
|
#include <iostream>
|
#endif
|
|
#include <iterator>
|
#include <utility>
|
#include <vector>
|
#include <limits>
|
|
#include <boost/geometry/algorithms/detail/throw_on_empty_input.hpp>
|
#include <boost/geometry/algorithms/not_implemented.hpp>
|
#include <boost/geometry/core/assert.hpp>
|
#include <boost/geometry/core/tag.hpp>
|
#include <boost/geometry/core/tags.hpp>
|
#include <boost/geometry/core/point_type.hpp>
|
#include <boost/geometry/strategies/distance.hpp>
|
#include <boost/geometry/strategies/distance_result.hpp>
|
#include <boost/geometry/util/range.hpp>
|
|
|
namespace boost { namespace geometry
|
{
|
|
#ifndef DOXYGEN_NO_DETAIL
|
namespace detail { namespace discrete_frechet_distance
|
{
|
|
template <typename size_type1 , typename size_type2,typename result_type>
|
class coup_mat
|
{
|
public:
|
coup_mat(size_type1 w, size_type2 h)
|
: m_data(w * h,-1), m_width(w), m_height(h)
|
{}
|
|
result_type & operator()(size_type1 i, size_type2 j)
|
{
|
BOOST_GEOMETRY_ASSERT(i < m_width && j < m_height);
|
return m_data[j * m_width + i];
|
}
|
|
private:
|
std::vector<result_type> m_data;
|
size_type1 m_width;
|
size_type2 m_height;
|
};
|
|
struct linestring_linestring
|
{
|
template <typename Linestring1, typename Linestring2, typename Strategy>
|
static inline typename distance_result
|
<
|
typename point_type<Linestring1>::type,
|
typename point_type<Linestring2>::type,
|
Strategy
|
>::type apply(Linestring1 const& ls1, Linestring2 const& ls2, Strategy const& strategy)
|
{
|
typedef typename distance_result
|
<
|
typename point_type<Linestring1>::type,
|
typename point_type<Linestring2>::type,
|
Strategy
|
>::type result_type;
|
typedef typename boost::range_size<Linestring1>::type size_type1;
|
typedef typename boost::range_size<Linestring2>::type size_type2;
|
|
|
boost::geometry::detail::throw_on_empty_input(ls1);
|
boost::geometry::detail::throw_on_empty_input(ls2);
|
|
size_type1 const a = boost::size(ls1);
|
size_type2 const b = boost::size(ls2);
|
|
|
//Coupling Matrix CoupMat(a,b,-1);
|
coup_mat<size_type1,size_type2,result_type> coup_matrix(a,b);
|
|
result_type const not_feasible = -100;
|
//findin the Coupling Measure
|
for (size_type1 i = 0 ; i < a ; i++ )
|
{
|
for(size_type2 j=0;j<b;j++)
|
{
|
result_type dis = strategy.apply(range::at(ls1,i), range::at(ls2,j));
|
if(i==0 && j==0)
|
coup_matrix(i,j) = dis;
|
else if(i==0 && j>0)
|
coup_matrix(i,j) =
|
(std::max)(coup_matrix(i,j-1), dis);
|
else if(i>0 && j==0)
|
coup_matrix(i,j) =
|
(std::max)(coup_matrix(i-1,j), dis);
|
else if(i>0 && j>0)
|
coup_matrix(i,j) =
|
(std::max)((std::min)(coup_matrix(i,j-1),
|
(std::min)(coup_matrix(i-1,j),
|
coup_matrix(i-1,j-1))),
|
dis);
|
else
|
coup_matrix(i,j) = not_feasible;
|
}
|
}
|
|
#ifdef BOOST_GEOMETRY_DEBUG_FRECHET_DISTANCE
|
//Print CoupLing Matrix
|
for(size_type i = 0; i <a; i++)
|
{
|
for(size_type j = 0; j <b; j++)
|
std::cout << coup_matrix(i,j) << " ";
|
std::cout << std::endl;
|
}
|
#endif
|
|
return coup_matrix(a-1,b-1);
|
}
|
};
|
|
}} // namespace detail::frechet_distance
|
#endif // DOXYGEN_NO_DETAIL
|
|
#ifndef DOXYGEN_NO_DISPATCH
|
namespace dispatch
|
{
|
template
|
<
|
typename Geometry1,
|
typename Geometry2,
|
typename Tag1 = typename tag<Geometry1>::type,
|
typename Tag2 = typename tag<Geometry2>::type
|
>
|
struct discrete_frechet_distance : not_implemented<Tag1, Tag2>
|
{};
|
|
template <typename Linestring1, typename Linestring2>
|
struct discrete_frechet_distance
|
<
|
Linestring1,
|
Linestring2,
|
linestring_tag,
|
linestring_tag
|
>
|
: detail::discrete_frechet_distance::linestring_linestring
|
{};
|
|
} // namespace dispatch
|
#endif // DOXYGEN_NO_DISPATCH
|
|
|
/*!
|
\brief Calculate discrete Frechet distance between two geometries (currently
|
works for LineString-LineString) using specified strategy.
|
\ingroup discrete_frechet_distance
|
\tparam Geometry1 \tparam_geometry
|
\tparam Geometry2 \tparam_geometry
|
\tparam Strategy A type fulfilling a DistanceStrategy concept
|
\param geometry1 Input geometry
|
\param geometry2 Input geometry
|
\param strategy Distance strategy to be used to calculate Pt-Pt distance
|
|
\qbk{distinguish,with strategy}
|
\qbk{[include reference/algorithms/discrete_frechet_distance.qbk]}
|
|
\qbk{
|
[heading Available Strategies]
|
\* [link geometry.reference.strategies.strategy_distance_pythagoras Pythagoras (cartesian)]
|
\* [link geometry.reference.strategies.strategy_distance_haversine Haversine (spherical)]
|
[/ \* more (currently extensions): Vincenty\, Andoyer (geographic) ]
|
|
[heading Example]
|
[discrete_frechet_distance_strategy]
|
[discrete_frechet_distance_strategy_output]
|
}
|
*/
|
template <typename Geometry1, typename Geometry2, typename Strategy>
|
inline typename distance_result
|
<
|
typename point_type<Geometry1>::type,
|
typename point_type<Geometry2>::type,
|
Strategy
|
>::type
|
discrete_frechet_distance(Geometry1 const& geometry1,
|
Geometry2 const& geometry2,
|
Strategy const& strategy)
|
{
|
return dispatch::discrete_frechet_distance
|
<
|
Geometry1, Geometry2
|
>::apply(geometry1, geometry2, strategy);
|
}
|
|
// Algorithm overload using default Pt-Pt distance strategy
|
|
/*!
|
\brief Calculate discrete Frechet distance between two geometries (currently
|
work for LineString-LineString).
|
\ingroup discrete_frechet_distance
|
\tparam Geometry1 \tparam_geometry
|
\tparam Geometry2 \tparam_geometry
|
\param geometry1 Input geometry
|
\param geometry2 Input geometry
|
|
\qbk{[include reference/algorithms/discrete_frechet_distance.qbk]}
|
|
\qbk{
|
[heading Example]
|
[discrete_frechet_distance]
|
[discrete_frechet_distance_output]
|
}
|
*/
|
template <typename Geometry1, typename Geometry2>
|
inline typename distance_result
|
<
|
typename point_type<Geometry1>::type,
|
typename point_type<Geometry2>::type
|
>::type
|
discrete_frechet_distance(Geometry1 const& geometry1, Geometry2 const& geometry2)
|
{
|
typedef typename strategy::distance::services::default_strategy
|
<
|
point_tag, point_tag,
|
typename point_type<Geometry1>::type,
|
typename point_type<Geometry2>::type
|
>::type strategy_type;
|
|
return discrete_frechet_distance(geometry1, geometry2, strategy_type());
|
}
|
|
}} // namespace boost::geometry
|
|
#endif // BOOST_GEOMETRY_ALGORITHMS_DISCRETE_FRECHET_DISTANCE_HPP
|