Java tutorial
/* * Copyright (C) 2014 by Array Systems Computing Inc. http://www.array.ca * * This program 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 http://www.gnu.org/licenses/ */ package org.esa.snap.engine_utilities.eo; import Jama.Matrix; import org.apache.commons.math3.util.FastMath; import org.esa.snap.core.datamodel.GeoPos; import org.esa.snap.core.gpf.OperatorException; import org.esa.snap.engine_utilities.datamodel.Orbits; import org.esa.snap.engine_utilities.datamodel.PosVector; public final class GeoUtils { private static final double EPS5 = 1e-5; private static final double EPS = 1e-10; public enum EarthModel { WGS84, GRS80 } private GeoUtils() { } /** * Convert geodetic coordinate into cartesian XYZ coordinate (WGS84 geodetic system is used). * * @param geoPos The geodetic coordinate of a given pixel. * @param xyz The xyz coordinates of the given pixel. */ public static void geo2xyz(final GeoPos geoPos, final double xyz[]) { geo2xyz(geoPos.lat, geoPos.lon, 0.0, xyz, EarthModel.WGS84); } /** * Convert geodetic coordinate into cartesian XYZ coordinate with specified geodetic system. * * @param latitude The latitude of a given pixel (in degree). * @param longitude The longitude of the given pixel (in degree). * @param altitude The altitude of the given pixel (in m) * @param xyz The xyz coordinates of the given pixel. * @param geoSystem The geodetic system. */ public static void geo2xyz(final double latitude, final double longitude, final double altitude, final double xyz[], final EarthModel geoSystem) { double a = 0.0; double e2 = 0.0; if (geoSystem == EarthModel.WGS84) { a = WGS84.a; e2 = WGS84.e2; } else if (geoSystem == EarthModel.GRS80) { a = GRS80.a; e2 = GRS80.e2; } else { throw new OperatorException("Incorrect geodetic system"); } final double lat = latitude * Constants.DTOR; final double lon = longitude * Constants.DTOR; final double sinLat = FastMath.sin(lat); final double cosLat = FastMath.cos(lat); final double N = a / Math.sqrt(1 - e2 * sinLat * sinLat); xyz[0] = (N + altitude) * cosLat * FastMath.cos(lon); // in m xyz[1] = (N + altitude) * cosLat * FastMath.sin(lon); // in m xyz[2] = ((1 - e2) * N + altitude) * sinLat; // in m } /** * Convert geodetic coordinate into cartesian XYZ coordinate with specified geodetic system. * * @param latitude The latitude of a given pixel (in degree). * @param longitude The longitude of the given pixel (in degree). * @param altitude The altitude of the given pixel (in m) * @param xyz The xyz coordinates of the given pixel. */ public static void geo2xyzWGS84(final double latitude, final double longitude, final double altitude, final PosVector xyz) { final double lat = latitude * Constants.DTOR; final double lon = longitude * Constants.DTOR; final double sinLat = FastMath.sin(lat); final double N = (WGS84.a / Math.sqrt(1.0 - WGS84.e2 * sinLat * sinLat)); final double NcosLat = (N + altitude) * FastMath.cos(lat); xyz.x = NcosLat * FastMath.cos(lon); // in m xyz.y = NcosLat * FastMath.sin(lon); // in m xyz.z = (N + altitude - WGS84.e2 * N) * sinLat; //xyz.z = (WGS84.e2inv * N + altitude) * sinLat; // in m } /** * Convert cartesian XYZ coordinate into geodetic coordinate (WGS84 geodetic system is used). * * @param xyz The xyz coordinate of the given pixel. * @param geoPos The geodetic coordinate of the given pixel. */ public static void xyz2geo(final double xyz[], final GeoPos geoPos) { xyz2geoWGS84(xyz, geoPos); } /** * Convert cartesian XYZ coordinate into geodetic coordinate with specified geodetic system. * * @param xyz The xyz coordinate of the given pixel. * @param geoPos The geodetic coordinate of the given pixel. * @param geoSystem The geodetic system. */ public static void xyz2geo(final double xyz[], final GeoPos geoPos, final EarthModel geoSystem) { double a = 0.0; double b = 0.0; double e2 = 0.0; double ep2 = 0.0; if (geoSystem == EarthModel.WGS84) { a = WGS84.a; b = WGS84.b; e2 = WGS84.e2; ep2 = WGS84.ep2; } else if (geoSystem == EarthModel.GRS80) { a = GRS80.a; b = GRS80.b; e2 = GRS80.e2; ep2 = GRS80.ep2; } else { throw new OperatorException("Incorrect geodetic system"); } final double x = xyz[0]; final double y = xyz[1]; final double z = xyz[2]; final double s = Math.sqrt(x * x + y * y); final double theta = FastMath.atan(z * a / (s * b)); geoPos.lon = (float) (FastMath.atan(y / x) * Constants.RTOD); if (geoPos.lon < 0.0 && y >= 0.0) { geoPos.lon += 180.0; } else if (geoPos.lon > 0.0 && y < 0.0) { geoPos.lon -= 180.0; } geoPos.lat = (float) (FastMath.atan((z + ep2 * b * FastMath.pow(FastMath.sin(theta), 3)) / (s - e2 * a * FastMath.pow(FastMath.cos(theta), 3))) * Constants.RTOD); } /** * Convert cartesian XYZ coordinate into geodetic coordinate with specified geodetic system. * * @param xyz The xyz coordinate of the given pixel. * @param geoPos The geodetic coordinate of the given pixel. */ public static void xyz2geoWGS84(final double xyz[], final GeoPos geoPos) { final double x = xyz[0]; final double y = xyz[1]; final double z = xyz[2]; final double s = Math.sqrt(x * x + y * y); final double theta = FastMath.atan(z * WGS84.a / (s * WGS84.b)); geoPos.lon = (float) (FastMath.atan(y / x) * Constants.RTOD); if (geoPos.lon < 0.0 && y >= 0.0) { geoPos.lon += 180.0; } else if (geoPos.lon > 0.0 && y < 0.0) { geoPos.lon -= 180.0; } geoPos.lat = (float) (FastMath.atan((z + WGS84.ep2 * WGS84.b * FastMath.pow(FastMath.sin(theta), 3)) / (s - WGS84.e2 * WGS84.a * FastMath.pow(FastMath.cos(theta), 3))) * Constants.RTOD); } /** * Convert polar coordinates to Cartesian vector. * <p> * <b>Definitions:</b> * <ul> * <li>Latitude: angle from XY-plane towards +Z-axis.</li> * <li>Longitude: angle in XY-plane measured from +X-axis towards +Y-axis.</li> * </ul> * <p> * Note: Apache's FastMath used in implementation. * * @param latitude The latitude of a given pixel (in degree). * @param longitude The longitude of the given pixel (in degree). * @param radius The radius of the given point (in m) * @param xyz The return array vector of X, Y and Z coordinates for the input point. * @author Petar Marikovic, PPO.labs */ public static void polar2cartesian(final double latitude, final double longitude, final double radius, final double xyz[]) { final double latRad = latitude * Constants.DTOR; final double lonRad = longitude * Constants.DTOR; final double sinLat = FastMath.sin(latRad); final double cosLat = FastMath.cos(latRad); xyz[0] = radius * cosLat * FastMath.cos(lonRad); xyz[1] = radius * cosLat * FastMath.sin(lonRad); xyz[2] = radius * sinLat; } /** * Convert Cartesian to Polar coordinates. * <p> * <b>Definitions:</b> * <ul> * <li>Latitude: angle from XY-plane towards +Z-axis.</li> * <li>Longitude: angle in XY-plane measured from +X-axis towards +Y-axis.</li> * </ul> * <p> * Implementation Details: Unlike for rest of utility methods GeoPos class container is not used for storing polar * coordinates. GeoPos fields are declared as floats and can introduced numerical errors, especially in radius/height. * * <p> * Note: Apache's FastMath used in implementation. * * * @param xyz Array of x, y, and z coordinates. * @param phiLamHeight Array of latitude (in radians), longitude (in radians), and radius (in meters). * @author Petar Marikovic, PPO.labs */ public static void cartesian2polar(final double[] xyz, final double[] phiLamHeight) { phiLamHeight[2] = Math.sqrt(xyz[0] * xyz[0] + xyz[1] * xyz[1] + xyz[2] * xyz[2]); phiLamHeight[1] = Math.atan2(xyz[1], xyz[0]); phiLamHeight[0] = FastMath.asin(xyz[2] / phiLamHeight[2]); } /** * Compute accurate target position for given orbit information using Newton's method. * * @param data The orbit data. * @param xyz The xyz coordinate for the target. * @param time The slant range time in seconds. */ public static void computeAccurateXYZ(final Orbits.OrbitVector data, final double[] xyz, final double time) { final double a = Constants.semiMajorAxis; final double b = Constants.semiMinorAxis; final double a2 = a * a; final double b2 = b * b; final double del = 0.001; final int maxIter = 10; Matrix X = new Matrix(3, 1); final Matrix F = new Matrix(3, 1); final Matrix J = new Matrix(3, 3); X.set(0, 0, xyz[0]); X.set(1, 0, xyz[1]); X.set(2, 0, xyz[2]); J.set(0, 0, data.xVel); J.set(0, 1, data.yVel); J.set(0, 2, data.zVel); final double time2 = FastMath.pow(time * Constants.halfLightSpeed, 2.0); for (int i = 0; i < maxIter; i++) { final double x = X.get(0, 0); final double y = X.get(1, 0); final double z = X.get(2, 0); final double dx = x - data.xPos; final double dy = y - data.yPos; final double dz = z - data.zPos; F.set(0, 0, data.xVel * dx + data.yVel * dy + data.zVel * dz); F.set(1, 0, dx * dx + dy * dy + dz * dz - time2); F.set(2, 0, x * x / a2 + y * y / a2 + z * z / b2 - 1); J.set(1, 0, 2.0 * dx); J.set(1, 1, 2.0 * dy); J.set(1, 2, 2.0 * dz); J.set(2, 0, 2.0 * x / a2); J.set(2, 1, 2.0 * y / a2); J.set(2, 2, 2.0 * z / b2); X = X.minus(J.inverse().times(F)); if (Math.abs(F.get(0, 0)) <= del && Math.abs(F.get(1, 0)) <= del && Math.abs(F.get(2, 0)) <= del) { break; } } xyz[0] = X.get(0, 0); xyz[1] = X.get(1, 0); xyz[2] = X.get(2, 0); } /** * // Given starting point GLON1,GLAT1, head1 = initial heading,and distance * // in meters, calculate destination GLON2,GLAT2, and head2=initial heading * // from destination to starting point * <p> * // Input: * // lon1: longitude * // lat1: latitude * // dist: distance in m * // head1: azimuth in degree measured in the diretion North east south west * <p> * // Output: * // GLON2: longitude * // GLAT2: latitude * // head2: azimuth in degree measured in the direction North east south west * // from (GLON2,GLAT2) to (GLON1, GLAT1) * * @param lon1 * @param lat1 * @param dist * @param head1 * @return */ public static LatLonHeading vincenty_direct(double lon1, double lat1, final double dist, final double head1) { final LatLonHeading pos = new LatLonHeading(); lat1 *= Constants.DTOR; lon1 *= Constants.DTOR; final double FAZ = head1 * Constants.DTOR; // Model WGS84: // F=1/298.25722210; // flatteing final double F = 0.0; // defF // equatorial radius final double R = 1.0 - F; double TU = R * FastMath.tan(lat1); final double SF = FastMath.sin(FAZ); final double CF = FastMath.cos(FAZ); double BAZ = 0.0; if (CF != 0.0) BAZ = Math.atan2(TU, CF) * 2.0; final double CU = 1.0 / Math.sqrt(TU * TU + 1.0); final double SU = TU * CU; final double SA = CU * SF; final double C2A = -SA * SA + 1.0; double X = Math.sqrt((1.0 / R / R - 1.0) * C2A + 1.0) + 1.0; X = (X - 2.0) / X; double C = 1.0 - X; C = (X * X / 4.0 + 1) / C; double D = (0.375 * X * X - 1.0) * X; TU = dist / R / WGS84.a / C; double Y = TU; double SY, CY, CZ, E; do { SY = FastMath.sin(Y); CY = FastMath.cos(Y); CZ = FastMath.cos(BAZ + Y); E = CZ * CZ * 2.0 - 1.0; C = Y; X = E * CY; Y = E + E - 1.0; Y = (((SY * SY * 4.0 - 3.0) * Y * CZ * D / 6.0 + X) * D / 4.0 - CZ) * SY * D + TU; } while (Math.abs(Y - C) > EPS); BAZ = CU * CY * CF - SU * SY; C = R * Math.sqrt(SA * SA + BAZ * BAZ); D = SU * CY + CU * SY * CF; pos.lat = Math.atan2(D, C); C = CU * CY - SU * SY * CF; X = Math.atan2(SY * SF, C); C = ((-3.0 * C2A + 4.0) * F + 4.0) * C2A * F / 16.0; D = ((E * CY * C + CZ) * SY * C + Y) * SA; pos.lon = lon1 + X - (1.0 - C) * D * F; BAZ = Math.atan2(SA, BAZ) + Constants.PI; pos.lon *= Constants.RTOD; pos.lat *= Constants.RTOD; pos.heading = BAZ * Constants.RTOD; while (pos.heading < 0) pos.heading += 360; return pos; } /** * // Given starting (GLON1,GLAT1) and end points (GLON2,GLAT2) * // calculate distance in meters and initial headings from start to * // end (return variable head1), * // and from end to start point (return variable head2) * <p> * // Input: * // lon1: longitude * // lat1: latitude * // lon2: longitude * // lat2: latitude * <p> * // Output: * // dist: distance in m * // head1: azimuth in degrees mesured in the direction North east south west * // from (lon1,lat1) to (lon2, lat2) * // head2: azimuth in degrees mesured in the direction North east south west * // from (lon2,lat2) to (lon1, lat1) * * @param lon1 * @param lat1 * @param lon2 * @param lat2 * @return */ public static DistanceHeading vincenty_inverse(double lon1, double lat1, double lon2, double lat2) { final DistanceHeading output = new DistanceHeading(); if ((Math.abs(lon1 - lon2) < EPS5) && (Math.abs(lat1 - lat2) < EPS5)) { output.distance = 0; output.heading1 = -1; output.heading2 = -1; return output; } lat1 *= Constants.DTOR; lat2 *= Constants.DTOR; lon1 *= Constants.DTOR; lon2 *= Constants.DTOR; // Model WGS84: // F=1/298.25722210; // flattening final double F = 0.0; //defF; final double R = 1 - F; double TU1 = R * FastMath.tan(lat1); double TU2 = R * FastMath.tan(lat2); final double CU1 = 1.0 / Math.sqrt(TU1 * TU1 + 1.0); final double SU1 = CU1 * TU1; final double CU2 = 1.0 / Math.sqrt(TU2 * TU2 + 1.0); double S = CU1 * CU2; double BAZ = S * TU2; double FAZ = BAZ * TU1; double X = lon2 - lon1; double SX, CX, SY, CY, Y, SA, C2A, CZ, E, C, D; do { SX = FastMath.sin(X); CX = FastMath.cos(X); TU1 = CU2 * SX; TU2 = BAZ - SU1 * CU2 * CX; SY = Math.sqrt(TU1 * TU1 + TU2 * TU2); CY = S * CX + FAZ; Y = Math.atan2(SY, CY); SA = S * SX / SY; C2A = -SA * SA + 1.; CZ = FAZ + FAZ; if (C2A > 0.) CZ = -CZ / C2A + CY; E = CZ * CZ * 2. - 1.; C = ((-3. * C2A + 4.) * F + 4.) * C2A * F / 16.; D = X; X = ((E * CY * C + CZ) * SY * C + Y) * SA; X = (1. - C) * X * F + lon2 - lon1; } while (Math.abs(D - X) > (0.01)); FAZ = Math.atan2(TU1, TU2); BAZ = Math.atan2(CU1 * SX, BAZ * CX - SU1 * CU2) + Constants.PI; X = Math.sqrt((1. / R / R - 1.) * C2A + 1.) + 1.; X = (X - 2.) / X; C = 1. - X; C = (X * X / 4. + 1.) / C; D = (0.375 * X * X - 1.) * X; X = E * CY; S = 1. - E - E; S = ((((SY * SY * 4. - 3.) * S * CZ * D / 6. - X) * D / 4. + CZ) * SY * D + Y) * C * WGS84.a * R; output.distance = S; output.heading1 = FAZ * Constants.RTOD; output.heading2 = BAZ * Constants.RTOD; while (output.heading1 < 0) output.heading1 += 360; while (output.heading2 < 0) output.heading2 += 360; return output; } public static class LatLonHeading { public double lat; public double lon; public double heading; } public static class DistanceHeading { public double distance; public double heading1; public double heading2; } public static interface WGS84 { public static final double a = 6378137.0; // m public static final double b = 6356752.3142451794975639665996337; //6356752.31424518; // m public static final double earthFlatCoef = 1.0 / ((a - b) / a); //298.257223563; public static final double e2 = 2.0 / earthFlatCoef - 1.0 / (earthFlatCoef * earthFlatCoef); public static final double e2inv = 1 - WGS84.e2; public static final double ep2 = e2 / (1 - e2); } public static interface GRS80 { public static final double a = 6378137; // m public static final double b = 6356752.314140; // m public static final double earthFlatCoef = 1.0 / ((a - b) / a); //298.257222101; public static final double e2 = 2.0 / earthFlatCoef - 1.0 / (earthFlatCoef * earthFlatCoef); public static final double ep2 = e2 / (1 - e2); } }