From 0f831bc3a212c09f01ff24b4be1f92f5ce0cc70f Mon Sep 17 00:00:00 2001 From: Luc Maisonobe <luc@orekit.org> Date: Tue, 25 Mar 2014 20:02:57 +0100 Subject: [PATCH] Use point and direction rather than two points for pixel intersection. --- .../rugged/core/BasicScanAlgorithm.java | 2 +- .../core/duvenhage/DuvenhageAlgorithm.java | 11 ++--- .../orekit/rugged/core/raster/SimpleTile.java | 48 +++++++++++-------- .../org/orekit/rugged/core/raster/Tile.java | 8 ++-- .../rugged/core/raster/SimpleTileTest.java | 33 ++++++++----- 5 files changed, 59 insertions(+), 43 deletions(-) diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java b/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java index 49351cbd..8836cbe0 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/BasicScanAlgorithm.java @@ -118,7 +118,7 @@ public class BasicScanAlgorithm implements IntersectionAlgorithm { for (final SimpleTile tile : scannedTiles) { for (int i = latitudeIndex(tile, minLatitude); i <= latitudeIndex(tile, maxLatitude); ++i) { for (int j = longitudeIndex(tile, minLongitude); j <= longitudeIndex(tile, maxLongitude); ++j) { - GeodeticPoint gp = tile.pixelIntersection(entryPoint, exitPoint, i, j); + GeodeticPoint gp = tile.pixelIntersection(entryPoint, ellipsoid.convertLos(entryPoint, los), i, j); if (gp != null) { final Vector3D point = ellipsoid.transform(gp); final double dot = Vector3D.dotProduct(point.subtract(position), los); diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java index b867f212..2f8aea9a 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/duvenhage/DuvenhageAlgorithm.java @@ -157,16 +157,13 @@ public class DuvenhageAlgorithm implements IntersectionAlgorithm { final GeodeticPoint exit, final int exitLat, final int exitLon) throws RuggedException, OrekitException { - System.out.println(entryLat + " " + entryLon + " / " + exitLat + " " + exitLon); if (entryLat == exitLat && entryLon == exitLon) { // we have narrowed the search down to a single Digital Elevation Model pixel - return tile.pixelIntersection(entry, exit, exitLat, exitLon); + return tile.pixelIntersection(entry, ellipsoid.convertLos(entry, los), exitLat, exitLon); } // find the deepest level in the min/max kd-tree at which entry and exit share a sub-tile final int level = tile.getMergeLevel(entryLat, entryLon, exitLat, exitLon); - System.out.println("level = " + level + ", min = " + tile.getMinElevation(exitLat, exitLon, level) + - ", max = " + tile.getMaxElevation(exitLat, exitLon, level)); if (level >= 0 && exit.getAltitude() >= tile.getMaxElevation(exitLat, exitLon, level)) { // the line-of-sight segment is fully above Digital Elevation Model // we can safely reject it and proceed to next part of the line-of-sight @@ -320,9 +317,9 @@ public class DuvenhageAlgorithm implements IntersectionAlgorithm { * @param position pixel position in ellipsoid frame * @return either p1 or p2, depending on which is closest to position */ - private Vector3D selectClosest(Vector3D p1, Vector3D p2, Vector3D position) { - return Vector3D.distance(p1, position) <= Vector3D.distance(p2, position) ? p1 : p2; - } + private Vector3D selectClosest(Vector3D p1, Vector3D p2, Vector3D position) { + return Vector3D.distance(p1, position) <= Vector3D.distance(p2, position) ? p1 : p2; + } /** Point at tile boundary. */ private static class LimitPoint { diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/raster/SimpleTile.java b/rugged-core/src/main/java/org/orekit/rugged/core/raster/SimpleTile.java index cc69f80c..cae84330 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/raster/SimpleTile.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/raster/SimpleTile.java @@ -16,6 +16,7 @@ */ package org.orekit.rugged.core.raster; +import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; import org.apache.commons.math3.util.FastMath; import org.apache.commons.math3.util.Precision; import org.orekit.bodies.GeodeticPoint; @@ -244,7 +245,7 @@ public class SimpleTile implements Tile { /** {@inheritDoc} */ @Override - public GeodeticPoint pixelIntersection(final GeodeticPoint pA, final GeodeticPoint pB, + public GeodeticPoint pixelIntersection(final GeodeticPoint p, final Vector3D los, final int latitudeIndex, final int longitudeIndex) throws RuggedException { @@ -257,15 +258,15 @@ public class SimpleTile implements Tile { final double z11 = getElevationAtIndices(latitudeIndex + 1, longitudeIndex + 1); // line-of-sight coordinates at close points - final double dxA = (pA.getLongitude() - x00) / longitudeStep; - final double dyA = (pA.getLatitude() - y00) / latitudeStep; - final double dzA = pA.getAltitude(); - final double dxB = (pB.getLongitude() - x00) / longitudeStep; - final double dyB = (pB.getLatitude() - y00) / latitudeStep; - final double dzB = pB.getAltitude(); + final double dxA = (p.getLongitude() - x00) / longitudeStep; + final double dyA = (p.getLatitude() - y00) / latitudeStep; + final double dzA = p.getAltitude(); + final double dxB = dxA + los.getX() / longitudeStep; + final double dyB = dyA + los.getY() / latitudeStep; + final double dzB = dzA + los.getZ(); - // points along line-of-sight can be defined as a linear combination - // between pA and pB depending on free variable t: p(t) = pA * (1 - t) + pB * t. + // points along line-of-sight can be defined as a linear progression + // along the line depending on free variable t: p(t) = p + t * los. // As the point latitude and longitude are linear with respect to t, // and as Digital Elevation Model is quadratic with respect to latitude // and longitude, the altitude of DEM at same horizontal position as @@ -307,8 +308,8 @@ public class SimpleTile implements Tile { } - final GeodeticPoint p1 = interpolate(t1, pA, dxA, dyA, pB, dxB, dyB); - final GeodeticPoint p2 = interpolate(t2, pA, dxA, dyA, pB, dxB, dyB); + final GeodeticPoint p1 = interpolate(t1, p, dxA, dyA, los); + final GeodeticPoint p2 = interpolate(t2, p, dxA, dyA, los); // select the first point along line-of-sight if (p1 == null) { @@ -321,23 +322,28 @@ public class SimpleTile implements Tile { } - /** Interpolate point on line. - * + /** Interpolate point along a line. + * @param t abscissa along the line + * @param p start point + * @param dxP relative coordinate of the start point with respect to current pixel + * @param dyP relative coordinate of the start point with respect to current pixel + * @param los direction of the line-of-sight, in geodetic space + * @return interpolated point along the line */ - private GeodeticPoint interpolate(final double t, - final GeodeticPoint pA, final double dxA, final double dyA, - final GeodeticPoint pB, final double dxB, final double dyB) { + private GeodeticPoint interpolate(final double t, final GeodeticPoint p, + final double dxP, final double dyP, + final Vector3D los) { if (Double.isInfinite(t)) { return null; } - final double dx = dxA * (1 - t) + dxB * t; - final double dy = dyA * (1 - t) + dyB * t; + final double dx = dxP + t * los.getX() / longitudeStep; + final double dy = dyP + t * los.getY() / latitudeStep; if (dx >= 0 && dx <= 1 && dy >= 0 && dy <= 1) { - return new GeodeticPoint(pA.getLatitude() * (1 - t) + pB.getLatitude() * t, - pA.getLongitude() * (1 - t) + pB.getLongitude() * t, - pA.getAltitude() * (1 - t) + pB.getAltitude() * t); + return new GeodeticPoint(p.getLatitude() + t * los.getY(), + p.getLongitude() + t * los.getX(), + p.getAltitude() + t * los.getZ()); } else { return null; } diff --git a/rugged-core/src/main/java/org/orekit/rugged/core/raster/Tile.java b/rugged-core/src/main/java/org/orekit/rugged/core/raster/Tile.java index 6a254e15..72ae22c3 100644 --- a/rugged-core/src/main/java/org/orekit/rugged/core/raster/Tile.java +++ b/rugged-core/src/main/java/org/orekit/rugged/core/raster/Tile.java @@ -16,6 +16,7 @@ */ package org.orekit.rugged.core.raster; +import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; import org.orekit.bodies.GeodeticPoint; import org.orekit.rugged.api.RuggedException; import org.orekit.rugged.api.UpdatableTile; @@ -145,8 +146,9 @@ public interface Tile extends UpdatableTile { throws RuggedException; /** Find the intersection of a line-of-sight and a Digital Elevation Model pixel. - * @param pA first point on the line (closer to satellite) - * @param pB second point on the line (closer to ground) + * @param p point on the line + * @param los line-of-sight, in the topocentric frame (East, North, Zenith) of the point, + * scaled to match radians in the horizontal plane and meters along the vertical axis * @param latitudeIndex latitude index of the Digital Elevation Model pixel * @param longitudeIndex longitude index of the Digital Elevation Model pixel * @return point corresponding to line-of-sight crossing the Digital Elevation Model surface @@ -154,7 +156,7 @@ public interface Tile extends UpdatableTile { * @exception RuggedException if intersection point cannot be computed * @exception OrekitException if intersection point cannot be converted to geodetic coordinates */ - GeodeticPoint pixelIntersection(GeodeticPoint pA, GeodeticPoint pB, + GeodeticPoint pixelIntersection(GeodeticPoint p, Vector3D los, int latitudeIndex, int longitudeIndex) throws RuggedException; diff --git a/rugged-core/src/test/java/org/orekit/rugged/core/raster/SimpleTileTest.java b/rugged-core/src/test/java/org/orekit/rugged/core/raster/SimpleTileTest.java index a2867eca..7b68448a 100644 --- a/rugged-core/src/test/java/org/orekit/rugged/core/raster/SimpleTileTest.java +++ b/rugged-core/src/test/java/org/orekit/rugged/core/raster/SimpleTileTest.java @@ -16,6 +16,7 @@ */ package org.orekit.rugged.core.raster; +import org.apache.commons.math3.geometry.euclidean.threed.Vector3D; import org.apache.commons.math3.util.FastMath; import org.junit.Assert; import org.junit.Test; @@ -187,10 +188,10 @@ public class SimpleTileTest { GeodeticPoint gpB = new GeodeticPoint(tile.getLatitudeAtIndex(20) + 0.7 * tile.getLatitudeStep(), tile.getLongitudeAtIndex(14) + 0.9 * tile.getLongitudeStep(), 10.0); - GeodeticPoint gpIAB = tile.pixelIntersection(gpA, gpB, 20, 14); + GeodeticPoint gpIAB = tile.pixelIntersection(gpA, los(gpA, gpB), 20, 14); checkInLine(gpA, gpB, gpIAB); checkOnTile(tile, gpIAB); - GeodeticPoint gpIBA = tile.pixelIntersection(gpB, gpA, 20, 14); + GeodeticPoint gpIBA = tile.pixelIntersection(gpB, los(gpB, gpA), 20, 14); checkInLine(gpA, gpB, gpIBA); checkOnTile(tile, gpIBA); @@ -218,10 +219,10 @@ public class SimpleTileTest { // the line from gpA to gpB should traverse the DEM twice within the tile // we use the points in the two different orders to retrieve both solutions - GeodeticPoint gpIAB = tile.pixelIntersection(gpA, gpB, 20, 14); + GeodeticPoint gpIAB = tile.pixelIntersection(gpA, los(gpA, gpB), 20, 14); checkInLine(gpA, gpB, gpIAB); checkOnTile(tile, gpIAB); - GeodeticPoint gpIBA = tile.pixelIntersection(gpB, gpA, 20, 14); + GeodeticPoint gpIBA = tile.pixelIntersection(gpB, los(gpB, gpA), 20, 14); checkInLine(gpA, gpB, gpIBA); checkOnTile(tile, gpIBA); @@ -247,7 +248,7 @@ public class SimpleTileTest { tile.getLongitudeAtIndex(14) + 0.9 * tile.getLongitudeStep(), 190.0); - Assert.assertNull(tile.pixelIntersection(gpA, gpB, 20, 14)); + Assert.assertNull(tile.pixelIntersection(gpA, los(gpA, gpB), 20, 14)); } @@ -267,10 +268,10 @@ public class SimpleTileTest { tile.getLongitudeAtIndex(0) + 0.50 * tile.getLongitudeStep(), 20.0); - GeodeticPoint gpIAB = tile.pixelIntersection(gpA, gpB, 0, 0); + GeodeticPoint gpIAB = tile.pixelIntersection(gpA, los(gpA, gpB), 0, 0); checkInLine(gpA, gpB, gpIAB); checkOnTile(tile, gpIAB); - GeodeticPoint gpIBA = tile.pixelIntersection(gpB, gpA, 0, 0); + GeodeticPoint gpIBA = tile.pixelIntersection(gpB, los(gpB, gpA), 0, 0); checkInLine(gpA, gpB, gpIBA); checkOnTile(tile, gpIBA); @@ -296,7 +297,7 @@ public class SimpleTileTest { tile.getLongitudeAtIndex(0) + 0.50 * tile.getLongitudeStep(), 55.0); - Assert.assertNull(tile.pixelIntersection(gpA, gpB, 0, 0)); + Assert.assertNull(tile.pixelIntersection(gpA, los(gpA, gpB), 0, 0)); } @@ -316,7 +317,7 @@ public class SimpleTileTest { tile.getLongitudeAtIndex(0) + 0.50 * tile.getLongitudeStep(), 50.0); - Assert.assertNull(tile.pixelIntersection(gpA, gpB, 0, 0)); + Assert.assertNull(tile.pixelIntersection(gpA, los(gpA, gpB), 0, 0)); } @@ -336,10 +337,10 @@ public class SimpleTileTest { tile.getLongitudeAtIndex(0) + 0.50 * tile.getLongitudeStep(), 37.5); - GeodeticPoint gpIAB = tile.pixelIntersection(gpA, gpB, 0, 0); + GeodeticPoint gpIAB = tile.pixelIntersection(gpA, los(gpA, gpB), 0, 0); checkInLine(gpA, gpB, gpIAB); checkOnTile(tile, gpIAB); - GeodeticPoint gpIBA = tile.pixelIntersection(gpB, gpA, 0, 0); + GeodeticPoint gpIBA = tile.pixelIntersection(gpB, los(gpB, gpA), 0, 0); checkInLine(gpA, gpB, gpIBA); checkOnTile(tile, gpIBA); @@ -352,6 +353,16 @@ public class SimpleTileTest { } + private Vector3D los(GeodeticPoint gpA, GeodeticPoint gpB) { + // this is a crude conversion into geodetic space + // intended *only* for the purposes of these tests + // it considers the geodetic space *is* perfectly Cartesian + // in the East, North, Zenith frame + return new Vector3D(gpB.getLongitude() - gpA.getLongitude(), + gpB.getLatitude() - gpA.getLatitude(), + gpB.getAltitude() - gpA.getAltitude()); + } + private void checkOutOfBound(double latitude, double longitude, Tile tile) { try { tile.interpolateElevation(latitude, longitude); -- GitLab