From 42b4f045c0dd831f383d6bd75e8fbfb9fdcb6290 Mon Sep 17 00:00:00 2001 From: Karl Wright Date: Thu, 21 Apr 2016 08:30:52 -0400 Subject: [PATCH] LUCENE-7226: Add leniency support for filtering points, in order to be able to use OSM data. --- .../spatial3d/geom/GeoPolygonFactory.java | 129 ++++++++++++++++-- .../lucene/spatial3d/geom/GeoPolygonTest.java | 8 +- 2 files changed, 118 insertions(+), 19 deletions(-) diff --git a/lucene/spatial3d/src/java/org/apache/lucene/spatial3d/geom/GeoPolygonFactory.java b/lucene/spatial3d/src/java/org/apache/lucene/spatial3d/geom/GeoPolygonFactory.java index 61903b61484..99fc7c90e4d 100755 --- a/lucene/spatial3d/src/java/org/apache/lucene/spatial3d/geom/GeoPolygonFactory.java +++ b/lucene/spatial3d/src/java/org/apache/lucene/spatial3d/geom/GeoPolygonFactory.java @@ -48,7 +48,7 @@ public class GeoPolygonFactory { final List pointList) { return makeGeoPolygon(planetModel, pointList, null); } - + /** Create a GeoPolygon using the specified points and holes, using order to determine * siding of the polygon. Much like ESRI, this method uses clockwise to indicate the space * on the same side of the shape as being inside, and counter-clockwise to indicate the @@ -63,10 +63,32 @@ public class GeoPolygonFactory { public static GeoPolygon makeGeoPolygon(final PlanetModel planetModel, final List pointList, final List holes) { + return makeGeoPolygon(planetModel, pointList, holes, 0.0); + } + + /** Create a GeoPolygon using the specified points and holes, using order to determine + * siding of the polygon. Much like ESRI, this method uses clockwise to indicate the space + * on the same side of the shape as being inside, and counter-clockwise to indicate the + * space on the opposite side as being inside. + * @param pointList is a list of the GeoPoints to build an arbitrary polygon out of. If points go + * clockwise from a given pole, then that pole should be within the polygon. If points go + * counter-clockwise, then that pole should be outside the polygon. + * @param holes is a list of polygons representing "holes" in the outside polygon. Null == none. + * @param leniencyValue is the maximum distance (in units) that a point can be from the plane and still be considered as + * belonging to the plane. Any value greater than zero may cause some of the provided points that are in fact outside + * the strict definition of co-planarity, but are within this distance, to be discarded for the purposes of creating a + * "safe" polygon. + * @return a GeoPolygon corresponding to what was specified, or null if a valid polygon cannot be generated + * from this input. + */ + public static GeoPolygon makeGeoPolygon(final PlanetModel planetModel, + final List pointList, + final List holes, + final double leniencyValue) { // First, exercise a sanity filter on the provided pointList, and remove identical points, linear points, and backtracks //System.err.println(" filtering "+pointList.size()+" points..."); //final long startTime = System.currentTimeMillis(); - final List filteredPointList = filterPoints(pointList); + final List filteredPointList = filterPoints(pointList, leniencyValue); //System.err.println(" ...done in "+(System.currentTimeMillis()-startTime)+"ms ("+((filteredPointList==null)?"degenerate":(filteredPointList.size()+" points"))+")"); if (filteredPointList == null) { return null; @@ -145,9 +167,10 @@ public class GeoPolygonFactory { /** Filter duplicate points and coplanar points. * @param input with input list of points + * @param leniencyValue is the allowed distance of a point from the plane for cleanup of overly detailed polygons * @return the filtered list, or null if we can't get a legit polygon from the input. */ - static List filterPoints(final List input) { + static List filterPoints(final List input, final double leniencyValue) { final List noIdenticalPoints = new ArrayList<>(input.size()); @@ -200,7 +223,7 @@ public class GeoPolygonFactory { for (int i = 0; i < noIdenticalPoints.size(); i++) { final SafePath startPath = new SafePath(null, noIdenticalPoints.get(i), i, null); // Search, with this as the start path. - final SafePath resultPath = findSafePath(startPath, noIdenticalPoints, getLegalIndex(i+1, noIdenticalPoints.size()), i); + final SafePath resultPath = findSafePath(startPath, noIdenticalPoints, getLegalIndex(i+1, noIdenticalPoints.size()), i, leniencyValue); if (resultPath != null && resultPath.previous != null) { // Read out result, maintaining ordering final List rval = new ArrayList<>(noIdenticalPoints.size()); @@ -220,10 +243,12 @@ public class GeoPolygonFactory { * @param pointIndex is the index of the point that represents the next possible point for consideration for path * extension. * @param startPointIndex is index of the point that starts the current path, so that we can know when we are done. + * @param leniencyValue is the maximum allowed distance of a point being skipped from the revised polygon. Pass zero if + * no leniency desired. * @return null if there was no safe path found, or the safe path if one was discovered. */ private static SafePath findSafePath(final SafePath currentPath, final List points, final int pointIndex, - final int startPointIndex) { + final int startPointIndex, final double leniencyValue) { //System.err.println("extending path..."); // Loop across all possible path extensions, and consider each in turn @@ -250,6 +275,24 @@ public class GeoPolygonFactory { } else if (considerPlane.evaluateIsZero(currentPath.previous.lastPoint)) { //System.err.println(" last point coplanar with this plane"); isChoiceLegal = false; + } else { + // To guarantee that no planes we build are coplanar with edge points, we need to verify everything back from + // considerEndPoint back to the start of the path. We build the edge from considerEndPoint back to each + // of the SafePath points already determined. Then, we need to look at all triangles that include that edge and + // the SafePath points in between. If all of those triangles are legal, we can be assured that adding the current + // proposed point is safe to do. + // This is, of course, a lot of work -- specifically, it's O(n^2) for each point in the path, which leads to an O(n^3) + // evaluation time overall!! + // The only alternative is to understand the cases under which these triangles would be introduced, and tailor the + // cleaning to catch those cases only. Still need to figure that out. The case that blows up is when *all* the points + // for a triangle are coplanar, so theoretically we don't even need to generate the triangle at all(!) + // + // Build a plane that represents the third edge in this triangle, to guarantee that we can compose + // the polygon from triangles + final Plane thirdPlane = new Plane(currentPath.previous.lastPoint, considerEndPoint); + if (thirdPlane.evaluateIsZero(considerStartPoint)) { + isChoiceLegal = false; + } } } } @@ -267,6 +310,13 @@ public class GeoPolygonFactory { } else if (considerPlane.evaluateIsZero(firstPlaneEndpoint.lastPoint)) { //System.err.println(" first point is coplanar with last plane"); isChoiceLegal = false; + } else { + // Build a plane that represents the third edge in this triangle, to guarantee that we can compose + // the polygon from triangles + final Plane thirdPlane = new Plane(considerStartPoint, firstPlaneEndpoint.lastPoint); + if (thirdPlane.evaluateIsZero(considerEndPoint)) { + isChoiceLegal = false; + } } } } @@ -275,7 +325,7 @@ public class GeoPolygonFactory { // All points between the start and end, if any, must be on the plane. int checkIndex = getLegalIndex(currentPath.lastPointIndex + 1, points.size()); while (checkIndex != considerPointIndex) { - if (!considerPlane.evaluateIsZero(points.get(checkIndex))) { + if (Math.abs(considerPlane.evaluate(points.get(checkIndex))) >= Vector.MINIMUM_RESOLUTION + leniencyValue) { // This possibility is no good. But does it say anything about other possibilities? I think // it may mean we don't have to consider any further extensions. I can't prove this, but // it makes this algorithm complete in not an insane period of time at least... @@ -296,7 +346,7 @@ public class GeoPolygonFactory { } //System.err.println(" adding to path: "+considerEndPoint+"; "+considerPlane); final SafePath newPath = new SafePath(currentPath, considerEndPoint, considerPointIndex, considerPlane); - final SafePath result = findSafePath(newPath, points, nextPointIndex, startPointIndex); + final SafePath result = findSafePath(newPath, points, nextPointIndex, startPointIndex, leniencyValue); if (result != null) { return result; } @@ -791,7 +841,9 @@ public class GeoPolygonFactory { // If there are less than three edges, something got messed up somehow. Don't know how this // can happen but check. if (edgeBuffer.size() < 3) { - throw new IllegalStateException("Ending edge buffer had only "+edgeBuffer.size()+" edges"); + // Linear... + // Here we can emit GeoWorld, but probably this means we had a broken poly to start with. + throw new IllegalArgumentException("Illegal polygon; polygon edges intersect each other"); } // Create the list of points @@ -970,11 +1022,12 @@ public class GeoPolygonFactory { } // Ok, figure out what we've accumulated. If it is enough for a polygon, build it. + if (includedEdges.size() < 2) { //System.out.println("Done edge "+currentEdge+": no poly found"); return false; } - + // It's enough to build a convex polygon //System.out.println("Edge "+currentEdge+": Found complex poly"); @@ -990,21 +1043,45 @@ public class GeoPolygonFactory { // Degenerate case!! There is no return edge -- or rather, we already have it. if (includedEdges.size() < 3) { // This means we found a degenerate cycle of edges. If we emit a polygon at this point it - // has no contents, so we've clearly done something wrong, but not sure what. - throw new IllegalArgumentException("polygon was illegal (degenerate illegal two-edge cyclical polygon encountered in processing)"); + // has no contents, so we generate no polygon. + return false; } + + // Now look for completely planar points. This too is a degeneracy condition that we should + // return "false" for. Edge edge = firstEdge; points.add(edge.startPoint); - int i = 0; + int k = 0; while (true) { if (edge == lastEdge) { break; } points.add(edge.endPoint); - internalEdges.set(i++, edge.isInternal); + internalEdges.set(k++, edge.isInternal); edge = edgeBuffer.getNext(edge); } returnIsInternal = lastEdge.isInternal; + + // Look for coplanarity; abort if so + for (int i = 0; i < points.size(); i++) { + final GeoPoint start = points.get(i); + final GeoPoint end = points.get(getLegalIndex(i + 1, points.size())); + // We have to find the next point that is not on the plane between start and end. + // If there is no such point, it's an error. + final Plane planeToFind = new Plane(start, end); + int endPointIndex = -1; + for (int j = 0; j < points.size(); j++) { + final int index = getLegalIndex(j + i + 2, points.size()); + if (!planeToFind.evaluateIsZero(points.get(index))) { + endPointIndex = index; + break; + } + } + if (endPointIndex == -1) { + return false; + } + } + edgeBuffer.clear(); } else { // Build the return edge (internal, of course) @@ -1015,12 +1092,14 @@ public class GeoPolygonFactory { final List edges = new ArrayList(includedEdges.size()); returnIsInternal = true; + // Now look for completely planar points. This too is a degeneracy condition that we should + // return "false" for. Edge edge = firstEdge; points.add(edge.startPoint); - int i = 0; + int k = 0; while (true) { points.add(edge.endPoint); - internalEdges.set(i++, edge.isInternal); + internalEdges.set(k++, edge.isInternal); edges.add(edge); if (edge == lastEdge) { break; @@ -1028,6 +1107,26 @@ public class GeoPolygonFactory { edge = edgeBuffer.getNext(edge); } + // Look for coplanarity; abort if so + for (int i = 0; i < points.size(); i++) { + final GeoPoint start = points.get(i); + final GeoPoint end = points.get(getLegalIndex(i + 1, points.size())); + // We have to find the next point that is not on the plane between start and end. + // If there is no such point, it's an error. + final Plane planeToFind = new Plane(start, end); + int endPointIndex = -1; + for (int j = 0; j < points.size(); j++) { + final int index = getLegalIndex(j + i + 2, points.size()); + if (!planeToFind.evaluateIsZero(points.get(index))) { + endPointIndex = index; + break; + } + } + if (endPointIndex == -1) { + return false; + } + } + // Modify the edge buffer edgeBuffer.replace(edges, returnEdge); } diff --git a/lucene/spatial3d/src/test/org/apache/lucene/spatial3d/geom/GeoPolygonTest.java b/lucene/spatial3d/src/test/org/apache/lucene/spatial3d/geom/GeoPolygonTest.java index 2f7151576c3..ce43c5b172f 100755 --- a/lucene/spatial3d/src/test/org/apache/lucene/spatial3d/geom/GeoPolygonTest.java +++ b/lucene/spatial3d/src/test/org/apache/lucene/spatial3d/geom/GeoPolygonTest.java @@ -44,7 +44,7 @@ public class GeoPolygonTest { originalPoints.add(point2); originalPoints.add(point2); originalPoints.add(point3); - final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints); + final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints, 0.0); assertEquals(3, filteredPoints.size()); assertEquals(point1, filteredPoints.get(0)); assertEquals(point2, filteredPoints.get(1)); @@ -57,7 +57,7 @@ public class GeoPolygonTest { originalPoints.add(point1); originalPoints.add(point3); originalPoints.add(point2); - final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints); + final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints, 0.0); assertEquals(3, filteredPoints.size()); assertEquals(point2, filteredPoints.get(0)); assertEquals(point1, filteredPoints.get(1)); @@ -71,7 +71,7 @@ public class GeoPolygonTest { originalPoints.add(point3); originalPoints.add(point4); originalPoints.add(point5); - final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints); + final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints, 0.0); assertEquals(3, filteredPoints.size()); assertEquals(point1, filteredPoints.get(0)); assertEquals(point3, filteredPoints.get(1)); @@ -85,7 +85,7 @@ public class GeoPolygonTest { originalPoints.add(point3); originalPoints.add(point4); System.err.println("Before: "+originalPoints); - final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints); + final List filteredPoints =GeoPolygonFactory.filterPoints(originalPoints, 0.0); System.err.println("After: "+filteredPoints); assertEquals(3, filteredPoints.size()); assertEquals(point5, filteredPoints.get(0));