-
Notifications
You must be signed in to change notification settings - Fork 83
/
Copy pathmathutils.hpp
115 lines (91 loc) · 3.86 KB
/
mathutils.hpp
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
#ifndef CAVC_MATHUTILS_HPP
#define CAVC_MATHUTILS_HPP
#include "internal/common.hpp"
#include <cmath>
#include <iterator>
namespace cavc {
namespace utils {
// absolute threshold to be used for comparing reals generally
template <typename Real> constexpr Real realThreshold() { return Real(1e-8); }
// absolute threshold to be used for reals in common geometric computation (e.g. to check for
// singularities)
template <typename Real> constexpr Real realPrecision() { return Real(1e-5); }
// absolute threshold to be used for joining slices together at end points
template <typename Real> constexpr Real sliceJoinThreshold() { return Real(1e-4); }
// absolute threshold to be used for pruning invalid slices for offset
template <typename Real> constexpr Real offsetThreshold() { return Real(1e-4); }
template <typename Real> constexpr Real pi() { return Real(3.14159265358979323846264338327950288); }
template <typename Real> constexpr Real tau() { return Real(2) * pi<Real>(); }
template <typename Real> bool fuzzyEqual(Real x, Real y, Real epsilon = realThreshold<Real>()) {
return std::abs(x - y) < epsilon;
}
template <typename Real>
bool fuzzyInRange(Real minValue, Real value, Real maxValue, Real epsilon = realThreshold<Real>()) {
return (value + epsilon > minValue) && (value < maxValue + epsilon);
}
/// Normalize radius to be between 0 and 2PI, e.g. -PI/4 becomes 7PI/8 and 5PI becomes PI.
template <typename Real> Real normalizeRadians(Real angle) {
if (angle >= Real(0) && angle <= tau<Real>()) {
return angle;
}
return angle - std::floor(angle / tau<Real>()) * tau<Real>();
}
/// Returns the smaller difference between two angles, result is negative if angle2 < angle1.
template <typename Real> Real deltaAngle(Real angle1, Real angle2) {
Real diff = normalizeRadians(angle2 - angle1);
if (diff > pi<Real>()) {
diff -= tau<Real>();
}
return diff;
}
/// Tests if angle is between a start and end angle (counter clockwise start to end, inclusive).
template <typename Real>
bool angleIsBetween(Real startAngle, Real endAngle, Real testAngle,
Real epsilon = realThreshold<Real>()) {
Real endSweep = normalizeRadians(endAngle - startAngle);
Real midSweep = normalizeRadians(testAngle - startAngle);
return midSweep < endSweep + epsilon;
}
template <typename Real>
bool angleIsWithinSweep(Real startAngle, Real sweepAngle, Real testAngle,
Real epsilon = realThreshold<Real>()) {
Real endAngle = startAngle + sweepAngle;
if (sweepAngle < Real(0)) {
return angleIsBetween(endAngle, startAngle, testAngle, epsilon);
}
return angleIsBetween(startAngle, endAngle, testAngle, epsilon);
}
/// Returns the solutions to for the quadratic equation -b +/- sqrt (b * b - 4 * a * c) / (2 * a).
template <typename Real>
std::pair<Real, Real> quadraticSolutions(Real a, Real b, Real c, Real discr) {
// Function avoids loss in precision due to taking the difference of two floating point values
// that are very near each other in value.
// See:
// https://math.stackexchange.com/questions/311382/solving-a-quadratic-equation-with-precision-when-using-floating-point-variables
CAVC_ASSERT(fuzzyEqual(b * b - Real(4) * a * c, discr), "discriminate is not correct");
Real sqrtDiscr = std::sqrt(discr);
Real denom = Real(2) * a;
Real sol1;
if (b < Real(0)) {
sol1 = (-b + sqrtDiscr) / denom;
} else {
sol1 = (-b - sqrtDiscr) / denom;
}
Real sol2 = (c / a) / sol1;
return std::make_pair(sol1, sol2);
}
template <typename T> std::size_t nextWrappingIndex(std::size_t index, const T &container) {
if (index == container.size() - 1) {
return 0;
}
return index + 1;
}
template <typename T> std::size_t prevWrappingIndex(std::size_t index, const T &container) {
if (index == 0) {
return container.size() - 1;
}
return index - 1;
}
} // namespace utils
} // namespace cavc
#endif // CAVC_MATHUTILS_HPP