356 lines
7.0 KiB
C++
Raw Normal View History

2015-08-01 20:36:28 +02:00
//
// FILE: Angle.cpp
// AUTHOR: Rob Tillaart
2022-10-29 12:45:36 +02:00
// VERSION: 0.1.13
2015-08-01 20:36:28 +02:00
// PURPOSE: library for Angle math for Arduino
2020-11-27 11:10:47 +01:00
// URL: https://github.com/RobTillaart/Angle
// http://forum.arduino.cc/index.php?topic=339402
2015-08-01 20:36:28 +02:00
//
2022-10-29 12:45:36 +02:00
// HISTORY: see changelog.md
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
#include "Angle.h"
2021-01-29 12:31:58 +01:00
AngleFormat::AngleFormat( const Angle &ref, AngleFormatMode format )
2021-12-13 13:48:16 +01:00
: angle(ref), mode(format)
{
}
size_t AngleFormat::printTo(Print& p) const
{
return angle.printTo( p, mode );
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
Angle::Angle(int dd, int mm, int ss, int tt)
{
neg = false;
d = dd;
m = mm;
s = ss;
t = tt;
2022-10-29 12:45:36 +02:00
// TODO
// normalize();
// assume only one (largest) parameter is negative at most...
if (d < 0) { d = -d; neg = true; }
if (m < 0) { m = -m; neg = true; }
if (s < 0) { s = -s; neg = true; }
if (t < 0) { t = -t; neg = true; }
2022-10-29 12:45:36 +02:00
// modulo might be faster
while (t >= 10000) { s++; t -= 10000; }
while (s >= 60) { m++; s -= 60; }
while (m >= 60) { d++; m -= 60; }
2022-10-29 12:45:36 +02:00
// check special case 0
if (d == 0 && m == 0 && s == 0 && t == 0) neg = false;
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
Angle::Angle(const double alpha)
{
double a = alpha;
neg = (alpha < 0);
2015-08-01 20:36:28 +02:00
if (neg) a = -a;
2015-08-01 20:44:45 +02:00
2015-08-01 20:36:28 +02:00
d = int(a);
a = a - d;
2022-10-29 12:45:36 +02:00
// unsigned long p = a * 3600000L;
// 3600 000 = 2^7 • 3^2 • 5^5 = 128 * 28125
// 2^7 = 128 will only affect exponent - no loss precision
// 28125 is less digits so less loss of significant digits.
// upgraded to 4 decimal seconds
// 36 000 000L = 256 * 140625
a = a * 256;
unsigned long p = round(a * 140625.0);
t = p % 10000UL;
p = p / 10000UL;
2015-08-01 20:36:28 +02:00
s = p % 60UL;
m = p / 60UL;
2022-10-29 12:45:36 +02:00
// check special case 0
if (d == 0 && m == 0 && s == 0 && t == 0) neg = false;
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
2021-12-13 13:48:16 +01:00
Angle::Angle(const char * str)
{
uint32_t yy = 0;
uint8_t d_cnt = 0;
neg = false;
2022-10-29 12:45:36 +02:00
// parse whole degrees
2021-12-13 13:48:16 +01:00
char *p = (char *) str;
d = 0;
2022-10-29 12:45:36 +02:00
// skip crap
while (!isdigit(*p) && (*p != '-')) p++;
2022-10-29 12:45:36 +02:00
// process sign
if (*p == '-')
{
neg = true;
p++;
}
if (*p == '+') p++;
2022-10-29 12:45:36 +02:00
// parse whole part into degrees;
while (isdigit(*p))
{
d *= 10;
d += (*p - '0');
p++;
}
2022-10-29 12:45:36 +02:00
// parse decimal part into an uint32_t
if (*p != '\0')
{
p++; // skip .
while (isdigit(*p) && d_cnt < 9)
{
d_cnt++;
yy *= 10;
yy += (*p - '0');
p++;
}
}
2022-10-29 12:45:36 +02:00
// make sure we have 9 decimal places.
while (d_cnt < 9)
{
d_cnt++;
yy *= 10;
}
2022-10-29 12:45:36 +02:00
// convert float to degrees. 1000000000 ~> 36000000 -> /250 * 9
// yy = yy * 4 / 125 + yy / 250; // just keeps the maths within 32 bits
yy = yy * 4 / 125;
2022-10-29 12:45:36 +02:00
yy = yy + (yy + 4)/ 8; // just keeps the maths within 32 bits
// split yy in m, s, tt
t = yy % 10000UL;
yy = yy / 10000UL;
s = yy % 60;
m = yy / 60;
2022-10-29 12:45:36 +02:00
// check special case 0
if (d == 0 && m == 0 && s == 0 && t == 0) neg = false;
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
// PRINTING
size_t Angle::printTo(Print& p, AngleFormatMode mode) const
2015-08-01 20:36:28 +02:00
{
unsigned char c = mode;
2022-10-29 12:45:36 +02:00
char separator[4] = ".\'\""; // "..."; // ALT-0176 = ° 179.59.59.9999
2015-08-01 20:36:28 +02:00
size_t n = 0;
if (neg) n += p.print('-');
2015-08-01 20:36:28 +02:00
n += p.print(d);
n += p.print(separator[0]);
if( --c )
{
if (m < 10) n += p.print('0');
n += p.print(m);
n += p.print(separator[1]);
if( --c )
{
if (s < 10) n += p.print('0');
n += p.print(s);
n += p.print(separator[2]);
if( --c )
{
if (t < 1000) n += p.print('0');
if (t < 100) n += p.print('0');
if (t < 10) n += p.print('0');
n += p.print(t);
}
}
}
2015-08-01 20:36:28 +02:00
return n;
};
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
double Angle::toDouble(void)
{
long v = t + s * 10000UL + m * 600000UL;
double val = ((1.0 / 140625.0) / 256) * v + d;
if (neg) val = -val;
2015-08-01 20:36:28 +02:00
return val;
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
// NEGATE
Angle Angle::operator - ()
{
Angle temp = *this;
if (temp.d == 0 && temp.m == 0 && temp.s == 0 && temp.t == 0)
{
temp.neg = false;
}
else
{
temp.neg = !neg;
}
return temp;
};
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
// BASIC MATH
Angle Angle::operator + (const Angle &a) // TOCHECK
2015-08-01 20:36:28 +02:00
{
2015-08-01 20:44:45 +02:00
return addHelper(a);
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
Angle& Angle::operator += (const Angle &a) // TOCHECK
2015-08-01 20:36:28 +02:00
{
2015-08-01 20:44:45 +02:00
*this = addHelper(a);
return *this;
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
Angle Angle::addHelper(const Angle &a) // TOCHECK
2015-08-01 20:36:28 +02:00
{
Angle temp = *this;
if (temp.neg == a.neg)
2015-08-01 20:36:28 +02:00
{
temp.d += a.d;
temp.m += a.m;
temp.s += a.s;
temp.t += a.t;
}
2015-08-01 20:44:45 +02:00
else
2015-08-01 20:36:28 +02:00
{
temp.d -= a.d;
2015-08-01 20:36:28 +02:00
temp.m -= a.m;
temp.s -= a.s;
temp.t -= a.t;
}
temp.normalize();
2015-08-01 20:44:45 +02:00
return temp;
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
Angle Angle::operator - (const Angle &a) // TOCHECK
2015-08-01 20:36:28 +02:00
{
return subHelper(a);
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
Angle& Angle::operator -= (const Angle &a) // TOCHECK
{
*this = subHelper(a);
return *this;
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
Angle Angle::subHelper(const Angle &a) // TOCHECK
2015-08-01 20:36:28 +02:00
{
Angle temp = *this;
if (temp.neg == a.neg)
2015-08-01 20:36:28 +02:00
{
temp.d -= a.d;
temp.m -= a.m;
temp.s -= a.s;
temp.t -= a.t;
}
2015-08-01 20:44:45 +02:00
else
2015-08-01 20:36:28 +02:00
{
temp.d += a.d;
2015-08-01 20:36:28 +02:00
temp.m += a.m;
temp.s += a.s;
temp.t += a.t;
}
temp.normalize();
return temp;
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
Angle Angle::operator * (const double dd)
{
return Angle(this->toDouble() * dd);
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
Angle Angle::operator / (const double dd)
{
return Angle(this->toDouble() / dd);
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
Angle& Angle::operator *= (const double dd)
{
*this = this->toDouble() * dd;
2015-08-01 20:36:28 +02:00
return *this;
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
Angle& Angle::operator /= (const double dd)
{
*this = this->toDouble() / dd;
2015-08-01 20:36:28 +02:00
return *this;
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
// !! can divide by zero
2015-08-01 20:36:28 +02:00
double Angle::operator / (Angle& a)
{
double f = this->toDouble();
double g = a.toDouble();
return f/g;
}
2021-01-29 12:31:58 +01:00
2015-08-01 20:36:28 +02:00
///////////////////////////////////////////////////////////
//
2022-10-29 12:45:36 +02:00
// PRIVATE
2015-08-01 20:36:28 +02:00
//
int Angle::compare(const Angle &a, const Angle &b)
{
2022-10-29 12:45:36 +02:00
// check sign first
if (!a.neg && b.neg) return 1;
if (a.neg && !b.neg) return -1;
2022-10-29 12:45:36 +02:00
// check abs value
int rv = 0;
if (a.d > b.d) rv = 1;
else if (a.d < b.d) rv = -1;
else if (a.m > b.m) rv = 1;
else if (a.m < b.m) rv = -1;
else if (a.s > b.s) rv = 1;
else if (a.s < b.s) rv = -1;
else if (a.t > b.t) rv = 1;
else if (a.t < b.t) rv = -1;
if (rv != 0 && a.neg) rv = -rv;
return rv;
2015-08-01 20:36:28 +02:00
}
2021-01-29 12:31:58 +01:00
2022-10-29 12:45:36 +02:00
void Angle::normalize() // TODO CHECK
2015-08-01 20:36:28 +02:00
{
while (t < 0) { s--; t += 10000; }
while (t >= 10000) { s++; t -= 10000; }
while (s < 0) { m--; s += 60; }
while (s >= 60) { m++; s -= 60; }
while (m < 0) { d--; m += 60; }
while (m >= 60) { d++; m -= 60; }
2015-08-01 20:36:28 +02:00
if (d < 0)
{
if (t != 0)
{
t = 10000 - t;
s++;
}
if (s != 0)
{
s = (60 - s) % 60;
m++;
}
if (m != 0)
{
m = (60 - m) % 60;
d++;
}
d = -d;
neg = !neg;
}
if (d == 0 && m == 0 && s == 0 && t == 0) neg = false;
2015-08-01 20:36:28 +02:00
}
2021-12-13 13:48:16 +01:00
2020-11-27 11:10:47 +01:00
// --- END OF FILE ---