mirror of https://github.com/apache/lucene.git
LUCENE-6487: Geo3D with WGS84 patch from Karl: GeoPoint.getLat & getLon.
git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/branches/lucene6487@1682021 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
560f3a1a01
commit
0a00734a60
|
@ -24,30 +24,79 @@ package org.apache.lucene.spatial.spatial4j.geo3d;
|
||||||
*/
|
*/
|
||||||
public class GeoPoint extends Vector {
|
public class GeoPoint extends Vector {
|
||||||
|
|
||||||
|
/** This is the lazily-evaluated magnitude. Some constructors include it, but others don't, and
|
||||||
|
* we try not to create extra computation by always computing it. */
|
||||||
protected double magnitude = Double.NEGATIVE_INFINITY;
|
protected double magnitude = Double.NEGATIVE_INFINITY;
|
||||||
|
|
||||||
|
/** Construct a GeoPoint from the trig functions of a lat and lon pair.
|
||||||
|
* @param planetModel is the planetModel to put the point on.
|
||||||
|
* @param sinLat is the sin of the latitude.
|
||||||
|
* @param sinLon is the sin of the longitude.
|
||||||
|
* @param cosLat is the cos of the latitude.
|
||||||
|
* @param cosLon is the cos of the longitude.
|
||||||
|
*/
|
||||||
public GeoPoint(final PlanetModel planetModel, final double sinLat, final double sinLon, final double cosLat, final double cosLon) {
|
public GeoPoint(final PlanetModel planetModel, final double sinLat, final double sinLon, final double cosLat, final double cosLon) {
|
||||||
this(computeMagnitude(planetModel, cosLat * cosLon, cosLat * sinLon, sinLat),
|
this(computeDesiredEllipsoidMagnitude(planetModel, cosLat * cosLon, cosLat * sinLon, sinLat),
|
||||||
cosLat * cosLon, cosLat * sinLon, sinLat);
|
cosLat * cosLon, cosLat * sinLon, sinLat);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Construct a GeoPoint from a latitude/longitude pair.
|
||||||
|
* @param planetModel is the planetModel to put the point on.
|
||||||
|
* @param lat is the latitude.
|
||||||
|
* @param lon is the longitude.
|
||||||
|
*/
|
||||||
public GeoPoint(final PlanetModel planetModel, final double lat, final double lon) {
|
public GeoPoint(final PlanetModel planetModel, final double lat, final double lon) {
|
||||||
this(planetModel, Math.sin(lat), Math.sin(lon), Math.cos(lat), Math.cos(lon));
|
this(planetModel, Math.sin(lat), Math.sin(lon), Math.cos(lat), Math.cos(lon));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Construct a GeoPoint from a unit (x,y,z) vector and a magnitude.
|
||||||
|
* @param magnitude is the desired magnitude, provided to put the point on the ellipsoid.
|
||||||
|
* @param x is the unit x value.
|
||||||
|
* @param y is the unit y value.
|
||||||
|
* @param z is the unit z value.
|
||||||
|
*/
|
||||||
public GeoPoint(final double magnitude, final double x, final double y, final double z) {
|
public GeoPoint(final double magnitude, final double x, final double y, final double z) {
|
||||||
super(x * magnitude, y * magnitude, z * magnitude);
|
super(x * magnitude, y * magnitude, z * magnitude);
|
||||||
this.magnitude = magnitude;
|
this.magnitude = magnitude;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Construct a GeoPoint from an (x,y,z) value.
|
||||||
|
* The (x,y,z) tuple must be on the desired ellipsoid.
|
||||||
|
* @param x is the ellipsoid point x value.
|
||||||
|
* @param y is the ellipsoid point y value.
|
||||||
|
* @param z is the ellipsoid point z value.
|
||||||
|
*/
|
||||||
public GeoPoint(final double x, final double y, final double z) {
|
public GeoPoint(final double x, final double y, final double z) {
|
||||||
super(x, y, z);
|
super(x, y, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Compute an arc distance between two points.
|
||||||
|
* @param v is the second point.
|
||||||
|
* @return the angle, in radians, between the two points.
|
||||||
|
*/
|
||||||
public double arcDistance(final GeoPoint v) {
|
public double arcDistance(final GeoPoint v) {
|
||||||
return Tools.safeAcos(dotProduct(v)/(magnitude() * v.magnitude()));
|
return Tools.safeAcos(dotProduct(v)/(magnitude() * v.magnitude()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Compute the latitude for the point.
|
||||||
|
*@return the latitude.
|
||||||
|
*/
|
||||||
|
public double getLatitude() {
|
||||||
|
return Math.asin(z / magnitude() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Compute the longitude for the point.
|
||||||
|
* @return the longitude value. Uses 0.0 if there is no computable longitude.
|
||||||
|
*/
|
||||||
|
public double getLongitude() {
|
||||||
|
if (Math.abs(x) < MINIMUM_RESOLUTION && Math.abs(y) < MINIMUM_RESOLUTION)
|
||||||
|
return 0.0;
|
||||||
|
return Math.atan2(y,z);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Compute the linear magnitude of the point.
|
||||||
|
* @return the magnitude.
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public double magnitude() {
|
public double magnitude() {
|
||||||
if (this.magnitude == Double.NEGATIVE_INFINITY) {
|
if (this.magnitude == Double.NEGATIVE_INFINITY) {
|
||||||
|
|
|
@ -56,7 +56,7 @@ public class Plane extends Vector {
|
||||||
*/
|
*/
|
||||||
public Plane(final PlanetModel planetModel, final double sinLat) {
|
public Plane(final PlanetModel planetModel, final double sinLat) {
|
||||||
super(0.0, 0.0, 1.0);
|
super(0.0, 0.0, 1.0);
|
||||||
D = -sinLat * computeMagnitude(planetModel, sinLat);
|
D = -sinLat * computeDesiredEllipsoidMagnitude(planetModel, sinLat);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -307,17 +307,25 @@ public class Vector {
|
||||||
return Math.sqrt(x * x + y * y + z * z);
|
return Math.sqrt(x * x + y * y + z * z);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Compute the magnitude of a vector projected to a given
|
/** Compute the desired magnitude of a unit vector projected to a given
|
||||||
* planet model.
|
* planet model.
|
||||||
|
* @param planetModel is the planet model.
|
||||||
|
* @param x is the unit vector x value.
|
||||||
|
* @param y is the unit vector y value.
|
||||||
|
* @param z is the unit vector z value.
|
||||||
|
* @return a magnitude value for that (x,y,z) that projects the vector onto the specified ellipsoid.
|
||||||
*/
|
*/
|
||||||
protected static double computeMagnitude(final PlanetModel planetModel, final double x, final double y, final double z) {
|
protected static double computeDesiredEllipsoidMagnitude(final PlanetModel planetModel, final double x, final double y, final double z) {
|
||||||
return 1.0 / Math.sqrt(x*x*planetModel.inverseAbSquared + y*y*planetModel.inverseAbSquared + z*z*planetModel.inverseCSquared);
|
return 1.0 / Math.sqrt(x*x*planetModel.inverseAbSquared + y*y*planetModel.inverseAbSquared + z*z*planetModel.inverseCSquared);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Compute the magnitude of a vector projected to a given
|
/** Compute the desired magnitude of a unit vector projected to a given
|
||||||
* planet model.
|
* planet model. The unit vector is specified only by a z value.
|
||||||
|
* @param planetModel is the planet model.
|
||||||
|
* @param z is the unit vector z value.
|
||||||
|
* @return a magnitude value for that z value that projects the vector onto the specified ellipsoid.
|
||||||
*/
|
*/
|
||||||
protected static double computeMagnitude(final PlanetModel planetModel, final double z) {
|
protected static double computeDesiredEllipsoidMagnitude(final PlanetModel planetModel, final double z) {
|
||||||
return 1.0 / Math.sqrt((1.0-z*z)*planetModel.inverseAbSquared + z*z*planetModel.inverseCSquared);
|
return 1.0 / Math.sqrt((1.0-z*z)*planetModel.inverseAbSquared + z*z*planetModel.inverseCSquared);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -237,7 +237,7 @@ public abstract class Geo3dShapeRectRelationTestCase extends RandomizedShapeTest
|
||||||
}
|
}
|
||||||
|
|
||||||
private Point geoPointToSpatial4jPoint(GeoPoint geoPoint) {
|
private Point geoPointToSpatial4jPoint(GeoPoint geoPoint) {
|
||||||
return ctx.makePoint(geoPoint.x * DistanceUtils.RADIANS_TO_DEGREES,
|
return ctx.makePoint(geoPoint.getLongitude() * DistanceUtils.RADIANS_TO_DEGREES,
|
||||||
geoPoint.y * DistanceUtils.RADIANS_TO_DEGREES);
|
geoPoint.getLongitude() * DistanceUtils.RADIANS_TO_DEGREES);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue