LUCENE-8680: Refactor EdgeTree#relateTriangle method

This commit is contained in:
iverase 2019-02-08 16:19:38 +01:00
parent 796fbaef76
commit 06c1ebc09e
3 changed files with 64 additions and 64 deletions

View File

@ -124,12 +124,12 @@ public abstract class EdgeTree {
return Relation.CELL_OUTSIDE_QUERY; return Relation.CELL_OUTSIDE_QUERY;
} }
protected Relation componentRelate(double minLat, double maxLat, double minLon, double maxLon) { /** Returns relation to the provided rectangle for this component */
return null; protected abstract Relation componentRelate(double minLat, double maxLat, double minLon, double maxLon);
}
protected Relation componentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) { /** Returns relation to the provided triangle for this component */
return null; protected abstract Relation componentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy);
}
private Relation internalComponentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) { private Relation internalComponentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) {
// compute bounding box of triangle // compute bounding box of triangle
@ -140,22 +140,7 @@ public abstract class EdgeTree {
if (maxLon < this.minLon || minLon > this.maxLon || maxLat < this.minLat || minLat > this.maxLat) { if (maxLon < this.minLon || minLon > this.maxLon || maxLat < this.minLat || minLat > this.maxLat) {
return Relation.CELL_OUTSIDE_QUERY; return Relation.CELL_OUTSIDE_QUERY;
} }
return componentRelateTriangle(ax, ay, bx, by, cx, cy);
Relation shapeRelation = componentRelateTriangle(ax, ay, bx, by, cx, cy);
if (shapeRelation != null) {
return shapeRelation;
}
// we cross
if ((shapeRelation = tree.relateTriangle(ax, ay, bx, by, cx, cy)) != Relation.CELL_OUTSIDE_QUERY) {
return shapeRelation;
}
if (pointInTriangle(tree.lon1, tree.lat1, ax, ay, bx, by, cx, cy) == true) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_OUTSIDE_QUERY;
} }
@ -169,18 +154,7 @@ public abstract class EdgeTree {
if (minLat <= this.minLat && maxLat >= this.maxLat && minLon <= this.minLon && maxLon >= this.maxLon) { if (minLat <= this.minLat && maxLat >= this.maxLat && minLon <= this.minLon && maxLon >= this.maxLon) {
return Relation.CELL_CROSSES_QUERY; return Relation.CELL_CROSSES_QUERY;
} }
return componentRelate(minLat, maxLat, minLon, maxLon);
Relation shapeRelation = componentRelate(minLat, maxLat, minLon, maxLon);
if (shapeRelation != null) {
return shapeRelation;
}
// we cross
if (tree.crosses(minLat, maxLat, minLon, maxLon)) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_OUTSIDE_QUERY;
} }
/** Creates tree from sorted components (with range low and high inclusive) */ /** Creates tree from sorted components (with range low and high inclusive) */
@ -402,7 +376,7 @@ public abstract class EdgeTree {
//This should be moved when LatLonShape is moved from sandbox! //This should be moved when LatLonShape is moved from sandbox!
/** /**
* Compute whether the given x, y point is in a triangle; uses the winding order method */ * Compute whether the given x, y point is in a triangle; uses the winding order method */
private static boolean pointInTriangle (double x, double y, double ax, double ay, double bx, double by, double cx, double cy) { protected static boolean pointInTriangle (double x, double y, double ax, double ay, double bx, double by, double cx, double cy) {
double minX = StrictMath.min(ax, StrictMath.min(bx, cx)); double minX = StrictMath.min(ax, StrictMath.min(bx, cx));
double minY = StrictMath.min(ay, StrictMath.min(by, cy)); double minY = StrictMath.min(ay, StrictMath.min(by, cy));
double maxX = StrictMath.max(ax, StrictMath.max(bx, cx)); double maxX = StrictMath.max(ax, StrictMath.max(bx, cx));

View File

@ -78,31 +78,6 @@ public final class Polygon2D extends EdgeTree {
return false; return false;
} }
@Override
protected Relation componentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) {
// check any holes
if (holes != null) {
Relation holeRelation = holes.relateTriangle(ax, ay, bx, by, cx, cy);
if (holeRelation == Relation.CELL_CROSSES_QUERY) {
return Relation.CELL_CROSSES_QUERY;
} else if (holeRelation == Relation.CELL_INSIDE_QUERY) {
return Relation.CELL_OUTSIDE_QUERY;
}
}
// check each corner: if < 3 are present, its cheaper than crossesSlowly
int numCorners = numberOfTriangleCorners(ax, ay, bx, by, cx, cy);
if (numCorners == 3) {
if (tree.relateTriangle(ax, ay, bx, by, cx, cy) == Relation.CELL_CROSSES_QUERY) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_INSIDE_QUERY;
} else if (numCorners > 0) {
return Relation.CELL_CROSSES_QUERY;
}
return null;
}
/** Returns relation to the provided rectangle for this component */
@Override @Override
protected Relation componentRelate(double minLat, double maxLat, double minLon, double maxLon) { protected Relation componentRelate(double minLat, double maxLat, double minLon, double maxLon) {
// check any holes // check any holes
@ -114,17 +89,53 @@ public final class Polygon2D extends EdgeTree {
return Relation.CELL_OUTSIDE_QUERY; return Relation.CELL_OUTSIDE_QUERY;
} }
} }
// check each corner: if < 4 are present, its cheaper than crossesSlowly // check each corner: if < 4 && > 0 are present, its cheaper than crossesSlowly
int numCorners = numberOfCorners(minLat, maxLat, minLon, maxLon); int numCorners = numberOfCorners(minLat, maxLat, minLon, maxLon);
if (numCorners == 4) { if (numCorners == 4) {
if (tree.crosses(minLat, maxLat, minLon, maxLon)) { if (tree.crosses(minLat, maxLat, minLon, maxLon)) {
return Relation.CELL_CROSSES_QUERY; return Relation.CELL_CROSSES_QUERY;
} }
return Relation.CELL_INSIDE_QUERY; return Relation.CELL_INSIDE_QUERY;
} else if (numCorners > 0) { } else if (numCorners == 0) {
if (minLat >= tree.lat1 && maxLat <= tree.lat1 && minLon >= tree.lon2 && maxLon <= tree.lon2) {
return Relation.CELL_CROSSES_QUERY; return Relation.CELL_CROSSES_QUERY;
} }
return null; if (tree.crosses(minLat, maxLat, minLon, maxLon)) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_OUTSIDE_QUERY;
}
return Relation.CELL_CROSSES_QUERY;
}
@Override
protected Relation componentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) {
// check any holes
if (holes != null) {
Relation holeRelation = holes.relateTriangle(ax, ay, bx, by, cx, cy);
if (holeRelation == Relation.CELL_CROSSES_QUERY) {
return Relation.CELL_CROSSES_QUERY;
} else if (holeRelation == Relation.CELL_INSIDE_QUERY) {
return Relation.CELL_OUTSIDE_QUERY;
}
}
// check each corner: if < 3 && > 0 are present, its cheaper than crossesSlowly
int numCorners = numberOfTriangleCorners(ax, ay, bx, by, cx, cy);
if (numCorners == 3) {
if (tree.relateTriangle(ax, ay, bx, by, cx, cy) == Relation.CELL_CROSSES_QUERY) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_INSIDE_QUERY;
} else if (numCorners == 0) {
if (pointInTriangle(tree.lon1, tree.lat1, ax, ay, bx, by, cx, cy) == true) {
return Relation.CELL_CROSSES_QUERY;
}
if (tree.relateTriangle(ax, ay, bx, by, cx, cy) == Relation.CELL_CROSSES_QUERY) {
return Relation.CELL_CROSSES_QUERY;
}
return Relation.CELL_OUTSIDE_QUERY;
}
return Relation.CELL_CROSSES_QUERY;
} }
private int numberOfTriangleCorners(double ax, double ay, double bx, double by, double cx, double cy) { private int numberOfTriangleCorners(double ax, double ay, double bx, double by, double cx, double cy) {

View File

@ -16,6 +16,8 @@
*/ */
package org.apache.lucene.geo; package org.apache.lucene.geo;
import org.apache.lucene.index.PointValues;
/** /**
* 2D line implementation represented as a balanced interval tree of edges. * 2D line implementation represented as a balanced interval tree of edges.
* <p> * <p>
@ -37,4 +39,17 @@ public final class Line2D extends EdgeTree {
} }
return (Line2D)createTree(components, 0, components.length - 1, false); return (Line2D)createTree(components, 0, components.length - 1, false);
} }
@Override
protected PointValues.Relation componentRelate(double minLat, double maxLat, double minLon, double maxLon) {
if (tree.crosses(minLat, maxLat, minLon, maxLon)) {
return PointValues.Relation.CELL_CROSSES_QUERY;
}
return PointValues.Relation.CELL_OUTSIDE_QUERY;
}
@Override
protected PointValues.Relation componentRelateTriangle(double ax, double ay, double bx, double by, double cx, double cy) {
return tree.relateTriangle(ax, ay, bx, by, cx, cy);
}
} }