Fixed errors in spherical polygons build from vertices.

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1555871 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Luc Maisonobe 2014-01-06 15:45:43 +00:00
parent 0d606bb16f
commit f2db15c31f
2 changed files with 128 additions and 70 deletions

View File

@ -211,7 +211,7 @@ public class SphericalPolygonsSet extends AbstractRegion<Sphere2D, Sphere1D> {
} }
/** Recursively build a tree by inserting cut sub-hyperplanes. /** Recursively build a tree by inserting cut sub-hyperplanes.
* @param hyperplaneThickness tolerance below which points are consider to * @param hyperplaneThickness tolerance below which points are considered to
* belong to the hyperplane (which is therefore more a slab) * belong to the hyperplane (which is therefore more a slab)
* @param node current tree node (it is a leaf node at the beginning * @param node current tree node (it is a leaf node at the beginning
* of the call) * of the call)
@ -264,12 +264,12 @@ public class SphericalPolygonsSet extends AbstractRegion<Sphere2D, Sphere1D> {
if (!outsideList.isEmpty()) { if (!outsideList.isEmpty()) {
insertEdges(hyperplaneThickness, node.getPlus(), outsideList); insertEdges(hyperplaneThickness, node.getPlus(), outsideList);
} else { } else {
node.getMinus().setAttribute(Boolean.FALSE); node.getPlus().setAttribute(Boolean.FALSE);
} }
if (!insideList.isEmpty()) { if (!insideList.isEmpty()) {
insertEdges(hyperplaneThickness, node.getMinus(), insideList); insertEdges(hyperplaneThickness, node.getMinus(), insideList);
} else { } else {
node.getPlus().setAttribute(Boolean.TRUE); node.getMinus().setAttribute(Boolean.TRUE);
} }
} }
@ -494,7 +494,7 @@ public class SphericalPolygonsSet extends AbstractRegion<Sphere2D, Sphere1D> {
private void split(final Circle splitCircle, private void split(final Circle splitCircle,
final List<Edge> outsideList, final List<Edge> insideList) { final List<Edge> outsideList, final List<Edge> insideList) {
// get the inside chord, synchronizing its phase with the edge itself // get the inside arc, synchronizing its phase with the edge itself
final double edgeStart = circle.getPhase(start.getLocation().getVector()); final double edgeStart = circle.getPhase(start.getLocation().getVector());
final double arcRelativeStart = MathUtils.normalizeAngle(circle.getInsideArc(splitCircle).getInf(), final double arcRelativeStart = MathUtils.normalizeAngle(circle.getInsideArc(splitCircle).getInf(),
edgeStart + FastMath.PI) - edgeStart; edgeStart + FastMath.PI) - edgeStart;
@ -502,80 +502,93 @@ public class SphericalPolygonsSet extends AbstractRegion<Sphere2D, Sphere1D> {
final double unwrappedEnd = arcRelativeStart - FastMath.PI; final double unwrappedEnd = arcRelativeStart - FastMath.PI;
// build the sub-edges // build the sub-edges
if (arcRelativeStart < length) { final double tolerance = circle.getTolerance();
if (unwrappedEnd > 0) { Vertex previousVertex = start;
// the edge starts inside the circle, then goes outside, then comes back inside if (unwrappedEnd >= length - tolerance) {
// create intermediate vertices // the edge is entirely contained inside the circle
final Vertex vExit = new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeEnd))); // we don't split anything
final Vertex vEnter = new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeStart))); insideList.add(this);
vExit.bindWith(splitCircle);
vEnter.bindWith(splitCircle);
// create sub-edges
final Edge eStartIn = new Edge(start, vExit, unwrappedEnd, circle);
final Edge eMiddleOut = new Edge(vExit, vEnter, arcRelativeStart - unwrappedEnd, circle);
final Edge eEndIn = new Edge(vEnter, end, length - arcRelativeStart, circle);
eStartIn.setNode(node);
eMiddleOut.setNode(node);
eEndIn.setNode(node);
// distribute the sub-edges in the appropriate lists
insideList.add(eStartIn);
insideList.add(eEndIn);
outsideList.add(eMiddleOut);
} else {
// the edge starts outside of the circle, then comes inside
// create intermediate vertices
final Vertex vEnter = new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeStart)));
vEnter.bindWith(splitCircle);
// create sub-edges
final Edge eStartOut = new Edge(start, vEnter, arcRelativeStart, circle);
final Edge eEndIn = new Edge(vEnter, end, length - arcRelativeStart, circle);
eStartOut.setNode(node);
eEndIn.setNode(node);
// distribute the sub-edges in the appropriate lists
outsideList.add(eStartOut);
insideList.add(eEndIn);
}
} else { } else {
if (unwrappedEnd > 0) {
if (unwrappedEnd > length) { // there are at least some parts of the edge that should be outside
// the edge is entirely contained inside the circle // (even is they are later be filtered out as being too small)
// we don't split anything double alreadyManagedLength = 0;
insideList.add(this); if (unwrappedEnd >= 0) {
// the start of the edge is inside the circle
previousVertex = addSubEdge(previousVertex,
new Vertex(new S2Point(circle.getPointAt(edgeStart + unwrappedEnd))),
unwrappedEnd, insideList, splitCircle);
alreadyManagedLength = unwrappedEnd;
}
if (arcRelativeStart >= length - tolerance) {
// the edge ends while still outside of the circle
if (unwrappedEnd >= 0) {
previousVertex = addSubEdge(previousVertex, end,
length - alreadyManagedLength, outsideList, splitCircle);
} else { } else {
// the edge starts inside the circle, then goes outside // the edge is entirely outside of the circle
// we don't split anything
// create intermediate vertices outsideList.add(this);
final Vertex vExit = new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeEnd)));
vExit.bindWith(splitCircle);
// create sub-edges
final Edge eStartIn = new Edge(start, vExit, arcRelativeEnd, circle);
final Edge eEndOut = new Edge(vExit, end, length - arcRelativeEnd, circle);
eStartIn.setNode(node);
eEndOut.setNode(node);
// distribute the sub-edges in the appropriate lists
insideList.add(eStartIn);
outsideList.add(eEndOut);
} }
} else { } else {
// the edge is entirely outside of the circle // the edge is long enough to enter inside the circle
// we don't split anything previousVertex = addSubEdge(previousVertex,
outsideList.add(this); new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeStart))),
arcRelativeStart - alreadyManagedLength, outsideList, splitCircle);
alreadyManagedLength = arcRelativeStart;
if (arcRelativeEnd >= length - tolerance) {
// the edge ends while still inside of the circle
previousVertex = addSubEdge(previousVertex, end,
length - alreadyManagedLength, insideList, splitCircle);
} else {
// the edge is long enough to exit outside of the circle
previousVertex = addSubEdge(previousVertex,
new Vertex(new S2Point(circle.getPointAt(edgeStart + arcRelativeStart))),
arcRelativeStart - alreadyManagedLength, insideList, splitCircle);
alreadyManagedLength = arcRelativeStart;
previousVertex = addSubEdge(previousVertex, end,
length - alreadyManagedLength, outsideList, splitCircle);
}
} }
} }
} }
/** Add a sub-edge to a list if long enough.
* <p>
* If the length of the sub-edge to add is smaller than the {@link Circle#getTolerance()}
* tolerance of the support circle, it will be ignored.
* </p>
* @param subStart start of the sub-edge
* @param subEnd end of the sub-edge
* @param subLength length of the sub-edge
* @param splitCircle circle splitting the edge in several parts
* @param list list where to put the sub-edge
* @return end vertex of the edge ({@code subEnd} if the edge was long enough and really
* added, {@code subStart} if the edge was too small and therefore ignored)
*/
private Vertex addSubEdge(final Vertex subStart, final Vertex subEnd, final double subLength,
final List<Edge> list, final Circle splitCircle) {
if (subLength <= circle.getTolerance()) {
// the edge is too short, we ignore it
return subStart;
}
// really add the edge
subEnd.bindWith(splitCircle);
final Edge edge = new Edge(subStart, subEnd, subLength, circle);
edge.setNode(node);
list.add(edge);
return subEnd;
}
} }
/** {@inheritDoc} */ /** {@inheritDoc} */

View File

@ -18,6 +18,7 @@ package org.apache.commons.math3.geometry.spherical.twod;
import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; import org.apache.commons.math3.geometry.euclidean.threed.Vector3D;
import org.apache.commons.math3.geometry.partitioning.Region.Location; import org.apache.commons.math3.geometry.partitioning.Region.Location;
import org.apache.commons.math3.geometry.partitioning.RegionFactory;
import org.apache.commons.math3.random.UnitSphereRandomVectorGenerator; import org.apache.commons.math3.random.UnitSphereRandomVectorGenerator;
import org.apache.commons.math3.random.Well1024a; import org.apache.commons.math3.random.Well1024a;
import org.apache.commons.math3.util.FastMath; import org.apache.commons.math3.util.FastMath;
@ -35,6 +36,8 @@ public class SphericalPolygonsSetTest {
Vector3D v = new Vector3D(random.nextVector()); Vector3D v = new Vector3D(random.nextVector());
Assert.assertEquals(Location.INSIDE, full.checkPoint(new S2Point(v))); Assert.assertEquals(Location.INSIDE, full.checkPoint(new S2Point(v)));
} }
Assert.assertEquals(4 * FastMath.PI, new SphericalPolygonsSet(0.01, new S2Point[0]).getSize(), 1.0e-10);
Assert.assertEquals(0, new SphericalPolygonsSet(0.01, new S2Point[0]).getBoundarySize(), 1.0e-10);
} }
@Test @Test
@ -43,7 +46,7 @@ public class SphericalPolygonsSetTest {
double sinTol = FastMath.sin(tol); double sinTol = FastMath.sin(tol);
SphericalPolygonsSet south = new SphericalPolygonsSet(Vector3D.MINUS_K, tol); SphericalPolygonsSet south = new SphericalPolygonsSet(Vector3D.MINUS_K, tol);
UnitSphereRandomVectorGenerator random = UnitSphereRandomVectorGenerator random =
new UnitSphereRandomVectorGenerator(3, new Well1024a(0x852fd2a0ed8d2f6dl)); new UnitSphereRandomVectorGenerator(3, new Well1024a(0x6b9d4a6ad90d7b0bl));
for (int i = 0; i < 1000; ++i) { for (int i = 0; i < 1000; ++i) {
Vector3D v = new Vector3D(random.nextVector()); Vector3D v = new Vector3D(random.nextVector());
if (v.getZ() < -sinTol) { if (v.getZ() < -sinTol) {
@ -51,8 +54,50 @@ public class SphericalPolygonsSetTest {
} else if (v.getZ() > sinTol) { } else if (v.getZ() > sinTol) {
Assert.assertEquals(Location.OUTSIDE, south.checkPoint(new S2Point(v))); Assert.assertEquals(Location.OUTSIDE, south.checkPoint(new S2Point(v)));
} else { } else {
Assert.assertEquals("" + v.getX() + " " + v.getY() + " " + v.getZ(), Assert.assertEquals(Location.BOUNDARY, south.checkPoint(new S2Point(v)));
Location.BOUNDARY, south.checkPoint(new S2Point(v))); }
}
}
@Test
public void testPositiveOctantByIntersection() {
double tol = 0.01;
double sinTol = FastMath.sin(tol);
RegionFactory<Sphere2D> factory = new RegionFactory<Sphere2D>();
SphericalPolygonsSet plusX = new SphericalPolygonsSet(Vector3D.PLUS_I, tol);
SphericalPolygonsSet plusY = new SphericalPolygonsSet(Vector3D.PLUS_J, tol);
SphericalPolygonsSet plusZ = new SphericalPolygonsSet(Vector3D.PLUS_K, tol);
SphericalPolygonsSet octant =
(SphericalPolygonsSet) factory.intersection(factory.intersection(plusX, plusY), plusZ);
UnitSphereRandomVectorGenerator random =
new UnitSphereRandomVectorGenerator(3, new Well1024a(0x9c9802fde3cbcf25l));
for (int i = 0; i < 1000; ++i) {
Vector3D v = new Vector3D(random.nextVector());
if ((v.getX() > sinTol) && (v.getY() > sinTol) && (v.getZ() > sinTol)) {
Assert.assertEquals(Location.INSIDE, octant.checkPoint(new S2Point(v)));
} else if ((v.getX() < -sinTol) || (v.getY() < -sinTol) || (v.getZ() < -sinTol)) {
Assert.assertEquals(Location.OUTSIDE, octant.checkPoint(new S2Point(v)));
} else {
Assert.assertEquals(Location.BOUNDARY, octant.checkPoint(new S2Point(v)));
}
}
}
@Test
public void testPositiveOctantByVertices() {
double tol = 0.01;
double sinTol = FastMath.sin(tol);
SphericalPolygonsSet octant = new SphericalPolygonsSet(tol, S2Point.PLUS_I, S2Point.PLUS_J, S2Point.PLUS_K);
UnitSphereRandomVectorGenerator random =
new UnitSphereRandomVectorGenerator(3, new Well1024a(0xb8fc5acc91044308l));
for (int i = 0; i < 1000; ++i) {
Vector3D v = new Vector3D(random.nextVector());
if ((v.getX() > sinTol) && (v.getY() > sinTol) && (v.getZ() > sinTol)) {
Assert.assertEquals(Location.INSIDE, octant.checkPoint(new S2Point(v)));
} else if ((v.getX() < -sinTol) || (v.getY() < -sinTol) || (v.getZ() < -sinTol)) {
Assert.assertEquals(Location.OUTSIDE, octant.checkPoint(new S2Point(v)));
} else {
Assert.assertEquals(Location.BOUNDARY, octant.checkPoint(new S2Point(v)));
} }
} }
} }