blob: 45efe20b9742a37b52048e657a77102b839b9a4b (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
|
#pragma once
#include "point2d.hpp"
#include "../base/math.hpp"
#include "../std/limits.hpp"
#include "../std/static_assert.hpp"
// Similarly to namespace m2 - 2d math, this is a namespace for nd math.
namespace mn
{
template <typename PointT> class DistanceToLineSquare
{
private:
// we do not support unsigned points!!!
STATIC_ASSERT(numeric_limits<typename PointT::value_type>::is_signed);
public:
void SetBounds(PointT const & p0, PointT const & p1)
{
m_P0 = p0;
m_P1 = p1;
m_D = m_P1 - m_P0;
m_D2 = DotProduct(m_D, m_D);
m_D2 = sqrt(m_D2);
if (my::AlmostEqual(m_D2, 0.0))
{
// make zero vector - then all DotProduct will be equal to zero
m_D = m2::PointD(0, 0);
}
else
{
// normalize vector
m_D = m_D / m_D2;
}
}
double operator() (PointT Y) const
{
m2::PointD const YmP0 = Y - m_P0;
double const t = DotProduct(m_D, YmP0);
if (t <= 0)
{
// Y is closest to P0.
return DotProduct(YmP0, YmP0);
}
if (t >= m_D2)
{
// Y is closest to P1.
PointT const YmP1 = Y - m_P1;
return DotProduct(YmP1, YmP1);
}
// Closest point is interior to segment.
return my::sq(CrossProduct(YmP0, m_D));
}
private:
PointT m_P0, m_P1;
m2::PointD m_D;
double m_D2;
};
}
|