org.hortonmachine.gears.utils.ENU Maven / Gradle / Ivy
/*
* This file is part of HortonMachine (http://www.hortonmachine.org)
* (C) HydroloGIS - www.hydrologis.com
*
* The HortonMachine is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
package org.hortonmachine.gears.utils;
import static java.lang.Math.atan;
import static java.lang.Math.atan2;
import static java.lang.Math.cos;
import static java.lang.Math.pow;
import static java.lang.Math.sin;
import static java.lang.Math.sqrt;
import static java.lang.Math.tan;
import static java.lang.Math.toDegrees;
import static java.lang.Math.toRadians;
import org.apache.commons.math3.linear.LUDecomposition;
import org.apache.commons.math3.linear.MatrixUtils;
import org.apache.commons.math3.linear.RealMatrix;
import org.hortonmachine.gears.utils.geometry.GeometryUtilities;
import org.hortonmachine.gears.utils.math.matrixes.MatrixException;
import org.locationtech.jts.geom.Coordinate;
import org.locationtech.jts.geom.Envelope;
import org.locationtech.jts.geom.Geometry;
import org.locationtech.jts.geom.Polygon;
/**
* Conversion of Geodetic coordinates to the Local Tangent Plane.
*
* Class that supports WGS84 to East-North-Up conversion. The conversions
* reference the base coordinate that is given at construction time.
*
* Math is available in the paper: Conversion of Geodetic coordinates to the
* Local Tangent Plane.
*
* The ecefToWgs84 method adapted from the goGPS project.
*
* Note that all coordinates need to have a Z available. If the Z is NaN, then it is set to 0.
*
* @author Andrea Antonello (www.hydrologis.com)
*/
public class ENU {
private static double semiMajorAxis = 6378137.0;
private static double smeiMinorAxis = 6356752.3142;
private static double flatness = (semiMajorAxis - smeiMinorAxis) / semiMajorAxis;
private static double eccentricityP2 = flatness * (2 - flatness);
private Coordinate _baseCoordinateLL;
private double _lambda;
private double _phi;
private double _sinLambda;
private double _cosLambda;
private double _cosPhi;
private double _sinPhi;
private double _N;
private RealMatrix _rotationMatrix;
private double _ecefROriginX;
private double _ecefROriginY;
private double _ecefROriginZ;
private RealMatrix _inverseRotationMatrix;
/**
* Create a new East North Up system.
*
* @param baseCoordinateLL the WGS84 coordinate to use a origin of the ENU.
*/
public ENU( Coordinate baseCoordinateLL ) {
checkZ(baseCoordinateLL);
this._baseCoordinateLL = baseCoordinateLL;
_lambda = toRadians(baseCoordinateLL.y);
_phi = toRadians(baseCoordinateLL.x);
_sinLambda = sin(_lambda);
_cosLambda = cos(_lambda);
_cosPhi = cos(_phi);
_sinPhi = sin(_phi);
_N = semiMajorAxis / sqrt(1 - eccentricityP2 * pow(_sinLambda, 2.0));
double[][] rot = new double[][]{//
{-_sinPhi, _cosPhi, 0}, //
{-_cosPhi * _sinLambda, -_sinLambda * _sinPhi, _cosLambda}, //
{_cosLambda * _cosPhi, _cosLambda * _sinPhi, _sinLambda},//
};
_rotationMatrix = MatrixUtils.createRealMatrix(rot);
_inverseRotationMatrix = new LUDecomposition(_rotationMatrix).getSolver().getInverse();
// the origin of the LTP expressed in ECEF-r coordinates
double h = _baseCoordinateLL.z;
_ecefROriginX = (h + _N) * _cosLambda * _cosPhi;
_ecefROriginY = (h + _N) * _cosLambda * _sinPhi;
_ecefROriginZ = (h + (1 - eccentricityP2) * _N) * _sinLambda;
}
/**
* Converts the wgs84 coordinate to ENU.
*
* @param cLL the wgs84 coordinate.
* @return the ENU coordinate.
* @throws MatrixException
*/
public Coordinate wgs84ToEnu( Coordinate cLL ) {
checkZ(cLL);
Coordinate cEcef = wgs84ToEcef(cLL);
Coordinate enu = ecefToEnu(cEcef);
return enu;
}
/**
* Converts the ENU coordinate to wgs84.
*
* @param enu the ENU coordinate.
* @return the wgs84 coordinate.
* @throws MatrixException
*/
public Coordinate enuToWgs84( Coordinate enu ) {
checkZ(enu);
Coordinate cEcef = enuToEcef(enu);
Coordinate wgs84 = ecefToWgs84(cEcef);
return wgs84;
}
/**
* Converts WGS84 coordinates to Earth-Centered Earth-Fixed (ECEF) coordinates.
*
* @param cLL the wgs84 coordinate.
* @return the ecef coordinate.
*/
public Coordinate wgs84ToEcef( Coordinate cLL ) {
double lambda = toRadians(cLL.y);
double phi = toRadians(cLL.x);
double sinLambda = sin(lambda);
double cosLambda = cos(lambda);
double cosPhi = cos(phi);
double sinPhi = sin(phi);
double N = semiMajorAxis / sqrt(1 - eccentricityP2 * pow(sinLambda, 2.0));
double h = cLL.z;
double x = (h + N) * cosLambda * cosPhi;
double y = (h + N) * cosLambda * sinPhi;
double z = (h + (1 - eccentricityP2) * N) * sinLambda;
return new Coordinate(x, y, z);
}
/**
* Converts an Earth-Centered Earth-Fixed (ECEF) coordinate to ENU.
*
* @param cEcef the ECEF coordinate.
* @return the ENU coordinate.
* @throws MatrixException
*/
public Coordinate ecefToEnu( Coordinate cEcef ) {
double deltaX = cEcef.x - _ecefROriginX;
double deltaY = cEcef.y - _ecefROriginY;
double deltaZ = cEcef.z - _ecefROriginZ;
double[][] deltas = new double[][]{{deltaX}, {deltaY}, {deltaZ}};
RealMatrix deltaMatrix = MatrixUtils.createRealMatrix(deltas);
RealMatrix enuMatrix = _rotationMatrix.multiply(deltaMatrix);
double[] column = enuMatrix.getColumn(0);
return new Coordinate(column[0], column[1], column[2]);
}
/**
* Converts an ENU coordinate to Earth-Centered Earth-Fixed (ECEF).
*
* @param cEnu the enu coordinate.
* @return the ecef coordinate.
*/
public Coordinate enuToEcef( Coordinate cEnu ) {
double[][] enu = new double[][]{{cEnu.x}, {cEnu.y}, {cEnu.z}};
RealMatrix enuMatrix = MatrixUtils.createRealMatrix(enu);
RealMatrix deltasMatrix = _inverseRotationMatrix.multiply(enuMatrix);
double[] column = deltasMatrix.getColumn(0);
double cecfX = column[0] + _ecefROriginX;
double cecfY = column[1] + _ecefROriginY;
double cecfZ = column[2] + _ecefROriginZ;
return new Coordinate(cecfX, cecfY, cecfZ);
}
/**
* Converts a Earth-Centered Earth-Fixed (ECEF) coordinate to WGS84.
*
* @param ecef the ecef coordinate.
* @return the wgs84 coordinate.
*/
public Coordinate ecefToWgs84( Coordinate ecef ) {
// Radius computation
double r = sqrt(pow(ecef.x, 2) + pow(ecef.y, 2) + pow(ecef.z, 2));
// Geocentric longitude
double lamGeoc = atan2(ecef.y, ecef.x);
// Geocentric latitude
double phiGeoc = atan(ecef.z / sqrt(pow(ecef.x, 2) + pow(ecef.y, 2)));
// Computation of geodetic coordinates
double psi = atan(tan(phiGeoc) / sqrt(1 - eccentricityP2));
double phiGeod = atan((r * sin(phiGeoc) + eccentricityP2 * semiMajorAxis / sqrt(1 - eccentricityP2) * pow(sin(psi), 3))
/ (r * cos(phiGeoc) - eccentricityP2 * semiMajorAxis * pow(cos(psi), 3)));
double lamGeod = lamGeoc;
double N = semiMajorAxis / sqrt(1 - eccentricityP2 * pow(sin(phiGeod), 2));
double h = r * cos(phiGeoc) / cos(phiGeod) - N;
double lon = toDegrees(lamGeod);
double lat = toDegrees(phiGeod);
return new Coordinate(lon, lat, h);
}
/**
* Converts a geometry from ENU to WGS.
*
* Note that the geometry is internally converted. Use clone() if you need a copy.
*
* @param geometryEnu
*/
public void convertGeometryFromEnuToWgs84( Geometry geometryEnu ) {
Coordinate[] coordinates = geometryEnu.getCoordinates();
for( int i = 0; i < coordinates.length; i++ ) {
Coordinate wgs84 = enuToWgs84(coordinates[i]);
coordinates[i].x = wgs84.x;
coordinates[i].y = wgs84.y;
coordinates[i].z = wgs84.z;
}
}
/**
* Converts a geometry from WGS to ENU.
*
* Note that the geometry is internally converted. Use clone() if you need a copy.
*
* @param geometryWgs
*/
public void convertGeometryFromWgsToEnu( Geometry geometryWgs ) {
Coordinate[] coordinates = geometryWgs.getCoordinates();
for( int i = 0; i < coordinates.length; i++ ) {
Coordinate enu = wgs84ToEnu(coordinates[i]);
coordinates[i].x = enu.x;
coordinates[i].y = enu.y;
coordinates[i].z = enu.z;
}
}
/**
* Converts an envelope from WGS to ENU.
*
* @param envelopeWgs
* @return the converted envelope.
*/
public Envelope convertEnvelopeFromWgsToEnu( Envelope envelopeWgs ) {
Polygon polygonEnu = GeometryUtilities.createPolygonFromEnvelope(envelopeWgs);
convertGeometryFromWgsToEnu(polygonEnu);
Envelope envelopeEnu = polygonEnu.getEnvelopeInternal();
return envelopeEnu;
}
private void checkZ( Coordinate coordinate ) {
if (Double.isNaN(coordinate.z)) {
coordinate.z = 0.0;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy