196 lines
6.3 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/AverageAngle/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-12-13 20:51:57 +01:00
[![Arduino-lint](https://github.com/RobTillaart/AverageAngle/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/AverageAngle/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/AverageAngle/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/AverageAngle/actions/workflows/jsoncheck.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/AverageAngle/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/AverageAngle.svg?maxAge=3600)](https://github.com/RobTillaart/AverageAngle/releases)
2021-12-13 20:51:57 +01:00
2020-11-27 11:10:47 +01:00
# AverageAngle
2021-01-29 12:31:58 +01:00
2021-12-13 20:51:57 +01:00
Arduino library to calculate correctly the average of multiple angles.
2020-11-27 11:10:47 +01:00
2021-10-18 21:44:20 +02:00
2020-11-27 11:10:47 +01:00
## Description
2022-10-29 13:59:29 +02:00
AverageAngle is a class to calculate the average of angles.
2020-11-27 11:10:47 +01:00
This is especially useful when angles are around 0 degrees,
e.g. from a compass sensor or the resultant of a track.
Example, the average angle of 359 and 1 is 0, not 179 (most of the time)
2021-12-13 20:51:57 +01:00
Furthermore the AverageAngle can also include the **length (weight)** of the angle as if it is a vector.
Default this length is set to 1 so all angles are by default of equal weight.
2020-11-27 11:10:47 +01:00
2021-12-13 20:51:57 +01:00
Example: The average angle of 359 (length = 2) and 1 (length = 1) is 359.something not zero.
2020-11-27 11:10:47 +01:00
2022-12-07 20:30:13 +01:00
The average angle is calculated by converting the added angle (in DEGREES, etc) to polar coordinates.
These (x,y) are added to a (sumX, sumY) and divided by the number of angles added so far.
2023-02-01 13:53:15 +01:00
#### Related
2022-12-07 20:30:13 +01:00
- https://github.com/RobTillaart/Angle
- https://github.com/RobTillaart/AngleConvertor
2023-02-01 13:53:15 +01:00
- https://github.com/RobTillaart/AverageAngle
2022-12-07 20:30:13 +01:00
- https://github.com/RobTillaart/runningAngle
2021-01-29 12:31:58 +01:00
2021-10-18 21:44:20 +02:00
2022-10-29 13:59:29 +02:00
#### AngleType
2022-12-07 20:30:13 +01:00
- **enum AngleType { DEGREES, RADIANS, GRADIANS }** idem.
2022-10-29 13:59:29 +02:00
2022-12-07 20:30:13 +01:00
| value | name | range |
|:-------:|:-----------|:-----------|
| 0 | DEGREES | 0 .. 360 |
| 1 | RADIANS | 0 .. 2PI |
| 2 | GRADIANS | 0 .. 400 | 100 GRADIANS == 90 DEGREES.
2022-10-29 13:59:29 +02:00
2021-01-29 12:31:58 +01:00
## Interface
2023-02-01 13:53:15 +01:00
```cpp
#include "AverageAngle.h"
```
#### Constructor
- **AverageAngle(AngleType type = DEGREES)** constructor, defaults to degrees.
- **AngleType type()** returns DEGREES, RADIANS or GRADIANS.
- **void setType(AngleType type)** changes type DEGREES, RADIANS or GRADIANS.
Type can be changed run time and still continue to add.
- **void reset()** clears internal counters.
#### Core
2022-12-07 20:30:13 +01:00
- **uint32_t add(float alpha, float length = 1.0)** add a new angle,
optional with length other than 1.
Returns the number of elements (count).
2023-02-01 13:53:15 +01:00
If the internal sumx or sumy is >= 10000, the error **AVERAGE_ANGLE_OVERFLOW** is set.
This indicates that the internal math is near or over its accuracy limits.
- **uint32_t count()** the number of angles added.
2022-12-07 20:30:13 +01:00
If count == 0, there is no average.
2023-02-01 13:53:15 +01:00
- **float getAverage()** returns the average, unless count == 0
2022-12-07 20:30:13 +01:00
or the internal sums == (0,0) in which case we have a singularity ==> NAN (not a number)
2023-02-01 13:53:15 +01:00
If NAN the error **AVERAGE_ANGLE_SINGULARITY** is set.
2022-10-29 13:59:29 +02:00
- **float getTotalLength()** the length of the resulting 'angle' when we see them as vectors.
2022-12-07 20:30:13 +01:00
If count == 0 ==> total length = 0.
2021-10-18 21:44:20 +02:00
- **float getAverageLength()** returns the average length of the angles added.
2022-12-07 20:30:13 +01:00
If count == 0 ==> average length = 0.
2023-02-01 13:53:15 +01:00
#### Error handling
- **int lastError()** return the last error detected.
| name | value |
|:----------------------------|:-------:|
| AVERAGE_ANGLE_OK | 0 |
| AVERAGE_ANGLE_OVERFLOW | -10 |
| AVERAGE_ANGLE_SINGULARITY | -20 |
#### Experimental Overflow
(since 0.2.0)
When the internal sumx or sumy is large (> 10000) the accuracy of the addition
becomes critical, leading to serious errors in the average and length functions.
To detect this the function **add()** sets the error **AVERAGE_ANGLE_OVERFLOW**.
This error can be checked with **lastError()**.
The function **add()** will add the new angle as good as possible.
Note this condition is independent of the **AngleType** as the internal math
uses radians. The condition will be triggered faster when the length parameter
is used.
The overflow threshold of 10000 can be patched in the .cpp file if needed.
As this feature is **experimental**, the trigger condition for overflow will
probably be redesigned in the future. See future section below.
2021-10-18 21:44:20 +02:00
## Gradians
2022-10-29 13:59:29 +02:00
Gradians a.k.a. **gon**, is a less often used unit for angles.
2021-12-13 20:51:57 +01:00
There are 100 gradians in a right angle. A full circle = 400 gradians.
2021-10-18 21:44:20 +02:00
https://en.wikipedia.org/wiki/Gradian
2021-01-29 12:31:58 +01:00
2023-02-01 13:53:15 +01:00
See also AngleConvertor library.
2020-11-27 11:10:47 +01:00
## Operation
If you want to average 5 compass readings you can just add the angles and
do not use the length parameter.
2021-12-13 20:51:57 +01:00
```cpp
2020-11-27 11:10:47 +01:00
AA.reset();
for (int i = 0; i < 5; i++)
{
AA.add(compass.readHeading());
2021-10-18 21:44:20 +02:00
delay(100); // e.g. compass read needs some time
2020-11-27 11:10:47 +01:00
}
Serial.println(AA.getAverage());
```
2021-10-18 21:44:20 +02:00
If you want to average a track, e.g. 5 steps North, 3 steps west etc,
you need to include the length of each step.
2021-12-13 20:51:57 +01:00
```cpp
2020-11-27 11:10:47 +01:00
AA.reset();
2021-10-18 21:44:20 +02:00
AA.add(90, 5); // 5 steps north
AA.add(180, 3); // 3 steps west
2020-11-27 11:10:47 +01:00
Serial.println(AA.getAverage());
Serial.println(AA.getTotalLength());
```
2022-12-07 20:30:13 +01:00
If you want to average an angle in DEGREES and an angle in RADIANS,
just change the type runtime.
```cpp
AA.reset();
AA.setType(DEGREES);
AA.add(90);
AA.setType(RADIANS);
AA.add(PI/3);
AA.setType(DEGREES);
Serial.println(AA.getAverage());
```
2021-10-18 21:44:20 +02:00
## Future
2022-12-07 20:30:13 +01:00
#### Must
2023-02-01 13:53:15 +01:00
- investigate if and how the internal math can be made more robust against overflow.
- use double iso float (will work on certain platforms) (must)
- uint32_t?
- accuracy threshold depends on float/double usage. (sizeof(double)==8)
- threshold depends on the units of length.
if all add's are using 10000 as length they have equal weight.
normalizing the weight? how? user responsibility?
- get set threshold via API?
- use of threshold versus error detection (sum - angle == previous or not)
- split OVERFLOW error in X and Y
2022-12-07 20:30:13 +01:00
#### Should
2023-02-01 13:53:15 +01:00
- add performance example
- add overflow example
- add singularity example
2022-12-07 20:30:13 +01:00
#### Could
2023-02-01 13:53:15 +01:00
2022-12-07 20:30:13 +01:00
- add a USER AngleType, in which the user can map 0..360 degrees to any range.
2022-10-29 13:59:29 +02:00
- float userFactor = 1.0; (default)
- can even be negative?
- use cases? e.g 0..4 quadrant?
2023-02-01 13:53:15 +01:00
- maybe better for the AngleConvertor class.
#### Wont
2020-11-27 11:10:47 +01:00