This project provides a header-only library for removing common failures in polygons for boost geometry.
The failure this library can remove are:
- self-intersection
- overlapping inners
- inners outside of the polygon outer
- remove 'spikes' or sub-polygons with very small area
- incorrect orientation of polygons
- invalid points in polygons
- unclosed polygons
The main purpose of this library is to remove self-intersection and spikes from polygons using the boost geometry library. For this, a general approach is used. Meaning, for every polygon it can remove self-intersection, possibly generating multiple output polygons
This library provides an approach for the problem described here:
https://barendgehrels.blogspot.com/2011/02/dissolving-pentagram.html
This is a header-only library so you can import the header file into your project and use the function 'correct' to remove errors from a polygon. For example:
#include "correct.hpp"
namespace bg = boost::geometry;
typedef bg::model::d2::point_xy<double> point;
typedef bg::model::polygon<point> polygon;
int main()
{
polygon poly;
boost::geometry::read_wkt("POLYGON((5 0, 2.5 9, 9.5 3.5, 0.5 3.5, 7.5 9, 5 0))", poly);
double remove_spike_threshold = 1E-12;
multi_polygon result;
geometry::correct(poly, result, remove_spike_threshold);
// Output polygon(s) are valid polygons
if(boost::geometry::is_valid(result))
std::cout << "Output polygon is valid" << std::endl;
else
std::cout << "Output polygon is not valid" << std::endl;
}
First example is a pentagram with self-intersection.
POLYGON ((5 0, 2.5 9, 9.5 3.5, 0.5 3.5, 7.5 9, 5 0))
Self-intersection is removed and a single polygon is generated
Second example is a polygon with a hole inside
POLYGON ((55 10, 141 237, 249 23, 21 171, 252 169, 24 89, 266 73, 55 10))
After removing the self-intersection, two polygons are generated
Finally an example with multiple intersections at same point
POLYGON ((0 0, 10 0, 0 10, 10 10, 0 0, 5 0, 5 10, 0 10, 0 5, 10 5, 10 0, 0 0))
Polygon is converted into multipolygon
Overlapping regions can be handled in different ways, for example the following polygon:
POLYGON ((10 70, 90 70, 90 50, 30 50, 30 30, 50 30, 50 90, 70 90, 70 10, 10 10, 10 70))
Can be generated using non-zero winding rule:
geometry::correct(poly, result, remove_spike_threshold);
Or using odd-even rule:
geometry::correct_odd_even(poly, result, remove_spike_threshold);
Odd-even rule generates more polygons and more holes
Timing of large polygon (1 outer, 298 inners, ~100.000 nodes):
1600 ms @ Intel(R) Pentium(R) Silver N5000 CPU (Mobile CPU)
The approach is an adaptation of the methods described in these papers:
https://www.sciencedirect.com/science/article/abs/pii/S0304397520304199
This approach uses the same approach as the described papers, but with a simpler implementation. Both papers describe how this intersection should be split and that all these intersection points should be visited twice to form the simple subpolygons. Here we just selected a different data structure which makes the implementation even more straightforward than described in these papers. It really is a general approach (for 2d polygons).
The following post describes a set of common polygon/multi-polygon errors which can be corrected using this library:
https://stackoverflow.com/questions/49902090/dataset-of-invalid-geometries-in-boostgeometry
"THE BEER-WARE LICENSE" (Revision 42):
Wouter van Kleunen wrote this file. As long as you retain this notice you can do whatever you want with this stuff. If we meet some day, and you think this stuff is worth it, you can buy me a beer in return
If you find this library useful, let me know you are using this!