// License: GPL. Copyright 2007 by Immanuel Scholz and others package org.openstreetmap.josm.data.coor; import static org.openstreetmap.josm.tools.I18n.trc; import static java.lang.Math.PI; import static java.lang.Math.asin; import static java.lang.Math.cos; import static java.lang.Math.sin; import static java.lang.Math.sqrt; import static java.lang.Math.toRadians; import org.openstreetmap.josm.Main; import org.openstreetmap.josm.data.Bounds; /** * GWT * * TODO: * several methods not implemented * latToString, lonToString: very rudimentary support * hashCode not implemented * * note * adapted equals * added no-arg constructor (required for RPC) */ /** * LatLon are unprojected latitude / longitude coordinates. * * This class is immutable. * * @author Imi */ public class LatLon extends Coordinate { /** * Minimum difference in location to not be represented as the same position. * The API returns 7 decimals. */ public static final double MAX_SERVER_PRECISION = 1e-7; // private static DecimalFormat cDmsMinuteFormatter = new DecimalFormat("00"); // private static DecimalFormat cDmsSecondFormatter = new DecimalFormat("00.0"); // private static DecimalFormat cDmMinuteFormatter = new DecimalFormat("00.000"); // public static DecimalFormat cDdFormatter; // static { // // Don't use the localized decimal separator. This way we can present // // a comma separated list of coordinates. // cDdFormatter = (DecimalFormat) NumberFormat.getInstance(Locale.UK); // cDdFormatter.applyPattern("###0.0000000"); // } /** * Replies true if lat is in the range [-90,90] * * @param lat the latitude * @return true if lat is in the range [-90,90] */ public static boolean isValidLat(double lat) { return lat >= -90d && lat <= 90d; } /** * Replies true if lon is in the range [-180,180] * * @param lon the longitude * @return true if lon is in the range [-180,180] */ public static boolean isValidLon(double lon) { return lon >= -180d && lon <= 180d; } // /** // * Replies the coordinate in degrees/minutes/seconds format // */ // public static String dms(double pCoordinate) { // // double tAbsCoord = Math.abs(pCoordinate); // int tDegree = (int) tAbsCoord; // double tTmpMinutes = (tAbsCoord - tDegree) * 60; // int tMinutes = (int) tTmpMinutes; // double tSeconds = (tTmpMinutes - tMinutes) * 60; // // return tDegree + "\u00B0" + cDmsMinuteFormatter.format(tMinutes) + "\'" // + cDmsSecondFormatter.format(tSeconds) + "\""; // } // // public static String dm(double pCoordinate) { // // double tAbsCoord = Math.abs(pCoordinate); // int tDegree = (int) tAbsCoord; // double tMinutes = (tAbsCoord - tDegree) * 60; // return tDegree + "\u00B0" + cDmMinuteFormatter.format(tMinutes) + "\'"; // } public LatLon(double lat, double lon) { super(lon, lat); } public LatLon(LatLon coor) { super(coor.lon(), coor.lat()); } protected LatLon() {} public double lat() { return y; } public final static String SOUTH = trc("compass", "S"); public final static String NORTH = trc("compass", "N"); public String latToString(CoordinateFormat d) { switch(d) { case DECIMAL_DEGREES: // return cDdFormatter.format(y); return y + ""; // FIXME case DEGREES_MINUTES_SECONDS: // return dms(y) + ((y < 0) ? SOUTH : NORTH); case NAUTICAL: // return dm(y) + ((y < 0) ? SOUTH : NORTH); case EAST_NORTH: // return cDdFormatter.format(Main.proj.latlon2eastNorth(this).north()); throw new UnsupportedOperationException("gwt: implement me"); default: return "ERR"; } } public double lon() { return x; } public final static String WEST = trc("compass", "W"); public final static String EAST = trc("compass", "E"); public String lonToString(CoordinateFormat d) { switch(d) { case DECIMAL_DEGREES: // return cDdFormatter.format(x); return x + ""; // FIXME case DEGREES_MINUTES_SECONDS: // return dms(x) + ((x < 0) ? WEST : EAST); case NAUTICAL: // return dm(x) + ((x < 0) ? WEST : EAST); case EAST_NORTH: // return cDdFormatter.format(Main.proj.latlon2eastNorth(this).east()); throw new UnsupportedOperationException("gwt: implement me"); default: return "ERR"; } } /** * @return <code>true</code> if the other point has almost the same lat/lon * values, only differing by no more than * 1 / {@link #MAX_SERVER_PRECISION MAX_SERVER_PRECISION}. */ public boolean equalsEpsilon(LatLon other) { double p = MAX_SERVER_PRECISION / 2; return Math.abs(lat()-other.lat()) <= p && Math.abs(lon()-other.lon()) <= p; } /** * @return <code>true</code>, if the coordinate is outside the world, compared * by using lat/lon. */ public boolean isOutSideWorld() { Bounds b = Main.proj.getWorldBoundsLatLon(); return lat() < b.getMin().lat() || lat() > b.getMax().lat() || lon() < b.getMin().lon() || lon() > b.getMax().lon(); } // /** // * @return <code>true</code> if this is within the given bounding box. // */ // public boolean isWithin(Bounds b) { // return lat() >= b.getMin().lat() && lat() <= b.getMax().lat() && lon() > b.getMin().lon() && lon() < b.getMax().lon(); // } /** * Computes the distance between this lat/lon and another point on the earth. * Uses Haversine formular. * @param other the other point. * @return distance in metres. */ public double greatCircleDistance(LatLon other) { double R = 6378135; double sinHalfLat = sin(toRadians(other.lat() - this.lat()) / 2); double sinHalfLon = sin(toRadians(other.lon() - this.lon()) / 2); double d = 2 * R * asin( sqrt(sinHalfLat*sinHalfLat + cos(toRadians(this.lat()))*cos(toRadians(other.lat()))*sinHalfLon*sinHalfLon)); // For points opposite to each other on the sphere, // rounding errors could make the argument of asin greater than 1 // (This should almost never happen.) if (java.lang.Double.isNaN(d)) { System.err.println("Error: NaN in greatCircleDistance"); d = PI * R; } return d; } /** * Returns the heading, in radians, that you have to use to get from * this lat/lon to another. * * @param other the "destination" position * @return heading */ public double heading(LatLon other) { double rv; if (other.lat() == lat()) { rv = (other.lon()>lon() ? Math.PI / 2 : Math.PI * 3 / 2); } else { rv = Math.atan((other.lon()-lon())/(other.lat()-lat())); if (rv < 0) { rv += Math.PI; } if (other.lon() < lon()) { rv += Math.PI; } } return rv; } // /** // * Returns this lat/lon pair in human-readable format. // * // * @return String in the format "lat=1.23456 deg, lon=2.34567 deg" // */ // public String toDisplayString() { // NumberFormat nf = NumberFormat.getInstance(); // nf.setMaximumFractionDigits(5); // return "lat=" + nf.format(lat()) + "\u00B0, lon=" + nf.format(lon()) + "\u00B0"; // } public LatLon interpolate(LatLon ll2, double proportion) { return new LatLon(this.lat() + proportion * (ll2.lat() - this.lat()), this.lon() + proportion * (ll2.lon() - this.lon())); } public LatLon getCenter(LatLon ll2) { return new LatLon((this.lat() + ll2.lat())/2.0, (this.lon() + ll2.lon())/2.0); } @Override public String toString() { return "LatLon[lat="+lat()+",lon="+lon()+"]"; } /** * Replies a clone of this lat LatLon, rounded to OSM precisions, i.e. to * MAX_SERVER_PRECISION * * @return a clone of this lat LatLon */ public LatLon getRoundedToOsmPrecision() { return new LatLon( Math.round(lat() / MAX_SERVER_PRECISION) * MAX_SERVER_PRECISION, Math.round(lon() / MAX_SERVER_PRECISION) * MAX_SERVER_PRECISION ); } @Override public int hashCode() { throw new UnsupportedOperationException("gwt: write me"); // final int prime = 31; // int result = super.hashCode(); // long temp; // temp = java.lang.Double.doubleToLongBits(x); // result = prime * result + (int) (temp ^ (temp >>> 32)); // temp = java.lang.Double.doubleToLongBits(y); // result = prime * result + (int) (temp ^ (temp >>> 32)); // return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (!super.equals(obj)) return false; if (getClass() != obj.getClass()) return false; Coordinate other = (Coordinate) obj; return x == other.x && y == other.y; } }