Subversion Repositories mkgmap

Rev

Rev 3470 | 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 final static short ON_BOUNDARY_MASK = 0x0001; // bit in flags is true if point lies on a boundary
        private final static short PRESERVED_MASK = 0x0002; // bit in flags is true if point should not be filtered out
        private final static short REPLACED_MASK = 0x0004;  // bit in flags is true if point was replaced
        private final static short TREAT_AS_NODE_MASK = 0x0008; // bit in flags is true if point should be treated as a node
        private final static short FIXME_NODE_MASK = 0x0010; // bit in flags is true if a node with this coords has a fixme tag
        private final static short REMOVE_MASK = 0x0020; // bit in flags is true if this point should be removed
        private final static short VIA_NODE_MASK = 0x0040; // bit in flags is true if a node with this coords is the via node of a RestrictionRelation
       
        private final static short PART_OF_BAD_ANGLE = 0x0080; // bit in flags is true if point should be treated as a node
        private final static short PART_OF_SHAPE2 = 0x0100; // use only in ShapeMerger
        private final static short END_OF_WAY = 0x0200; // use only in WrongAngleFixer
        private final static short HOUSENUMBER_NODE = 0x0400; // start/end of house number interval
       
        public final static int HIGH_PREC_BITS = 30;
        public final static int DELTA_SHIFT = 6;
       
        public final static double R = 6378137.0; // Radius of earth as defined by WGS84
        public final static double U = R * 2 * Math.PI; // circumference of earth (WGS84)
       
        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 30 bit lat value
        private final byte lonDelta; // delta to 30 bit lon value
        private final static byte MAX_DELTA = 16; // max delta abs value that is considered okay
        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 lat30 = toBit30(latitude);
                int lon30 = toBit30(longitude);
                this.latDelta = (byte) ((this.latitude << 6) - lat30);
                this.lonDelta = (byte) ((this.longitude << 6) - lon30);
               
                // verify math
                assert (this.latitude << 6) - latDelta == lat30;
                assert (this.longitude << 6) - lonDelta == lon30;
        }
       
        private Coord (int lat, int lon, byte latDelta, byte lonDelta){
                this.latitude = lat;
                this.longitude = lon;
                this.latDelta = latDelta;
                this.lonDelta = lonDelta;
        }
       
        public static Coord makeHighPrecCoord(int lat30, int lon30){
                int lat24 = (lat30 + (1 << 5)) >> 6;  
                int lon24 = (lon30 + (1 << 5)) >> 6;
                byte dLat = (byte) ((lat24 << 6) - lat30);
                byte dLon = (byte) ((lon24 << 6) - lon30);
                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;
        }

        /**
         * Should this Coord be treated like a Garmin node in short arc removal?
         * The value has no meaning outside of short arc removal.
         * @return true if this coord should be treated like a Garmin node, else false
         */

        public boolean isTreatAsNode() {
                return (flags & TREAT_AS_NODE_MASK) != 0;
        }

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

        public void setTreatAsNode(boolean treatAsNode) {
                if (treatAsNode)
                        this.flags |= TREAT_AS_NODE_MASK;
                else
                        this.flags &= ~TREAT_AS_NODE_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  
         */

        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  
         */

        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 if this is the beginning/end of a house number interval
         */

        public boolean isNumberNode(){
                return (flags & HOUSENUMBER_NODE) != 0;
        }
       
        /**
         * Set or unset flag for {@link WrongAngleFixer}
         * @param b true or false
         */

        public void setNumberNode(boolean b) {
                if (b)
                        this.flags |= HOUSENUMBER_NODE;
                else
                        this.flags &= ~HOUSENUMBER_NODE;
        }
       
        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 == null || !(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 = int30ToRadians(getHighPrecLat());
                double lat2 = int30ToRadians(point.getHighPrecLat());
                double lon1 = int30ToRadians(getHighPrecLon());
                double lon2 = int30ToRadians(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));
                double distance= dRad * R;
                return distance;
        }

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

        public double distanceOnRhumbLine(Coord point){
                double lat1 = int30ToRadians(getHighPrecLat());
                double lat2 = int30ToRadians(point.getHighPrecLat());
                double lon1 = int30ToRadians(getHighPrecLon());
                double lon2 = int30ToRadians(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
            double dist = distRad * R;

            return dist;
               
        }

        /**
         * 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 dLat30 = other.getHighPrecLat() - getHighPrecLat();
                int dLon30 = other.getHighPrecLon() - getHighPrecLon();
                if (dLon30 == 0 || Math.abs(dLat30) < 1000000 && Math.abs(dLon30) < 1000000 ){
                        // distances are rather small, we can use flat earth approximation
                        int lat30 = (int) (getHighPrecLat() + dLat30 * fraction);
                        int lon30 = (int) (getHighPrecLon() + dLon30 * fraction);
                        return makeHighPrecCoord(lat30, lon30);
                }
                double brng = this.bearingToOnRhumbLine(other, true);
                double dist = this.distance(other) * fraction;
                return this.destOnRhumLine(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 = int30ToRadians(getHighPrecLat());
                double lat2 = int30ToRadians(point.getHighPrecLat());
                double lon1 = int30ToRadians(getHighPrecLon());
                double lon2 = int30ToRadians(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 = int30ToRadians(this.getHighPrecLat());
                double lat2 = int30ToRadians(point.getHighPrecLat());
                double lon1 = int30ToRadians(this.getHighPrecLon());
                double lon2 = int30ToRadians(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 30 bits value.
         * This allows higher precision than the 24 bits
         * used in map units.
         * @param l The lat or long as decimal degrees.
         * @return An integer value with 30 bit precision.
         */

        private static int toBit30(double l) {
                double DELTA = 360.0D / (1 << 30) / 2; //Correct rounding
                if (l > 0)
                        return (int) ((l + DELTA) * (1 << 30)/360);
                return (int) ((l - DELTA) * (1 << 30)/360);
               
        }

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

        final static double BIT30_RAD_FACTOR = 2 * Math.PI / (1 << 30);
       
        /**
         * Convert to radians using high precision
         * @param val30 a longitude/latitude value with 30 bit precision
         * @return an angle in radians.
         */

        public static double int30ToRadians(int val30){
                return BIT30_RAD_FACTOR * val30;
        }

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

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

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

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

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

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

        public boolean hasAlternativePos(){
                if (getOnBoundary())
                        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())
                        return list;
                byte modLatDelta = 0;
                byte 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 lat30 = getHighPrecLat();
                int lon30 = getHighPrecLon();
                modLatDelta = (byte) ((modLat<<6) - lat30);
                modLonDelta = (byte) ((modLon<<6) - lon30);
                assert modLatDelta >= -63 && modLatDelta <= 63;
                assert modLonDelta >= -63 && modLonDelta <= 63;
                if (modLat != latitude){
                        if (modLon != longitude)
                                list.add(new Coord(modLat, modLon, modLatDelta, modLonDelta));
                        list.add(new Coord(modLat, longitude, modLatDelta, lonDelta));
                }
                if (modLon != longitude)
                        list.add(new Coord(latitude, modLon, latDelta, 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 destOnRhumLine(double dist, double brng){
            double distRad = dist / R; // angular distance in radians
                double lat1 = int30ToRadians(this.getHighPrecLat());
                double lon1 = int30ToRadians(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);
                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.destOnRhumLine(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.int30ToRadians((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;
        }
       
}