LUCENE-7241: Fix large polygon test point logic to deal properly with holes.

This commit is contained in:
Karl Wright 2016-05-03 06:45:53 -04:00
parent 21b87bc7c4
commit 9e09c5cbb0
1 changed files with 27 additions and 9 deletions

View File

@ -171,12 +171,12 @@ public class GeoPolygonFactory {
final List<List<GeoPoint>> pointsList = new ArrayList<>(); final List<List<GeoPoint>> pointsList = new ArrayList<>();
List<GeoPoint> testPointShape = null; BestShape testPointShape = null;
for (final PolygonDescription shape : shapesList) { for (final PolygonDescription shape : shapesList) {
// Convert this shape and its holes to a general list of shapes. We also need to identify exactly one // Convert this shape and its holes to a general list of shapes. We also need to identify exactly one
// legal, non-degenerate shape with no children that we can use to find a test point. We also optimize // legal, non-degenerate shape with no children that we can use to find a test point. We also optimize
// to choose as small as possible a polygon for determining the in-set-ness of the test point. // to choose as small as possible a polygon for determining the in-set-ness of the test point.
testPointShape = convertPolygon(pointsList, shape, testPointShape); testPointShape = convertPolygon(pointsList, shape, testPointShape, true);
} }
// If there's no polygon we can use to determine a test point, we throw up. // If there's no polygon we can use to determine a test point, we throw up.
@ -189,12 +189,16 @@ public class GeoPolygonFactory {
final Random generator = new Random(1234); final Random generator = new Random(1234);
for (int counter = 0; counter < 1000000; counter++) { for (int counter = 0; counter < 1000000; counter++) {
// Pick the next random pole // Pick the next random pole
final GeoPoint pole = pickPole(generator, planetModel, testPointShape); final GeoPoint pole = pickPole(generator, planetModel, testPointShape.points);
// Is it inside or outside? // Is it inside or outside?
final Boolean isPoleInside = isInsidePolygon(pole, testPointShape); final Boolean isPoleInside = isInsidePolygon(pole, testPointShape.points);
if (isPoleInside != null) { if (isPoleInside != null) {
// Legal pole // Legal pole
return new GeoComplexPolygon(planetModel, pointsList, pole, isPoleInside); if (isPoleInside == testPointShape.poleMustBeInside) {
return new GeoComplexPolygon(planetModel, pointsList, pole, isPoleInside);
} else {
return new GeoComplexPolygon(planetModel, pointsList, new GeoPoint(-pole.x, -pole.y, -pole.z), !isPoleInside);
}
} }
// If pole choice was illegal, try another one // If pole choice was illegal, try another one
} }
@ -208,7 +212,7 @@ public class GeoPolygonFactory {
* @param testPointShape is the current best choice for a low-level polygon to evaluate. * @param testPointShape is the current best choice for a low-level polygon to evaluate.
* @return an updated best-choice for a test point polygon, and update the points list. * @return an updated best-choice for a test point polygon, and update the points list.
*/ */
private static List<GeoPoint> convertPolygon(final List<List<GeoPoint>> pointsList, final PolygonDescription shape, List<GeoPoint> testPointShape) { private static BestShape convertPolygon(final List<List<GeoPoint>> pointsList, final PolygonDescription shape, BestShape testPointShape, final boolean mustBeInside) {
// First, remove duplicate points. If degenerate, just ignore the shape. // First, remove duplicate points. If degenerate, just ignore the shape.
final List<GeoPoint> filteredPoints = filterPoints(shape.points); final List<GeoPoint> filteredPoints = filterPoints(shape.points);
if (filteredPoints == null) { if (filteredPoints == null) {
@ -218,8 +222,8 @@ public class GeoPolygonFactory {
// Non-degenerate. Check if this is a candidate for in-set determination. // Non-degenerate. Check if this is a candidate for in-set determination.
if (shape.holes.size() == 0) { if (shape.holes.size() == 0) {
// This shape is a candidate for a test point. // This shape is a candidate for a test point.
if (testPointShape == null || testPointShape.size() > filteredPoints.size()) { if (testPointShape == null || testPointShape.points.size() > filteredPoints.size()) {
testPointShape = filteredPoints; testPointShape = new BestShape(filteredPoints, mustBeInside);
} }
} }
@ -227,13 +231,27 @@ public class GeoPolygonFactory {
// Now, do all holes too // Now, do all holes too
for (final PolygonDescription hole : shape.holes) { for (final PolygonDescription hole : shape.holes) {
testPointShape = convertPolygon(pointsList, hole, testPointShape); testPointShape = convertPolygon(pointsList, hole, testPointShape, !mustBeInside);
} }
// Done; return the updated test point shape. // Done; return the updated test point shape.
return testPointShape; return testPointShape;
} }
/**
* Class for tracking the best shape for finding a pole, and whether or not the pole
* must be inside or outside of the shape.
*/
private static class BestShape {
public final List<GeoPoint> points;
public boolean poleMustBeInside;
public BestShape(final List<GeoPoint> points, final boolean poleMustBeInside) {
this.points = points;
this.poleMustBeInside = poleMustBeInside;
}
}
/** /**
* Create a GeoPolygon using the specified points and holes and a test point. * Create a GeoPolygon using the specified points and holes and a test point.
* *