GeographicLib 2.1.2
|
Polygon areas. More...
#include <GeographicLib/PolygonArea.hpp>
Public Member Functions | |
PolygonAreaT (const GeodType &earth, bool polyline=false) | |
void | Clear () |
void | AddPoint (real lat, real lon) |
void | AddEdge (real azi, real s) |
unsigned | Compute (bool reverse, bool sign, real &perimeter, real &area) const |
unsigned | TestPoint (real lat, real lon, bool reverse, bool sign, real &perimeter, real &area) const |
unsigned | TestEdge (real azi, real s, bool reverse, bool sign, real &perimeter, real &area) const |
Inspector functions | |
Math::real | EquatorialRadius () const |
Math::real | Flattening () const |
void | CurrentPoint (real &lat, real &lon) const |
unsigned | NumberPoints () const |
bool | Polyline () const |
Related Functions | |
(Note that these are not member functions.) | |
typedef PolygonAreaT< Geodesic > | PolygonArea |
typedef PolygonAreaT< GeodesicExact > | PolygonAreaExact |
typedef PolygonAreaT< Rhumb > | PolygonAreaRhumb |
Polygon areas.
This computes the area of a polygon whose edges are geodesics using the method given in Section 6 of
Arbitrarily complex polygons are allowed. In the case self-intersecting of polygons the area is accumulated "algebraically", e.g., the areas of the 2 loops in a figure-8 polygon will partially cancel.
This class lets you add vertices and edges one at a time to the polygon. The sequence must start with a vertex and thereafter vertices and edges can be added in any order. Any vertex after the first creates a new edge which is the shortest geodesic from the previous vertex. In some cases there may be two or many such shortest geodesics and the area is then not uniquely defined. In this case, either add an intermediate vertex or add the edge as an edge (by defining its direction and length).
The area and perimeter are accumulated at two times the standard floating point precision to guard against the loss of accuracy with many-sided polygons. At any point you can ask for the perimeter and area so far. There's an option to treat the points as defining a polyline instead of a polygon; in that case, only the perimeter is computed.
This is a templated class to allow it to be used with Geodesic, GeodesicExact, and Rhumb. GeographicLib::PolygonArea, GeographicLib::PolygonAreaExact, and GeographicLib::PolygonAreaRhumb are typedefs for these cases.
For GeographicLib::PolygonArea (edges defined by Geodesic), an upper bound on the error is about 0.1 m2 per vertex. However this is a wildly pessimistic estimate in most cases. A more realistic estimate of the error is given by a test involving 107 approximately regular polygons on the WGS84 ellipsoid. The centers and the orientations of the polygons were uniformly distributed, the number of vertices was log-uniformly distributed in [3, 300], and the center to vertex distance log-uniformly distributed in [0.1 m, 9000 km].
Using double precision (the standard precision for GeographicLib), the maximum error in the perimeter was 200 nm, and the maximum error in the area was
0.0013 m^2 for perimeter < 10 km 0.0070 m^2 for perimeter < 100 km 0.070 m^2 for perimeter < 1000 km 0.11 m^2 for all perimeters
The errors are given in terms of the perimeter, because it is expected that the errors depend mainly on the number of edges and the edge lengths.
Using long doubles (GEOGRPAHICLIB_PRECISION = 3), the maximum error in the perimeter was 200 pm, and the maximum error in the area was
0.7 mm^2 for perim < 10 km 3.2 mm^2 for perimeter < 100 km 21 mm^2 for perimeter < 1000 km 45 mm^2 for all perimeters
GeodType | the geodesic class to use. |
Example of use:
Planimeter is a command-line utility providing access to the functionality of PolygonAreaT.
Definition at line 97 of file PolygonArea.hpp.
|
inline |
Constructor for PolygonAreaT.
[in] | earth | the Geodesic object to use for geodesic calculations. |
[in] | polyline | if true that treat the points as defining a polyline instead of a polygon (default = false). |
Definition at line 128 of file PolygonArea.hpp.
References GeographicLib::PolygonAreaT< GeodType >::Clear().
|
inline |
Clear PolygonAreaT, allowing a new polygon to be started.
Definition at line 140 of file PolygonArea.hpp.
References GeographicLib::Math::NaN().
Referenced by GeographicLib::PolygonAreaT< GeodType >::PolygonAreaT().
void GeographicLib::PolygonAreaT< GeodType >::AddPoint | ( | real | lat, |
real | lon | ||
) |
Add a point to the polygon or polyline.
[in] | lat | the latitude of the point (degrees). |
[in] | lon | the longitude of the point (degrees). |
lat should be in the range [−90°, 90°].
Definition at line 70 of file PolygonArea.cpp.
void GeographicLib::PolygonAreaT< GeodType >::AddEdge | ( | real | azi, |
real | s | ||
) |
Add an edge to the polygon or polyline.
[in] | azi | azimuth at current point (degrees). |
[in] | s | distance from current point to next point (meters). |
This does nothing if no points have been added yet. Use PolygonAreaT::CurrentPoint to determine the position of the new vertex.
Definition at line 89 of file PolygonArea.cpp.
unsigned GeographicLib::PolygonAreaT< GeodType >::Compute | ( | bool | reverse, |
bool | sign, | ||
real & | perimeter, | ||
real & | area | ||
) | const |
Return the results so far.
[in] | reverse | if true then clockwise (instead of counter-clockwise) traversal counts as a positive area. |
[in] | sign | if true then return a signed result for the area if the polygon is traversed in the "wrong" direction instead of returning the area for the rest of the earth. |
[out] | perimeter | the perimeter of the polygon or length of the polyline (meters). |
[out] | area | the area of the polygon (meters2); only set if polyline is false in the constructor. |
More points can be added to the polygon after this call.
Definition at line 105 of file PolygonArea.cpp.
unsigned GeographicLib::PolygonAreaT< GeodType >::TestPoint | ( | real | lat, |
real | lon, | ||
bool | reverse, | ||
bool | sign, | ||
real & | perimeter, | ||
real & | area | ||
) | const |
Return the results assuming a tentative final test point is added; however, the data for the test point is not saved. This lets you report a running result for the perimeter and area as the user moves the mouse cursor. Ordinary floating point arithmetic is used to accumulate the data for the test point; thus the area and perimeter returned are less accurate than if PolygonAreaT::AddPoint and PolygonAreaT::Compute are used.
[in] | lat | the latitude of the test point (degrees). |
[in] | lon | the longitude of the test point (degrees). |
[in] | reverse | if true then clockwise (instead of counter-clockwise) traversal counts as a positive area. |
[in] | sign | if true then return a signed result for the area if the polygon is traversed in the "wrong" direction instead of returning the area for the rest of the earth. |
[out] | perimeter | the approximate perimeter of the polygon or length of the polyline (meters). |
[out] | area | the approximate area of the polygon (meters2); only set if polyline is false in the constructor. |
lat should be in the range [−90°, 90°].
Definition at line 131 of file PolygonArea.cpp.
unsigned GeographicLib::PolygonAreaT< GeodType >::TestEdge | ( | real | azi, |
real | s, | ||
bool | reverse, | ||
bool | sign, | ||
real & | perimeter, | ||
real & | area | ||
) | const |
Return the results assuming a tentative final test point is added via an azimuth and distance; however, the data for the test point is not saved. This lets you report a running result for the perimeter and area as the user moves the mouse cursor. Ordinary floating point arithmetic is used to accumulate the data for the test point; thus the area and perimeter returned are less accurate than if PolygonAreaT::AddEdge and PolygonAreaT::Compute are used.
[in] | azi | azimuth at current point (degrees). |
[in] | s | distance from current point to final test point (meters). |
[in] | reverse | if true then clockwise (instead of counter-clockwise) traversal counts as a positive area. |
[in] | sign | if true then return a signed result for the area if the polygon is traversed in the "wrong" direction instead of returning the area for the rest of the earth. |
[out] | perimeter | the approximate perimeter of the polygon or length of the polyline (meters). |
[out] | area | the approximate area of the polygon (meters2); only set if polyline is false in the constructor. |
Definition at line 167 of file PolygonArea.cpp.
References GeographicLib::Math::NaN().
|
inline |
Definition at line 250 of file PolygonArea.hpp.
|
inline |
Definition at line 256 of file PolygonArea.hpp.
|
inline |
Report the previous vertex added to the polygon or polyline.
[out] | lat | the latitude of the point (degrees). |
[out] | lon | the longitude of the point (degrees). |
If no points have been added, then NaNs are returned. Otherwise, lon will be in the range [−180°, 180°].
Definition at line 267 of file PolygonArea.hpp.
|
inline |
Report the number of points currently in the polygon or polyline.
If no points have been added, then 0 is returned.
Definition at line 277 of file PolygonArea.hpp.
|
inline |
Report whether the current object is a polygon or a polyline.
Definition at line 284 of file PolygonArea.hpp.
|
related |
Polygon areas using Geodesic. This should be used if the flattening is small.
Definition at line 294 of file PolygonArea.hpp.
|
related |
Polygon areas using GeodesicExact. (But note that the implementation of areas in GeodesicExact uses a high order series and this is only accurate for modest flattenings.)
Definition at line 303 of file PolygonArea.hpp.
|
related |
Polygon areas using Rhumb.
Definition at line 310 of file PolygonArea.hpp.