Subversion Repositories mkgmap

Rev

Rev 4407 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (C) 2006 Steve Ratcliffe
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License version 2 as
 *  published by the Free Software Foundation.
 *
 *  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.
 *
 *
 * Author: Steve Ratcliffe
 * Create date: 11-Dec-2006
 */

package uk.me.parabola.imgfmt.app;

import java.util.ArrayList;
import java.util.List;
import java.util.Locale;

import uk.me.parabola.imgfmt.Utils;
import uk.me.parabola.mkgmap.filters.ShapeMergeFilter;

/**
 * A point coordinate in unshifted map-units.
 * A map unit is 360/2^24 degrees.  In some places <i>shifted</i> coordinates
 * are used, which means that they are divided by some power of two to save
 * space in the file.
 *
 * You can create one of these with lat/long by calling the constructor with
 * double args.
 *
 * See also http://www.movable-type.co.uk/scripts/latlong.html
 *
 * @author Steve Ratcliffe
 */

public class Coord implements Comparable<Coord> {
        private static final short ON_BOUNDARY_MASK = 0x0001; // bit in flags is true if point lies on a boundary
        private static final short PRESERVED_MASK = 0x0002; // bit in flags is true if point should not be filtered out
        private static final short REPLACED_MASK = 0x0004;  // bit in flags is true if point was replaced
        private static final short ADDED_BY_CLIPPER_MASK = 0x0008; // bit in flags is true if point was added by clipper
        private static final short FIXME_NODE_MASK = 0x0010; // bit in flags is true if a node with this coords has a fixme tag
        private static final short REMOVE_MASK = 0x0020; // bit in flags is true if this point should be removed
        private static final short VIA_NODE_MASK = 0x0040; // bit in flags is true if a node with this coords is the via node of a RestrictionRelation
       
        private static final short PART_OF_BAD_ANGLE = 0x0080; // bit in flags is true if point should be treated as a node
        private static final short PART_OF_SHAPE2 = 0x0100; // use only in ShapeMerger
        private static final short END_OF_WAY = 0x0200; // use only in WrongAngleFixer
        private static final short HOUSENUMBER_NODE = 0x0400; // start/end of house number interval
        private static final short ADDED_HOUSENUMBER_NODE = 0x0800; // node was added for house numbers
        private static final short ON_COUNTRY_BORDER = 0x1000; // node is on a country border
       
        private static final int HIGH_PREC_BITS = 30;
        public static final int DELTA_SHIFT = HIGH_PREC_BITS - 24;
        private static final int MAX_DELTA = 1 << (DELTA_SHIFT - 2); // max delta abs value that is considered okay
        private static final long FACTOR_HP = 1L << HIGH_PREC_BITS;
        private static final int HIGH_PREC_UNUSED_BITS = Integer.SIZE - HIGH_PREC_BITS;
       
        public static final double R = 6378137.0; // Radius of earth at equator as defined by WGS84
        public static final double U = R * 2 * Math.PI; // circumference of earth at equator (WGS84)
        public static final double MEAN_EARTH_RADIUS = 6371000; // earth is a flattened sphere
       
        private final int latitude;
        private final int longitude;
        private byte highwayCount; // number of highways that use this point
        private short flags; // further attributes
        private final byte latDelta; // delta to high precision latitude value
        private final byte lonDelta; // delta to high precision longitude value
        private short approxDistanceToDisplayedCoord = -1;

        /**
         * Construct from co-ordinates that are already in map-units.
         * @param latitude The latitude in map units.
         * @param longitude The longitude in map units.
         */

        public Coord(int latitude, int longitude) {
                this.latitude = latitude;
                this.longitude = longitude;
                latDelta = lonDelta = 0;
        }

        /**
         * Construct from regular latitude and longitude.
         * @param latitude The latitude in degrees.
         * @param longitude The longitude in degrees.
         */

        public Coord(double latitude, double longitude) {
                this.latitude = Utils.toMapUnit(latitude);
                this.longitude = Utils.toMapUnit(longitude);
                int latHighPrec = toHighPrec(latitude);
                int lonHighPrec = toHighPrec(longitude);
                this.latDelta = (byte) ((this.latitude << DELTA_SHIFT) - latHighPrec);
                this.lonDelta = (byte) ((this.longitude << DELTA_SHIFT) - lonHighPrec);

                // verify math
                assert (this.latitude << DELTA_SHIFT) - (int) latDelta == latHighPrec;
                assert (this.longitude << DELTA_SHIFT) - (int) lonDelta == lonHighPrec;
        }
       
        private Coord (int lat, int lon, byte latDelta, byte lonDelta){
                this.latitude = lat;
                this.longitude = lon;
                this.latDelta = latDelta;
                this.lonDelta = lonDelta;
        }
       
        /**
         * Constructor for high precision values.
         * @param latHighPrec latitude in high precision
         * @param lonHighPrec longitude in high precision
         * @return Coord instance
         */

        public static Coord makeHighPrecCoord(int latHighPrec, int lonHighPrec){
                int lat24 = (latHighPrec + (1 << (DELTA_SHIFT - 1))) >> DELTA_SHIFT;
                int lon24 = (lonHighPrec + (1 << (DELTA_SHIFT - 1))) >> DELTA_SHIFT;
                byte dLat = (byte) ((lat24 << DELTA_SHIFT) - latHighPrec);
                byte dLon = (byte) ((lon24 << DELTA_SHIFT) - lonHighPrec);
                return new Coord(lat24, lon24, dLat, dLon);
        }
       
        /**
         * Construct from other coord instance, copies
         * the lat/lon values in high precision
         * @param other
         */

        public Coord(Coord other) {
                this.latitude = other.latitude;
                this.longitude = other.longitude;
                this.latDelta = other.latDelta;
                this.lonDelta = other.lonDelta;
                this.approxDistanceToDisplayedCoord = other.approxDistanceToDisplayedCoord;
        }

        public int getLatitude() {
                return latitude;
        }

        public int getLongitude() {
                return longitude;
        }

        /**
         * @return the route node id
         */

        public int getId() {
                return 0;
        }

        public int getHighwayCount() {
                return highwayCount;
        }

        /**
         * Increase the counter how many highways use this coord.
         */

        public void incHighwayCount() {
                // don't let it wrap
                if(highwayCount < Byte.MAX_VALUE)
                        ++highwayCount;
        }

        /**
         * Decrease the counter how many highways use this coord.
         */

        public void decHighwayCount() {
                // don't let it wrap
                if(highwayCount > 0)
                        --highwayCount;
        }
       
        /**
         * Resets the highway counter to 0.
         */

        public void resetHighwayCount() {
                highwayCount = 0;
        }
       
        public boolean getOnBoundary() {
                return (flags & ON_BOUNDARY_MASK) != 0;
        }

        public void setOnBoundary(boolean onBoundary) {
                if (onBoundary)
                        this.flags |= ON_BOUNDARY_MASK;
                else
                        this.flags &= ~ON_BOUNDARY_MASK;
        }

        public boolean preserved() {
                return (flags & PRESERVED_MASK) != 0 || (flags & HOUSENUMBER_NODE) != 0;
        }

        public void preserved(boolean preserved) {
                if (preserved)
                        this.flags |= PRESERVED_MASK;
                else
                        this.flags &= ~PRESERVED_MASK;
        }

        /**
         * Returns if this coord was marked to be replaced in short arc removal.
         * @return True means the replacement has to be looked up.
         */

        public boolean isReplaced() {
                return (flags & REPLACED_MASK) != 0;
        }

        /**
         * Mark a point as replaced in short arc removal process.
         * @param replaced true or false
         */

        public void setReplaced(boolean replaced) {
                if (replaced)
                        this.flags |= REPLACED_MASK;
                else
                        this.flags &= ~REPLACED_MASK;
        }

        /**
         * @return true if this coord was added by a clipper
         */

        public boolean isAddedByClipper() {
                return (flags & ADDED_BY_CLIPPER_MASK) != 0;
        }

        /**
         * Mark the Coord to be treated like a Node in short arc removal
         * @param treatAsNode true or false
         */

        public void setAddedByClipper(boolean b) {
                if (b)
                        this.flags |= ADDED_BY_CLIPPER_MASK;
                else
                        this.flags &= ~ADDED_BY_CLIPPER_MASK;
        }
       
        /**
         * Does this coordinate belong to a node with a fixme tag?
         * Note that the value is set after evaluating the points style.
         * @return true if the fixme flag is set, else false
         */

        public boolean isFixme() {
                return (flags & FIXME_NODE_MASK) != 0;
        }
       
        public void setFixme(boolean b) {
                if (b)
                        this.flags |= FIXME_NODE_MASK;
                else
                        this.flags &= ~FIXME_NODE_MASK;
        }
       
        public boolean isToRemove() {
                return (flags & REMOVE_MASK) != 0;
        }
       
        public void setRemove(boolean b) {
                if (b)
                        this.flags |= REMOVE_MASK;
                else
                        this.flags &= ~REMOVE_MASK;
        }
       
        /**
         * @return true if this coordinate belong to a via node of a restriction relation
         */

        public boolean isViaNodeOfRestriction() {
                return (flags & VIA_NODE_MASK) != 0;
        }

        /**
         * @param b true: Mark the coordinate as via node of a restriction relation
         */

        public void setViaNodeOfRestriction(boolean b) {
                if (b)
                        this.flags |= VIA_NODE_MASK;
                else
                        this.flags &= ~VIA_NODE_MASK;
        }
       
        /**
         * Should this Coord be treated by the removeWrongAngle method=
         * The value has no meaning outside of StyledConverter.
         * @return true if this coord is part of a line that has a big bearing error.
         */

        public boolean isPartOfBadAngle() {
                return (flags & PART_OF_BAD_ANGLE) != 0;
        }

        /**
         * Mark the Coord to be part of a line which has a big bearing
         * error because of the rounding to map units.
         * @param b true or false
         */

        public void setPartOfBadAngle(boolean b) {
                if (b)
                        this.flags |= PART_OF_BAD_ANGLE;
                else
                        this.flags &= ~PART_OF_BAD_ANGLE;
        }

        /**
         * Get flag for {@link ShapeMergeFilter}
         * The value has no meaning outside of {@link ShapeMergeFilter}
         * @return flag value
         */

        public boolean isPartOfShape2() {
                return (flags & PART_OF_SHAPE2) != 0;
        }

        /**
         * Set or unset flag for {@link ShapeMergeFilter}
         * @param b true or false
         */

        public void setPartOfShape2(boolean b) {
                if (b)
                        this.flags |= PART_OF_SHAPE2;
                else
                        this.flags &= ~PART_OF_SHAPE2;
        }

        /**
         * Get flag for {@link WrongAngleFixer}
         * The value has no meaning outside of {@link WrongAngleFixer}
         * @return flag value
         */

        public boolean isEndOfWay() {
                return (flags & END_OF_WAY) != 0;
        }

        /**
         * Set or unset flag for {@link WrongAngleFixer}
         * @param b true or false
         */

        public void setEndOfWay(boolean b) {
                if (b)
                        this.flags |= END_OF_WAY;
                else
                        this.flags &= ~END_OF_WAY;
        }

        /**
         * @return true if this is the beginning/end of a house number interval
         */

        public boolean isNumberNode(){
                return (flags & HOUSENUMBER_NODE) != 0;
        }
       
        /**
         * @param b true or false
         */

        public void setNumberNode(boolean b) {
                if (b)
                        this.flags |= HOUSENUMBER_NODE;
                else
                        this.flags &= ~HOUSENUMBER_NODE;
        }
       
        /**
         * @return true if this was added by the housenumber processing
         */

        public boolean isAddedNumberNode(){
                return (flags & ADDED_HOUSENUMBER_NODE) != 0;
        }
       
        /**
         * @param b true or false
         */

        public void setAddedNumberNode(boolean b) {
                if (b)
                        this.flags |= ADDED_HOUSENUMBER_NODE;
                else
                        this.flags &= ~ADDED_HOUSENUMBER_NODE;
        }
       
        /**
         * @return true if this was marked as place on country border.
         */

        public boolean getOnCountryBorder() {
                return (flags & ON_COUNTRY_BORDER) != 0;
        }

        /**
         * Mark as place on a country border
         * @param onCountryBorder
         */

        public void setOnCountryBorder(boolean onCountryBorder) {
                if (onCountryBorder)
                        this.flags |= ON_COUNTRY_BORDER;
                else
                        this.flags &= ~ON_COUNTRY_BORDER;
        }
       
        public int hashCode() {
                // Use a factor for latitude to span over the whole integer range:
                // max lat: 4194304
                // max lon: 8388608
                // max hashCode: 2118123520 < 2147483647 (Integer.MAX_VALUE)
                return 503 * latitude + longitude;
        }

        /**
         * Compares the coordinates that are displayed in the map
         */

        public boolean equals(Object obj) {
                if (!(obj instanceof Coord))
                        return false;
                Coord other = (Coord) obj;
                return latitude == other.latitude && longitude == other.longitude;
        }
       
        /**
         * Compares the coordinates using the delta values.
         * XXX: Note that
         * p1.highPrecEquals(p2) is not always equal to p1.equals(p2)
         * @param other
         * @return
         */

        public boolean highPrecEquals(Coord other) {
                if (other == null)
                        return false;
                if (this == other)
                        return true;
                return getHighPrecLat() == other.getHighPrecLat() && getHighPrecLon() == other.getHighPrecLon();
        }

        /**
         * Distance to other point in metres, using
         * "flat earth approximation" or rhumb-line algo
         */

        public double distance(Coord other) {
                double d1 = U / 360 * Math.sqrt(distanceInDegreesSquared(other));
                if (d1 < 10000)
                        return d1; // error is below 0.01 m
                // for long distances, use more complex algorithm
                return distanceOnRhumbLine(other);
        }

        /**
         * Square of distance to other point in metres, using
         * "flat earth approximation"
         */

        public double distanceInDegreesSquared(Coord other) {
                if (this == other || highPrecEquals(other))
                        return 0;
               
                double lat1 = getLatDegrees();
                double lat2 = other.getLatDegrees();
                double long1 = getLonDegrees();
                double long2 = other.getLonDegrees();
                               
                double latDiff;
                if (lat1 < lat2)
                        latDiff = lat2 - lat1;
                else
                        latDiff = lat1 - lat2; 
                if (latDiff > 90)
                        latDiff -= 180;

                double longDiff;
                if (long1 < long2)
                        longDiff = long2 - long1;
                else
                        longDiff = long1 - long2;
                if (longDiff > 180)
                        longDiff -= 360;

                // scale longDiff by cosine of average latitude
                longDiff *= Math.cos(Math.PI / 180 * Math.abs((lat1 + lat2) / 2));

                return (latDiff * latDiff) + (longDiff * longDiff);
        }
       
        /**
         * Distance to other point in metres following a great circle path, without
         * flat earth approximation, slower but better with large
         * distances and big deltas in lat AND lon.
         * Similar to code in JOSM
         */

        public double distanceHaversine (Coord point){
                double lat1 = hpToRadians(getHighPrecLat());
                double lat2 = hpToRadians(point.getHighPrecLat());
                double lon1 = hpToRadians(getHighPrecLon());
                double lon2 = hpToRadians(point.getHighPrecLon());
                double sinMidLat = Math.sin((lat1-lat2)/2);
                double sinMidLon = Math.sin((lon1-lon2)/2);
                double dRad = 2*Math.asin(Math.sqrt(sinMidLat*sinMidLat + Math.cos(lat1)*Math.cos(lat2)*sinMidLon*sinMidLon));
                return dRad * R; // the distance
        }

        /**
         * Distance to other point in metres following the shortest rhumb line.
         */

        public double distanceOnRhumbLine(Coord point){
                double lat1 = hpToRadians(getHighPrecLat());
                double lat2 = hpToRadians(point.getHighPrecLat());
                double lon1 = hpToRadians(getHighPrecLon());
                double lon2 = hpToRadians(point.getHighPrecLon());
               
                // see http://williams.best.vwh.net/avform.htm#Rhumb

                double dLat = lat2 - lat1;
                double dLon = Math.abs(lon2 - lon1);
                // if dLon over 180ยฐ take shorter rhumb line across the anti-meridian:
                if (Math.abs(dLon) > Math.PI) dLon = dLon>0 ? -(2*Math.PI-dLon) : (2*Math.PI+dLon);

                // on Mercator projection, longitude distances shrink by latitude; q is the 'stretch factor'
                // q becomes ill-conditioned along E-W line (0/0); use empirical tolerance to avoid it
                double deltaPhi = Math.log(Math.tan(lat2/2+Math.PI/4)/Math.tan(lat1/2+Math.PI/4));
                double q = Math.abs(deltaPhi) > 10e-12 ? dLat/deltaPhi : Math.cos(lat1);

                // distance is pythagoras on 'stretched' Mercator projection
                double distRad = Math.sqrt(dLat*dLat + q*q*dLon*dLon); // angular distance in radians
                return distRad * R; // the distance
        }

        /**
         * Distance to other point in high precision squared units
         */

        public long distanceInHighPrecSquared(Coord other) {
                int dLatHp = other.getHighPrecLat() - getHighPrecLat();
                int dLonHp = other.getHighPrecLon() - getHighPrecLon();
                dLonHp = (dLonHp << HIGH_PREC_UNUSED_BITS) >> HIGH_PREC_UNUSED_BITS; // fix wrap-around earth
                return (long)dLatHp * dLatHp + (long)dLonHp * dLonHp;
        }

        /**
         * Calculate point on the line this->other. If d is the distance between this and other,
         * the point is {@code fraction * d} metres from this.
         * For small distances between this and other we use a flat earth approximation,
         * for large distances this could result in errors of many metres, so we use
         * the rhumb line calculations.
         */

        public Coord makeBetweenPoint(Coord other, double fraction) {
                int dlatHp = other.getHighPrecLat() - getHighPrecLat();
                int dlonHp = other.getHighPrecLon() - getHighPrecLon();
                if (dlonHp == 0 || Math.abs(dlatHp) < 1000000 && Math.abs(dlonHp) < 1000000) {
                        // distances are rather small, we can use flat earth approximation
                        int latHighPrec = (int)Math.round(getHighPrecLat() + dlatHp * fraction);
                        int lonHighPrec = (int)Math.round(getHighPrecLon() + dlonHp * fraction);
                        return makeHighPrecCoord(latHighPrec, lonHighPrec);
                }
                double brng = this.bearingToOnRhumbLine(other, true);
                double dist = this.distance(other) * fraction;
                return this.destOnRhumbLine(dist, brng);
        }

       
        /**
         * returns bearing (in degrees) from current point to another point
         * following a rhumb line
         */

        public double bearingTo(Coord point) {
                return bearingToOnRhumbLine(point, false);
        }

        /**
         * returns bearing (in degrees) from current point to another point
         * following a great circle path
         * @param point the other point
         * @param needHighPrec set to true if you need a very high precision
         */

        public double bearingToOnGreatCircle(Coord point, boolean needHighPrec) {
                // use high precision values for this
                double lat1 = hpToRadians(getHighPrecLat());
                double lat2 = hpToRadians(point.getHighPrecLat());
                double lon1 = hpToRadians(getHighPrecLon());
                double lon2 = hpToRadians(point.getHighPrecLon());

                double dlon = lon2 - lon1;

                double y = Math.sin(dlon) * Math.cos(lat2);
                double x = Math.cos(lat1)*Math.sin(lat2) -
                                Math.sin(lat1)*Math.cos(lat2)*Math.cos(dlon);
                double brngRad = needHighPrec ? Math.atan2(y, x) : Utils.atan2_approximation(y, x);
                return brngRad * 180 / Math.PI;
        }

        /**
         * returns bearing (in degrees) from current point to another point
         * following shortest rhumb line
         * @param point the other point
         * @param needHighPrec set to true if you need a very high precision
         */

        public double bearingToOnRhumbLine(Coord point, boolean needHighPrec){
                double lat1 = hpToRadians(this.getHighPrecLat());
                double lat2 = hpToRadians(point.getHighPrecLat());
                double lon1 = hpToRadians(this.getHighPrecLon());
                double lon2 = hpToRadians(point.getHighPrecLon());

                double dLon = lon2-lon1;
                // if dLon over 180ยฐ take shorter rhumb line across the anti-meridian:
                if (Math.abs(dLon) > Math.PI) dLon = dLon>0 ? -(2*Math.PI-dLon) : (2*Math.PI+dLon);

                double deltaPhi = Math.log(Math.tan(lat2/2+Math.PI/4)/Math.tan(lat1/2+Math.PI/4));

                double brngRad = needHighPrec ? Math.atan2(dLon, deltaPhi) : Utils.atan2_approximation(dLon, deltaPhi);
                return brngRad * 180 / Math.PI;
        }

       
        /**
         * Sort lexicographically by longitude, then latitude.
         *
         * This ordering is used for sorting entries in NOD3.
         */

        public int compareTo(Coord other) {
                if (longitude == other.getLongitude()) {
                        if (latitude == other.getLatitude())
                                return 0;
                        return latitude > other.getLatitude() ? 1 : -1;
                }
                return longitude > other.getLongitude() ? 1 : -1;
        }                      

        /**
         * Returns a string representation of the object.
         *
         * @return a string representation of the object.
         */

        public String toString() {
                return (latitude) + "/" + (longitude);
        }

        public String toDegreeString() {
                return String.format(Locale.ENGLISH, "%.6f,%.6f",
                        getLatDegrees(),
                        getLonDegrees());
        }

        protected String toOSMURL(int zoom) {
                return ("http://www.openstreetmap.org/?mlat=" +
                                String.format(Locale.ENGLISH, "%.6f", getLatDegrees()) +
                                "&mlon=" +
                                String.format(Locale.ENGLISH, "%.6f", getLonDegrees()) +
                                "&zoom=" +
                                zoom);
        }

        public String toOSMURL() {
                return toOSMURL(17);
        }

        /**
         * Convert latitude or longitude to HIGH_PREC_BITS bits value.
         * This allows higher precision than the 24 bits
         * used in map units.
         * @param degrees The latitude or longitude as decimal degrees.
         * @return An integer value with {@code HIGH_PREC_BITS} bit precision.
         */

        private static int toHighPrec(double degrees) {
                final double DELTA = 360.0D / FACTOR_HP / 2; // Correct rounding
                double v = (degrees > 0) ? degrees + DELTA : degrees - DELTA;
                return (int) (v * FACTOR_HP / 360);
        }

        /* Factor for conversion to radians using HIGH_PREC_BITS bits
         * (Math.PI / 180) * (360.0 / (1 << HIGH_PREC_BITS))
         */

        static final double HIGH_PREC_RAD_FACTOR = 2 * Math.PI / FACTOR_HP;
       
        /**
         * Convert to radians using high precision
         * @param valHighPrec a longitude/latitude value with HIGH_PREC_BITS bit precision
         * @return an angle in radians.
         */

        public static double hpToRadians(int valHighPrec){
                return HIGH_PREC_RAD_FACTOR * valHighPrec;
        }

        /**
         * @return Latitude as signed HIGH_PREC_BITS bit integer
         */

        public int getHighPrecLat() {
                return (latitude << DELTA_SHIFT) - (int) latDelta;
        }

        /**
         * @return Longitude as signed HIGH_PREC_BITS bit integer
         */

        public int getHighPrecLon() {
                return (longitude << DELTA_SHIFT) - (int) lonDelta;
        }
       
        /**
         * @return latitude in degrees with highest avail. precision
         */

        public double getLatDegrees(){
                return (360.0D / FACTOR_HP) * getHighPrecLat();
        }
       
        /**
         * @return longitude in degrees with highest avail. precision
         */

        public double getLonDegrees(){
                return (360.0D / FACTOR_HP) * getHighPrecLon();
        }
       
        public Coord getDisplayedCoord(){
                return new Coord(latitude,longitude);
        }

        /**
         * Check if the rounding to 24 bit resolution caused large error. If so, the point may be placed
         * at an alternative position.
         * @return true if rounding error is large.
         */

        public boolean hasAlternativePos(){
                if (getOnBoundary() || getOnCountryBorder())
                        return false;
                return (Math.abs(latDelta) > MAX_DELTA || Math.abs(lonDelta) > MAX_DELTA);
        }
        /**
         * Calculate up to three points with equal
         * high precision coordinate, but
         * different map unit coordinates.
         * @return a list of Coord instances, is empty if alternative positions are too far
         */

        public List<Coord> getAlternativePositions(){
                ArrayList<Coord> list = new ArrayList<>();
                if (getOnBoundary() || getOnCountryBorder())
                        return list;
                int modLatDelta = 0;
                int modLonDelta = 0;
               
                int modLat = latitude;
                int modLon = longitude;
                if (latDelta > MAX_DELTA)
                        modLat--;
                else if (latDelta < -MAX_DELTA)
                        modLat++;
                if (lonDelta > MAX_DELTA)
                        modLon--;
                else if (lonDelta < -MAX_DELTA)
                        modLon++;
                int latHighPrec = getHighPrecLat();
                int lonHighPrec = getHighPrecLon();
                modLatDelta = (modLat << DELTA_SHIFT) - latHighPrec;
                modLonDelta = (modLon << DELTA_SHIFT) - lonHighPrec;
                assert modLatDelta >= Byte.MIN_VALUE && modLatDelta <= Byte.MAX_VALUE;
                assert modLonDelta >= Byte.MIN_VALUE && modLonDelta <= Byte.MAX_VALUE;
                if (modLat != latitude){
                        if (modLon != longitude)
                                list.add(new Coord(modLat, modLon, (byte)modLatDelta, (byte)modLonDelta));
                        list.add(new Coord(modLat, longitude, (byte)modLatDelta, lonDelta));
                }
                if (modLon != longitude)
                        list.add(new Coord(latitude, modLon, latDelta, (byte)modLonDelta));
                /* verify math
                for(Coord co:list){
                        double d = distance(new Coord (co.getLatitude(),co.getLongitude()));
                        assert d < 3.0;
                }
                */

                return list;
        }
       
        /**
         * @return approximate distance in cm
         */

        public short getDistToDisplayedPoint() {
                if (approxDistanceToDisplayedCoord < 0) {
                        approxDistanceToDisplayedCoord = (short) Math.round(getDisplayedCoord().distance(this) * 100);
                }
                return approxDistanceToDisplayedCoord;
        }
       
        /**
         * Get the coord that is {@code dist} metre away travelling with course
         * {@code brng} on a rhumb-line.
         * @param dist distance in m
         * @param brng bearing in degrees
         * @return a new Coord instance
         */

        public Coord destOnRhumbLine(double dist, double brng){
                double distRad = dist / R; // angular distance in radians
                double lat1 = hpToRadians(this.getHighPrecLat());
                double lon1 = hpToRadians(this.getHighPrecLon());

                double brngRad = Math.toRadians(brng);

                double deltaLat = distRad * Math.cos(brngRad);

                double lat2 = lat1 + deltaLat;
                // check for some daft bugger going past the pole, normalise latitude if so
                if (Math.abs(lat2) > Math.PI/2) lat2 = lat2>0 ? Math.PI-lat2 : -Math.PI-lat2;
                double lon2;
                // catch special case: normalised value would be -8388608
                if (this.getLongitude() == 8388608 && brng == 0)
                        lon2 = lon1;
                else {
                        double deltaPhi = Math.log(Math.tan(lat2/2+Math.PI/4)/Math.tan(lat1/2+Math.PI/4));
                        double q = Math.abs(deltaPhi) > 10e-12 ? deltaLat / deltaPhi : Math.cos(lat1); // E-W course becomes ill-conditioned with 0/0

                        double deltaLon = distRad*Math.sin(brngRad)/q;

                        lon2 = lon1 + deltaLon;

                        lon2 = (lon2 + 3*Math.PI) % (2*Math.PI) - Math.PI; // normalise to -180..+180ยบ
                }

                return new Coord(Math.toDegrees(lat2), Math.toDegrees(lon2));
        }
       
        /**
         * Calculate the distance in metres to the rhumb line
         * defined by coords a and b.
         * @param a start point
         * @param b end point
         * @return perpendicular distance in m.
         */

        public double distToLineSegment(Coord a, Coord b){
                double ap = a.distance(this);
                double ab = a.distance(b);
                double bp = b.distance(this);
                if (ap == 0 || bp == 0)
                        return 0;
                double abpa = (ab+ap+bp)/2;
                double dx = abpa-ab;
                double dist;
                if (dx < 0){
                        // simple calculation using Herons formula will fail
                        // calculate x, the point on line a-b which is as far away from a as this point
                        double b_ab = a.bearingToOnRhumbLine(b, true);
                        Coord x = a.destOnRhumbLine(ap, b_ab);
                        // this dist between these two points is not exactly
                        // the perpendicul distance, but close enough
                        dist = x.distance(this);
                }
                else
                        dist = 2 * Math.sqrt(abpa * (abpa-ab) * (abpa-ap) * (abpa-bp)) / ab;
                return dist;
        }

        /**
         * Calculate distance to rhumb line segment a-b.
         * @param a point a
         * @param b point b
         * @return distance in m
         */

        public double shortestDistToLineSegment(Coord a, Coord b){
                int aLon = a.getHighPrecLon();
                int bLon = b.getHighPrecLon();
                int pLon = this.getHighPrecLon();
                int aLat = a.getHighPrecLat();
                int bLat = b.getHighPrecLat();
                int pLat = this.getHighPrecLat();
               
                double deltaLon = bLon - aLon;
                double deltaLat = bLat - aLat;

                double frac;
                if (deltaLon == 0 && deltaLat == 0){
                        frac = 0;
                }
                else {
                        // scale for longitude deltas by cosine of average latitude
                        double scale = Math.cos(Coord.hpToRadians((aLat + bLat + pLat) / 3) );
                        double deltaLonAP = scale * (pLon - aLon);
                        deltaLon = scale * deltaLon;
                        if (deltaLon == 0 && deltaLat == 0)
                                frac = 0;
                        else
                                frac = (deltaLonAP * deltaLon + (pLat - aLat) * deltaLat) / (deltaLon * deltaLon + deltaLat * deltaLat);
                }

                double distance;
                if (frac <= 0) {
                        distance = a.distance(this);
                } else if (frac >= 1) {
                        distance = b.distance(this);
                } else {
                        distance = this.distToLineSegment(a, b);
                }
                return distance;
        }
       
        /**
         * @return a new coordinate at the specified distance (metres) away along the specified bearing (degrees)
         * uses "Destination point given distance and bearing from start point" formula from
         * http://www.movable-type.co.uk/scripts/latlong.html
         */

        public Coord offset(double bearingInDegrees, double distanceInMetres) {
                double bearing = Math.toRadians(bearingInDegrees);
                double angularDistance = distanceInMetres / MEAN_EARTH_RADIUS;
                double lat = Math.toRadians(getLatDegrees());
                double lon = Math.toRadians(getLonDegrees());
                double newLat = Math.asin(Math.sin(lat) * Math.cos(angularDistance) + Math.cos(lat) * Math.sin(angularDistance) * Math.cos(bearing));
                double newLon = lon + Math.atan2(Math.sin(bearing) * Math.sin(angularDistance) * Math.cos(lat), Math.cos(angularDistance) - Math.sin(lat) * Math.sin(newLat));
                return new Coord(Math.toDegrees(newLat), Math.toDegrees(newLon));
        }

        /**
         * Calculate if this point lies on the left or right of the line through the given points.
         * @param p0 first point
         * @param p2 second point
         * @return positive value if on left, negative value if on the right, 0 if on the line
         */

        public long isLeft(final Coord p1, final Coord p2) {
                long p1Lat = p1.getHighPrecLat();
                long p1Lon = p1.getHighPrecLon();
                return ((long) p2.getHighPrecLon() - p1Lon) * ((long) this.getHighPrecLat() - p1Lat)
                                - ((long) p2.getHighPrecLat() - p1Lat) * ((long) this.getHighPrecLon() - p1Lon);
        }
}