Subversion Repositories mkgmap

Rev

Rev 4525 | 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: 03-Dec-2006
 */

package uk.me.parabola.imgfmt;

import java.awt.geom.Line2D;
import java.io.Closeable;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.Calendar;
import java.util.Date;
import java.util.zip.GZIPInputStream;

import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.imgfmt.app.ImgFileWriter;
/**
 * Some miscellaneous functions that are used within the .img code.
 *
 * @author Steve Ratcliffe
 */

public class Utils {
        public static final int MIN_LAT_MAP_UNITS = toMapUnit(-90);
        public static final int MAX_LAT_MAP_UNITS = toMapUnit(90);
        public static final int MIN_LON_MAP_UNITS = toMapUnit(-180);
        public static final int MAX_LON_MAP_UNITS = toMapUnit(180);

        /**
         * Routine to convert a string to bytes and pad with a character up
         * to a given length.
         * Only to be used for strings that are expressible in latin1.
         *
         * @param s The original string.
         * @param len The length to pad to.
         * @param pad The byte used to pad.
         * @return An array created from the string.
         */

        public static byte[] toBytes(String s, int len, byte pad) {
                if (s == null)
                        throw new IllegalArgumentException("null string provided");

                byte[] out = new byte[len];
                for (int i = 0; i < len; i++) {
                        if (i > s.length()) {
                                out[i] = pad;
                        } else {
                                out[i] = (byte) s.charAt(i);
                        }
                }
                return out;
        }

        public static byte[] toBytes(String s) {
                return toBytes(s, s.length(), (byte) 0);
        }

        /**
         * Convert from bytes to a string.  Only to be used when the character set
         * is ascii or latin1.
         *
         * @param buf A byte buffer to get the bytes from.  Should be ascii or latin1.
         * @param off The offset into buf.
         * @param len The length to get.
         * @return A string.
         */

        public static String bytesToString(ByteBuffer buf, int off, int len) {
                if (buf == null)
                        throw new IllegalArgumentException("null byte buffer provided");

                byte[] bbuf = new byte[len];
                buf.position(off);
                buf.get(bbuf);
                char[] cbuf = new char[len];
                for (int i = 0; i < bbuf.length; i++) {
                        cbuf[i] = (char) bbuf[i];
                }

                return new String(cbuf);
        }

        /**
         * Set the creation date.  Note that the year is encoded specially.
         *
         * @param buf The buffer to write into.  It must have been properly positioned
         * beforehand.
         * @param date The date to set.
         */

        public static void setCreationTime(ByteBuffer buf, Date date) {
                Calendar cal = Calendar.getInstance();

                if (date != null)
                        cal.setTime(date);

                fillBufFromTime(buf, cal);
        }

        /**
         * A map unit is an integer value that is 1/(2^24) degrees of latitude or
         * longitude.
         *
         * @param l The lat or long as decimal degrees.
         * @return An integer value in map units.
         */

        public static int toMapUnit(double l) {
                double DELTA = 360.0D / (1 << 24) / 2; //Correct rounding
                if (l > 0)
                        return (int) ((l + DELTA) * (1 << 24)/360);
                else
                        return (int) ((l - DELTA) * (1 << 24)/360);
        }

        /**
         * Convert a date into the in-file representation of a date.
         *
         * @param date The date.
         * @return A byte stream in .img format.
         */

        public static byte[] makeCreationTime(Date date) {
                Calendar cal = Calendar.getInstance();

                if (date != null)
                        cal.setTime(date);

                byte[] ret = new byte[7];

                ByteBuffer buf = ByteBuffer.wrap(ret);
                buf.order(ByteOrder.LITTLE_ENDIAN);
                fillBufFromTime(buf, cal);
               
                return ret;
        }

        private static void fillBufFromTime(ByteBuffer buf, Calendar cal) {
                buf.putChar((char) cal.get(Calendar.YEAR));
                buf.put((byte) (cal.get(Calendar.MONTH)+1));
                buf.put((byte) cal.get(Calendar.DAY_OF_MONTH));
                buf.put((byte) cal.get(Calendar.HOUR_OF_DAY));
                buf.put((byte) cal.get(Calendar.MINUTE));
                buf.put((byte) cal.get(Calendar.SECOND));
        }

        /**
         * Make a date from the garmin representation.
         * @param date The bytes representing the date.
         * @return A java date.
         */

        public static Date makeCreationTime(byte[] date) {
                Calendar cal = Calendar.getInstance();

                int y = ((date[1] & 0xff) << 8) + (date[0] & 0xff);
                cal.set(y, date[2]-1, date[3], date[4], date[5], date[6]);

                return cal.getTime();
        }

        /**
         * Convert an angle in map units to degrees.
         */

        public static double toDegrees(int val) {
                return (double) val * (360.0 / (1 << 24));
        }

        /**
         * Convert an angle in map units to radians.
         */

        public static double toRadians(int mapunits) {
                return toDegrees(mapunits) * (Math.PI / 180);
        }

        public static void closeFile(Closeable f) {
                if (f != null) {
                        try {
                                f.close();
                        } catch (IOException e) {
                                e.printStackTrace();
                        }
                }
        }

        /**
         * Open a file and apply filters necessary for reading it such as
         * decompression.
         *
         * @param name The file to open.
         * @return A stream that will read the file, positioned at the beginning.
         * @throws FileNotFoundException If the file cannot be opened for any reason.
         */

        public static InputStream openFile(String name) throws FileNotFoundException {
                InputStream is = new FileInputStream(name);
                if (name.endsWith(".gz")) {
                        try {
                                is = new GZIPInputStream(is);
                        } catch (IOException e) {
                                throw new FileNotFoundException( "Could not read as compressed file");
                        }
                }
                return is;
        }

        public static String joinPath(String dir, String basename, String ext) {
                return joinPath(dir, basename + "." + ext);
        }
        public static String joinPath(String dir, String basename) {
                File file = new File(dir, basename);
                return file.getAbsolutePath();
        }

        /**
         * Rounds an integer up to the nearest multiple of {@code 2^shift}.
         * Works with both positive and negative integers.
         * @param val the integer to round up.
         * @param shift the power of two to round up to.
         * @return the rounded integer.
         */

        public static int roundUp(int val, int shift) {
                return (val + (1 << shift) - 1) >>> shift << shift;
        }
       
        /**
         * Calculates the angle between the two segments (c1,c2),(c2,c3).
         * It is assumed that the segments are rhumb lines, not great circle paths.
         * @param c1 first point
         * @param c2 second point
         * @param c3 third point
         * @return angle between the two segments in degree [-180;180]
         */

        public static double getAngle(Coord c1, Coord c2, Coord c3) {
                double a = c2.bearingTo(c1);
                double b = c2.bearingTo(c3);
                double angle = b - (a - 180);
                while(angle > 180)
                        angle -= 360;
                while(angle < -180)
                        angle += 360;
               
                return angle;
        }
       
        /**
         * Calculates the angle between the two segments (c1,c2),(c2,c3)
         * using the coords in map units.
         * @param c1 first point
         * @param c2 second point
         * @param c3 third point
         * @return angle between the two segments in degree [-180;180]
         */

        public static double getDisplayedAngle(Coord c1, Coord c2, Coord c3) {
                return getAngle(c1.getDisplayedCoord(), c2.getDisplayedCoord(), c3.getDisplayedCoord());
        }

        public final static int NOT_STRAIGHT = 0;
        public final static int STRAIGHT_SPIKE = 1;
        public final static int STRICTLY_STRAIGHT = 2;
        /**
         * Checks if the two segments (c1,c2),(c2,c3) form a straight line.
         * @param c1 first point
         * @param c2 second point
         * @param c3 third point
         * @return NOT_STRAIGHT, STRAIGHT_SPIKE or STRICTLY_STRAIGHT
         */

        public static int isStraight(Coord c1, Coord c2, Coord c3) {
                if (c1.equals(c3))
                        return STRAIGHT_SPIKE;
                long area;
                // calculate the area that is enclosed by the three points
                // (as if a closing line is drawn from c3 back to c1)
                area = ((long)c1.getLongitude() * c2.getLatitude() -
                                (long)c2.getLongitude() * c1.getLatitude());
                area += ((long)c2.getLongitude() * c3.getLatitude() -
                                (long)c3.getLongitude() * c2.getLatitude());
                area += ((long)c3.getLongitude() * c1.getLatitude() -
                                (long)c1.getLongitude() * c3.getLatitude());
                if (area == 0){
                        // area is empty-> points lie on a straight line
                        int delta1 = c1.getLatitude() - c2.getLatitude();
                        int delta2 = c2.getLatitude() - c3.getLatitude();
                        if (delta1 < 0 && delta2 > 0 || delta1 > 0 && delta2 < 0)
                                return STRAIGHT_SPIKE;
                        delta1 = c1.getLongitude() - c2.getLongitude();
                        delta2 = c2.getLongitude() - c3.getLongitude();
                        if (delta1 < 0 && delta2 > 0 || delta1 > 0 && delta2 < 0)
                                return STRAIGHT_SPIKE;
                        return STRICTLY_STRAIGHT;
                }
                // line is not straight
                return NOT_STRAIGHT;
               
        }

        /**
         * Checks if the two segments (c1,c2),(c2,c3) form a straight line
         * using high precision coordinates.
         * @param c1 first point
         * @param c2 second point
         * @param c3 third point
         * @return NOT_STRAIGHT, STRAIGHT_SPIKE or STRICTLY_STRAIGHT
         */

        public static int isHighPrecStraight(Coord c1, Coord c2, Coord c3) {
                if (c1.highPrecEquals(c3))
                        return STRAIGHT_SPIKE;
                long area;
                long c1Lat = c1.getHighPrecLat();
                long c2Lat = c2.getHighPrecLat();
                long c3Lat = c3.getHighPrecLat();
                long c1Lon = c1.getHighPrecLon();
                long c2Lon = c2.getHighPrecLon();
                long c3Lon = c3.getHighPrecLon();
                // calculate the area that is enclosed by the three points
                // (as if a closing line is drawn from c3 back to c1)
                area = c1Lon * c2Lat - c2Lon * c1Lat;
                area += c2Lon * c3Lat - c3Lon * c2Lat;
                area += c3Lon * c1Lat - c1Lon * c3Lat;
                if (area == 0){
                        // area is empty-> points lie on a straight line
                        long delta1 = c1Lat - c2Lat;
                        long delta2 = c2Lat - c3Lat;
                        if (delta1 < 0 && delta2 > 0 || delta1 > 0 && delta2 < 0)
                                return STRAIGHT_SPIKE;
                        delta1 = c1Lon - c2Lon;
                        delta2 = c2Lon - c3Lon;
                        if (delta1 < 0 && delta2 > 0 || delta1 > 0 && delta2 < 0)
                                return STRAIGHT_SPIKE;
                        return STRICTLY_STRAIGHT;
                }
                // line is not straight
                return NOT_STRAIGHT;
               
        }

        /**
         * approximate atan2, much faster than Math.atan2()
         * Based on a 50-year old arctan approximation due to Hastings
         */

        private final static double PI_BY_2 = Math.PI / 2;
        // |error| < 0.005
        public static double atan2_approximation( double y, double x )
        {
                if ( x == 0.0f )
                {
                        if ( y > 0.0f ) return PI_BY_2 ;
                        if ( y == 0.0f ) return 0.0f;
                        return -PI_BY_2 ;
                }
                double atan;
                double z = y/x;
                if ( Math.abs( z ) < 1.0f )
                {
                        atan = z/(1.0f + 0.28f*z*z);
                        if ( x < 0.0f )
                        {
                                if ( y < 0.0f ) return atan - Math.PI;
                                return atan + Math.PI;
                        }
                }
                else
                {
                        atan = PI_BY_2  - z/(z*z + 0.28f);
                        if ( y < 0.0f ) return atan - Math.PI;
                }
                return atan;
        }
       
        /**
         * calculate a long value for the latitude and longitude of a coord
         * in high precision.
         * @param co
         * @return a long that can be used as a key in HashMaps
         */

        public static long coord2Long(Coord co){
                int latHp = co.getHighPrecLat();
                int lonHp = co.getHighPrecLon();
               
                return (latHp & 0xffffffffL) << 32 | (lonHp & 0xffffffffL);
        }
       
        public static int numberToPointerSize(int n) {
        // moved from imgfmt/app/mdr/MdrSection.java and app/typ/TYPFile.java
                if (n <= 0xff)
                        return 1;
                else if (n <= 0xffff)
                        return 2;
                else if (n <= 0xffffff)
                        return 3;
                else
                        return 4;
        }

        public static void put3sLongitude(ImgFileWriter writer, int longitude) {
                // handle special case, write -180/-8388608 instead of 180/8388608 to avoid assertion
                if (longitude == Utils.MAX_LON_MAP_UNITS)
                        writer.put3s(Utils.MIN_LON_MAP_UNITS);
                else
                        writer.put3s(longitude);
        }

        /**
     * Finds the intersection of two line segments, also if one ends on the other
     * See https://de.wikipedia.org/wiki/Geod%C3%A4tisches_Rechnen#Geradenschnitt
     * @param p1_1 the coordinates of the start point of the first specified line segment
     * @param p1_2 the coordinates of the end point of the first specified line segment
     * @param p2_1 the coordinates of the start point of the second specified line segment
     * @param p2_2 the coordinates of the end point of the second specified line segment
     * @return null if no intersection was found, else a new Coord instance with the coordinates of the intersection
         */

        public static Coord getSegmentSegmentIntersection (Coord p1_1, Coord p1_2, Coord p2_1, Coord p2_2) {
                long x1 = p1_1.getHighPrecLon();
                long y1 = p1_1.getHighPrecLat();
                long x2 = p1_2.getHighPrecLon();
                long y2 = p1_2.getHighPrecLat();
                long x3 = p2_1.getHighPrecLon();
                long y3 = p2_1.getHighPrecLat();
                long x4 = p2_2.getHighPrecLon();
                long y4 = p2_2.getHighPrecLat();
               
                if (!Line2D.linesIntersect(x1, y1, x2, y2, x3, y3, x4, y4))
                        return null;
               
                long divider = (y2 - y1) * (x4 - x3) - (y4 - y3) * (x2 - x1);
                if (divider == 0)
                        return null; // parallel
                double h = ((y4 - y3) * (x1 - x4) - (y1 - y4) * (x4 - x3)) / (double) divider;

                if (h < 0 || h > 1) {
                        return null; // intersection of lines is not on given segment
                }
                double xs = x1 + h * (x2 - x1);
                double ys = y1 + h * (y2 - y1);
                return Coord.makeHighPrecCoord((int)Math.round(ys), (int) Math.round(xs));
        }
               
}