2013-04-18 23:35:02 +02:00
|
|
|
package net.osmand.util;
|
|
|
|
|
|
|
|
|
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.Comparator;
|
|
|
|
import java.util.List;
|
|
|
|
|
2016-08-02 09:56:07 +02:00
|
|
|
import net.osmand.binary.BinaryMapIndexReader.SearchRequest;
|
2013-04-18 23:35:02 +02:00
|
|
|
import net.osmand.data.LatLon;
|
|
|
|
import net.osmand.data.MapObject;
|
2013-10-26 14:40:59 +02:00
|
|
|
import net.osmand.data.QuadPoint;
|
2016-08-02 09:56:07 +02:00
|
|
|
import net.osmand.data.QuadRect;
|
2015-01-13 19:32:57 +01:00
|
|
|
import net.osmand.util.GeoPointParserUtil.GeoParsedPoint;
|
2013-04-18 23:35:02 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-05-27 10:50:24 +02:00
|
|
|
* This utility class includes :
|
2013-04-18 23:35:02 +02:00
|
|
|
* 1. distance algorithms
|
|
|
|
* 2. finding center for array of nodes
|
|
|
|
* 3. tile evaluation algorithms
|
|
|
|
*/
|
|
|
|
public class MapUtils {
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2016-05-30 22:09:20 +02:00
|
|
|
public static final double MIN_LATITUDE = -85.0511;
|
|
|
|
public static final double MAX_LATITUDE = 85.0511;
|
2016-05-27 11:28:47 +02:00
|
|
|
public static final double LATITUDE_TURN = 180.0;
|
2016-05-30 22:09:20 +02:00
|
|
|
public static final double MIN_LONGITUDE = -180.0;
|
|
|
|
public static final double MAX_LONGITUDE = 180.0;
|
2016-05-27 11:28:47 +02:00
|
|
|
public static final double LONGITUDE_TURN = 360.0;
|
|
|
|
|
2016-05-27 10:50:24 +02:00
|
|
|
// TODO change the hostname back to osm.org once HTTPS works for it
|
|
|
|
// https://github.com/openstreetmap/operations/issues/2
|
|
|
|
private static final String BASE_SHORT_OSM_URL = "https://openstreetmap.org/go/";
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
|
|
|
* This array is a lookup table that translates 6-bit positive integer
|
|
|
|
* index values into their "Base64 Alphabet" equivalents as specified
|
|
|
|
* in Table 1 of RFC 2045.
|
|
|
|
*/
|
|
|
|
private static final char intToBase64[] = {
|
|
|
|
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
|
|
|
|
'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
|
|
|
|
'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
|
|
|
|
'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
|
2015-01-13 12:09:14 +01:00
|
|
|
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '_', '~'
|
2013-04-18 23:35:02 +02:00
|
|
|
};
|
|
|
|
|
2016-05-27 10:50:24 +02:00
|
|
|
public static double getDistance(LatLon l, double latitude, double longitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
return getDistance(l.getLatitude(), l.getLongitude(), latitude, longitude);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
private static double scalarMultiplication(double xA, double yA, double xB, double yB, double xC, double yC) {
|
|
|
|
// Scalar multiplication between (AB, AC)
|
2016-05-27 10:50:24 +02:00
|
|
|
return (xB - xA) * (xC - xA) + (yB - yA) * (yC - yA);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public static double getOrthogonalDistance(double lat, double lon, double fromLat, double fromLon, double toLat, double toLon) {
|
|
|
|
return getDistance(getProjection(lat, lon, fromLat, fromLon, toLat, toLon), lat, lon);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public static LatLon getProjection(double lat, double lon, double fromLat, double fromLon, double toLat, double toLon) {
|
|
|
|
// not very accurate computation on sphere but for distances < 1000m it is ok
|
|
|
|
double mDist = (fromLat - toLat) * (fromLat - toLat) + (fromLon - toLon) * (fromLon - toLon);
|
|
|
|
double projection = scalarMultiplication(fromLat, fromLon, toLat, toLon, lat, lon);
|
|
|
|
double prlat;
|
|
|
|
double prlon;
|
|
|
|
if (projection < 0) {
|
|
|
|
prlat = fromLat;
|
|
|
|
prlon = fromLon;
|
|
|
|
} else if (projection >= mDist) {
|
|
|
|
prlat = toLat;
|
|
|
|
prlon = toLon;
|
|
|
|
} else {
|
|
|
|
prlat = fromLat + (toLat - fromLat) * (projection / mDist);
|
|
|
|
prlon = fromLon + (toLon - fromLon) * (projection / mDist);
|
|
|
|
}
|
|
|
|
return new LatLon(prlat, prlon);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2015-11-30 00:41:35 +01:00
|
|
|
public static double getProjectionCoeff(double lat, double lon, double fromLat, double fromLon, double toLat, double toLon) {
|
|
|
|
// not very accurate computation on sphere but for distances < 1000m it is ok
|
|
|
|
double mDist = (fromLat - toLat) * (fromLat - toLat) + (fromLon - toLon) * (fromLon - toLon);
|
|
|
|
double projection = scalarMultiplication(fromLat, fromLon, toLat, toLon, lat, lon);
|
|
|
|
double prlat;
|
|
|
|
double prlon;
|
|
|
|
if (projection < 0) {
|
|
|
|
return 0;
|
|
|
|
} else if (projection >= mDist) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return (projection / mDist);
|
|
|
|
}
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
private static double toRadians(double angdeg) {
|
|
|
|
// return Math.toRadians(angdeg);
|
|
|
|
return angdeg / 180.0 * Math.PI;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
|
|
|
* Gets distance in meters
|
|
|
|
*/
|
2016-05-27 10:50:24 +02:00
|
|
|
public static double getDistance(double lat1, double lon1, double lat2, double lon2) {
|
2013-06-18 11:00:26 +02:00
|
|
|
double R = 6372.8; // for haversine use R = 6372.8 km instead of 6371 km
|
2016-05-27 10:50:24 +02:00
|
|
|
double dLat = toRadians(lat2 - lat1);
|
|
|
|
double dLon = toRadians(lon2 - lon1);
|
|
|
|
double a = Math.sin(dLat / 2) * Math.sin(dLat / 2) +
|
|
|
|
Math.cos(toRadians(lat1)) * Math.cos(toRadians(lat2)) *
|
|
|
|
Math.sin(dLon / 2) * Math.sin(dLon / 2);
|
2013-06-18 11:00:26 +02:00
|
|
|
//double c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1-a));
|
|
|
|
//return R * c * 1000;
|
|
|
|
// simplyfy haversine:
|
2013-07-31 20:22:39 +02:00
|
|
|
return (2 * R * 1000 * Math.asin(Math.sqrt(a)));
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
|
|
|
* Gets distance in meters
|
|
|
|
*/
|
2016-05-27 10:50:24 +02:00
|
|
|
public static double getDistance(LatLon l1, LatLon l2) {
|
2013-04-18 23:35:02 +02:00
|
|
|
return getDistance(l1.getLatitude(), l1.getLongitude(), l2.getLatitude(), l2.getLongitude());
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public static double checkLongitude(double longitude) {
|
2016-05-31 22:15:18 +02:00
|
|
|
if (longitude >= MIN_LONGITUDE && longitude <= MAX_LONGITUDE) {
|
2014-06-27 01:25:38 +02:00
|
|
|
return longitude;
|
|
|
|
}
|
2016-05-30 22:09:20 +02:00
|
|
|
while (longitude <= MIN_LONGITUDE || longitude > MAX_LONGITUDE) {
|
2013-04-18 23:35:02 +02:00
|
|
|
if (longitude < 0) {
|
2016-05-27 11:28:47 +02:00
|
|
|
longitude += LONGITUDE_TURN;
|
2013-04-18 23:35:02 +02:00
|
|
|
} else {
|
2016-05-27 11:28:47 +02:00
|
|
|
longitude -= LONGITUDE_TURN;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return longitude;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public static double checkLatitude(double latitude) {
|
2016-05-30 22:09:20 +02:00
|
|
|
if (latitude >= MIN_LATITUDE && latitude <= MAX_LATITUDE) {
|
2016-05-28 08:30:52 +02:00
|
|
|
return latitude;
|
|
|
|
}
|
2013-04-18 23:35:02 +02:00
|
|
|
while (latitude < -90 || latitude > 90) {
|
|
|
|
if (latitude < 0) {
|
2016-05-27 11:28:47 +02:00
|
|
|
latitude += LATITUDE_TURN;
|
2013-04-18 23:35:02 +02:00
|
|
|
} else {
|
2016-05-27 11:28:47 +02:00
|
|
|
latitude -= LATITUDE_TURN;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
}
|
2016-05-30 22:09:20 +02:00
|
|
|
if (latitude < MIN_LATITUDE) {
|
|
|
|
return MIN_LATITUDE;
|
|
|
|
} else if (latitude > MAX_LATITUDE) {
|
|
|
|
return MAX_LATITUDE;
|
2016-05-27 10:50:24 +02:00
|
|
|
}
|
2013-04-18 23:35:02 +02:00
|
|
|
return latitude;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static int get31TileNumberX(double longitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
longitude = checkLongitude(longitude);
|
2013-08-03 14:59:26 +02:00
|
|
|
long l = 1L << 31;
|
2016-05-27 10:50:24 +02:00
|
|
|
return (int) ((longitude + 180d)/360d * l);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static int get31TileNumberY(double latitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
latitude = checkLatitude(latitude);
|
2016-05-27 10:50:24 +02:00
|
|
|
double eval = Math.log(Math.tan(toRadians(latitude)) + 1/Math.cos(toRadians(latitude)));
|
2013-08-03 14:59:26 +02:00
|
|
|
long l = 1L << 31;
|
2016-05-27 10:50:24 +02:00
|
|
|
if (eval > Math.PI) {
|
2013-04-18 23:35:02 +02:00
|
|
|
eval = Math.PI;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
return (int) ((1 - eval / Math.PI) / 2 * l);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double get31LongitudeX(int tileX) {
|
|
|
|
return MapUtils.getLongitudeFromTile(21, tileX / 1024f);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double get31LatitudeY(int tileY) {
|
2016-01-23 10:27:53 +01:00
|
|
|
return MapUtils.getLatitudeFromTile(21, tileY / 1024f);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
2016-05-27 10:50:24 +02:00
|
|
|
* Theses methods operate with degrees (evaluating tiles & vice versa)
|
2013-04-18 23:35:02 +02:00
|
|
|
* degree longitude measurements (-180, 180) [27.56 Minsk]
|
2016-05-27 10:50:24 +02:00
|
|
|
* // degree latitude measurements (90, -90) [53.9]
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double getTileNumberX(float zoom, double longitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
longitude = checkLongitude(longitude);
|
2014-06-13 11:31:57 +02:00
|
|
|
final double powZoom = getPowZoom(zoom);
|
|
|
|
double dz = (longitude + 180d)/360d * powZoom;
|
|
|
|
if (dz >= powZoom) {
|
|
|
|
return powZoom - 0.01;
|
|
|
|
}
|
|
|
|
return dz;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double getTileNumberY(float zoom, double latitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
latitude = checkLatitude(latitude);
|
2016-05-27 10:50:24 +02:00
|
|
|
double eval = Math.log(Math.tan(toRadians(latitude)) + 1/Math.cos(toRadians(latitude)));
|
2013-04-18 23:35:02 +02:00
|
|
|
if (Double.isInfinite(eval) || Double.isNaN(eval)) {
|
2016-05-27 10:50:24 +02:00
|
|
|
latitude = latitude < 0 ? -89.9 : 89.9;
|
|
|
|
eval = Math.log(Math.tan(toRadians(latitude)) + 1/Math.cos(toRadians(latitude)));
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2013-07-31 20:22:39 +02:00
|
|
|
return (1 - eval / Math.PI) / 2 * getPowZoom(zoom);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double getTileEllipsoidNumberY(float zoom, double latitude) {
|
2013-04-18 23:35:02 +02:00
|
|
|
final double E2 = (double) latitude * Math.PI / 180;
|
|
|
|
final long sradiusa = 6378137;
|
|
|
|
final long sradiusb = 6356752;
|
2016-05-27 10:50:24 +02:00
|
|
|
final double J2 = (double) Math.sqrt(sradiusa * sradiusa - sradiusb * sradiusb) / sradiusa;
|
2013-04-18 23:35:02 +02:00
|
|
|
final double M2 = (double) Math.log((1 + Math.sin(E2))
|
2016-05-27 10:50:24 +02:00
|
|
|
/ (1 - Math.sin(E2))) / 2 - J2 * Math.log((1 + J2 * Math.sin(E2)) / (1 - J2 * Math.sin(E2))) / 2;
|
2013-04-18 23:35:02 +02:00
|
|
|
final double B2 = getPowZoom(zoom);
|
|
|
|
return B2 / 2 - M2 * B2 / 2 / Math.PI;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double getLatitudeFromEllipsoidTileY(float zoom, float tileNumberY) {
|
2013-04-18 23:35:02 +02:00
|
|
|
final double MerkElipsK = 0.0000001;
|
|
|
|
final long sradiusa = 6378137;
|
|
|
|
final long sradiusb = 6356752;
|
|
|
|
final double FExct = (double) Math.sqrt(sradiusa * sradiusa
|
|
|
|
- sradiusb * sradiusb)
|
|
|
|
/ sradiusa;
|
|
|
|
final double TilesAtZoom = getPowZoom(zoom);
|
|
|
|
double result = (tileNumberY - TilesAtZoom / 2)
|
|
|
|
/ -(TilesAtZoom / (2 * Math.PI));
|
|
|
|
result = (2 * Math.atan(Math.exp(result)) - Math.PI / 2) * 180
|
|
|
|
/ Math.PI;
|
|
|
|
double Zu = result / (180 / Math.PI);
|
|
|
|
double yy = (tileNumberY - TilesAtZoom / 2);
|
|
|
|
|
|
|
|
double Zum1 = Zu;
|
|
|
|
Zu = Math.asin(1 - ((1 + Math.sin(Zum1)) * Math.pow(1 - FExct * Math.sin(Zum1), FExct))
|
|
|
|
/ (Math.exp((2 * yy) / -(TilesAtZoom / (2 * Math.PI))) * Math.pow(1 + FExct * Math.sin(Zum1), FExct)));
|
|
|
|
while (Math.abs(Zum1 - Zu) >= MerkElipsK) {
|
|
|
|
Zum1 = Zu;
|
|
|
|
Zu = Math.asin(1 - ((1 + Math.sin(Zum1)) * Math.pow(1 - FExct * Math.sin(Zum1), FExct))
|
|
|
|
/ (Math.exp((2 * yy) / -(TilesAtZoom / (2 * Math.PI))) * Math.pow(1 + FExct * Math.sin(Zum1), FExct)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return Zu * 180 / Math.PI;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
2014-08-05 11:09:15 +02:00
|
|
|
public static double getTileDistanceWidth(float zoom) {
|
|
|
|
LatLon ll = new LatLon(30, MapUtils.getLongitudeFromTile(zoom, 0));
|
|
|
|
LatLon ll2 = new LatLon(30, MapUtils.getLongitudeFromTile(zoom, 1));
|
2016-05-27 10:50:24 +02:00
|
|
|
return getDistance(ll, ll2);
|
2014-08-05 11:09:15 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2014-10-01 00:27:01 +02:00
|
|
|
public static double getLongitudeFromTile(double zoom, double x) {
|
2013-04-18 23:35:02 +02:00
|
|
|
return x / getPowZoom(zoom) * 360.0 - 180.0;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static double getPowZoom(double zoom) {
|
|
|
|
if (zoom >= 0 && zoom - Math.floor(zoom) < 0.001f) {
|
|
|
|
return 1 << ((int) zoom);
|
2013-04-18 23:35:02 +02:00
|
|
|
} else {
|
|
|
|
return Math.pow(2, zoom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static float calcDiffPixelX(float rotateSin, float rotateCos, float dTileX, float dTileY, float tileSize) {
|
|
|
|
return (rotateCos * dTileX - rotateSin * dTileY) * tileSize;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
public static float calcDiffPixelY(float rotateSin, float rotateCos, float dTileX, float dTileY, float tileSize) {
|
|
|
|
return (rotateSin * dTileX + rotateCos * dTileY) * tileSize;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2014-09-27 11:44:37 +02:00
|
|
|
public static double getLatitudeFromTile(float zoom, double y) {
|
2013-04-18 23:35:02 +02:00
|
|
|
int sign = y < 0 ? -1 : 1;
|
2014-09-27 11:44:37 +02:00
|
|
|
return Math.atan(sign * Math.sinh(Math.PI * (1 - 2 * y / getPowZoom(zoom)))) * 180d / Math.PI;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
public static int getPixelShiftX(float zoom, double long1, double long2, double tileSize) {
|
2013-04-18 23:35:02 +02:00
|
|
|
return (int) ((getTileNumberX(zoom, long1) - getTileNumberX(zoom, long2)) * tileSize);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
public static int getPixelShiftY(float zoom, double lat1, double lat2, double tileSize) {
|
2013-04-18 23:35:02 +02:00
|
|
|
return (int) ((getTileNumberY(zoom, lat1) - getTileNumberY(zoom, lat2)) * tileSize);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
public static void sortListOfMapObject(List<? extends MapObject> list, final double lat, final double lon) {
|
2013-04-18 23:35:02 +02:00
|
|
|
Collections.sort(list, new Comparator<MapObject>() {
|
|
|
|
@Override
|
|
|
|
public int compare(MapObject o1, MapObject o2) {
|
|
|
|
return Double.compare(MapUtils.getDistance(o1.getLocation(), lat, lon), MapUtils.getDistance(o2.getLocation(),
|
|
|
|
lat, lon));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2015-01-07 22:08:04 +01:00
|
|
|
public static String buildGeoUrl(double latitude, double longitude, int zoom) {
|
2016-05-27 10:50:24 +02:00
|
|
|
return "geo:" + ((float) latitude) + "," + ((float) longitude) + "?z=" + zoom;
|
2015-01-07 22:08:04 +01:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
// Examples
|
|
|
|
// System.out.println(buildShortOsmUrl(51.51829d, 0.07347d, 16)); // http://osm.org/go/0EEQsyfu
|
|
|
|
// System.out.println(buildShortOsmUrl(52.30103d, 4.862927d, 18)); // http://osm.org/go/0E4_JiVhs
|
|
|
|
// System.out.println(buildShortOsmUrl(40.59d, -115.213d, 9)); // http://osm.org/go/TelHTB--
|
2016-05-27 10:50:24 +02:00
|
|
|
public static String buildShortOsmUrl(double latitude, double longitude, int zoom) {
|
|
|
|
return BASE_SHORT_OSM_URL + createShortLinkString(latitude, longitude, zoom) + "?m";
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
|
2015-01-13 19:32:57 +01:00
|
|
|
public static String createShortLinkString(double latitude, double longitude, int zoom) {
|
2013-08-03 14:59:26 +02:00
|
|
|
long lat = (long) (((latitude + 90d)/180d)*(1L << 32));
|
|
|
|
long lon = (long) (((longitude + 180d)/360d)*(1L << 32));
|
2013-04-18 23:35:02 +02:00
|
|
|
long code = interleaveBits(lon, lat);
|
|
|
|
String str = "";
|
2016-05-27 10:50:24 +02:00
|
|
|
// add eight to the zoom level, which approximates an accuracy of one pixel in a tile.
|
2013-04-18 23:35:02 +02:00
|
|
|
for (int i = 0; i < Math.ceil((zoom + 8) / 3d); i++) {
|
2016-05-27 10:50:24 +02:00
|
|
|
str += intToBase64[(int) ((code >> (58 - 6 * i)) & 0x3f)];
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
// append characters onto the end of the string to represent
|
|
|
|
// partial zoom levels (characters themselves have a granularity of 3 zoom levels).
|
|
|
|
for (int j = 0; j < (zoom + 8) % 3; j++) {
|
|
|
|
str += '-';
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2015-01-13 19:32:57 +01:00
|
|
|
public static GeoParsedPoint decodeShortLinkString(String s) {
|
2015-01-13 12:09:14 +01:00
|
|
|
// convert old shortlink format to current one
|
|
|
|
s = s.replaceAll("@", "~");
|
2015-01-13 19:32:57 +01:00
|
|
|
int i = 0;
|
2013-04-18 23:35:02 +02:00
|
|
|
long x = 0;
|
|
|
|
long y = 0;
|
2015-01-13 19:32:57 +01:00
|
|
|
int z = -8;
|
2013-04-18 23:35:02 +02:00
|
|
|
|
2015-01-13 19:32:57 +01:00
|
|
|
for (i = 0; i < s.length(); i++) {
|
|
|
|
int digit = -1;
|
2013-04-18 23:35:02 +02:00
|
|
|
char c = s.charAt(i);
|
2015-01-13 19:32:57 +01:00
|
|
|
for (int j = 0; j < intToBase64.length; j++)
|
2013-04-18 23:35:02 +02:00
|
|
|
if (c == intToBase64[j]) {
|
2015-01-13 19:32:57 +01:00
|
|
|
digit = j;
|
2013-04-18 23:35:02 +02:00
|
|
|
break;
|
|
|
|
}
|
2015-01-13 19:32:57 +01:00
|
|
|
if (digit < 0)
|
|
|
|
break;
|
|
|
|
if (digit < 0)
|
|
|
|
break;
|
|
|
|
// distribute 6 bits into x and y
|
|
|
|
x <<= 3;
|
|
|
|
y <<= 3;
|
|
|
|
for (int j = 2; j >= 0; j--) {
|
2016-05-27 10:50:24 +02:00
|
|
|
x |= ((digit & (1 << (j + j + 1))) == 0 ? 0 : (1 << j));
|
|
|
|
y |= ((digit & (1 << (j + j))) == 0 ? 0 : (1 << j));
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2015-01-13 19:32:57 +01:00
|
|
|
z += 3;
|
|
|
|
}
|
|
|
|
double lon = x * Math.pow(2, 2 - 3 * i) * 90. - 180;
|
|
|
|
double lat = y * Math.pow(2, 2 - 3 * i) * 45. - 90;
|
|
|
|
// adjust z
|
2016-05-27 10:50:24 +02:00
|
|
|
if (i < s.length() && s.charAt(i) == '-') {
|
2015-01-13 19:32:57 +01:00
|
|
|
z -= 2;
|
2016-05-27 10:50:24 +02:00
|
|
|
if (i + 1 < s.length() && s.charAt(i + 1) == '-')
|
2015-01-13 19:32:57 +01:00
|
|
|
z++;
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2015-01-13 19:32:57 +01:00
|
|
|
return new GeoParsedPoint(lat, lon, z);
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* interleaves the bits of two 32-bit numbers. the result is known as a Morton code.
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
2016-07-29 02:42:03 +02:00
|
|
|
public static long interleaveBits(long x, long y) {
|
2013-04-18 23:35:02 +02:00
|
|
|
long c = 0;
|
2016-05-27 10:50:24 +02:00
|
|
|
for (byte b = 31; b >= 0; b--) {
|
2013-04-18 23:35:02 +02:00
|
|
|
c = (c << 1) | ((x >> b) & 1);
|
|
|
|
c = (c << 1) | ((y >> b) & 1);
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate rotation diff D, that R (rotate) + D = T (targetRotate)
|
2016-05-27 10:50:24 +02:00
|
|
|
* D is between -180, 180
|
|
|
|
*
|
2013-04-18 23:35:02 +02:00
|
|
|
* @param rotate
|
|
|
|
* @param targetRotate
|
2016-05-27 10:50:24 +02:00
|
|
|
* @return
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
|
|
|
public static float unifyRotationDiff(float rotate, float targetRotate) {
|
|
|
|
float d = targetRotate - rotate;
|
2016-05-27 10:50:24 +02:00
|
|
|
while (d >= 180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
d -= 360;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
while (d < -180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
d += 360;
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
|
|
|
* Calculate rotation diff D, that R (rotate) + D = T (targetRotate)
|
2016-05-27 10:50:24 +02:00
|
|
|
* D is between -180, 180
|
|
|
|
*
|
2013-04-18 23:35:02 +02:00
|
|
|
* @param rotate
|
2013-09-29 16:49:17 +02:00
|
|
|
* @return
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
|
|
|
public static float unifyRotationTo360(float rotate) {
|
2016-05-27 10:50:24 +02:00
|
|
|
while (rotate < -180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
rotate += 360;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
while (rotate > +180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
rotate -= 360;
|
|
|
|
}
|
|
|
|
return rotate;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-05-27 10:50:24 +02:00
|
|
|
* @param diff align difference between 2 angles ]-PI, PI]
|
|
|
|
* @return
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
|
|
|
public static double alignAngleDifference(double diff) {
|
2016-05-27 10:50:24 +02:00
|
|
|
while (diff > Math.PI) {
|
2013-04-18 23:35:02 +02:00
|
|
|
diff -= 2 * Math.PI;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
while (diff <= -Math.PI) {
|
2013-04-18 23:35:02 +02:00
|
|
|
diff += 2 * Math.PI;
|
|
|
|
}
|
|
|
|
return diff;
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
/**
|
2016-05-27 10:50:24 +02:00
|
|
|
* diff align difference between 2 angles [-180, 180]
|
|
|
|
*
|
|
|
|
* @return
|
2013-04-18 23:35:02 +02:00
|
|
|
*/
|
|
|
|
public static double degreesDiff(double a1, double a2) {
|
|
|
|
double diff = a1 - a2;
|
2016-05-27 10:50:24 +02:00
|
|
|
while (diff > 180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
diff -= 360;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
while (diff <= -180) {
|
2013-04-18 23:35:02 +02:00
|
|
|
diff += 360;
|
|
|
|
}
|
|
|
|
return diff;
|
2016-05-27 10:50:24 +02:00
|
|
|
}
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
|
2014-02-05 21:10:04 +01:00
|
|
|
public static double convert31YToMeters(double y1, double y2) {
|
2013-04-18 23:35:02 +02:00
|
|
|
// translate into meters
|
|
|
|
return (y1 - y2) * 0.01863d;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2014-02-05 21:10:04 +01:00
|
|
|
public static double convert31XToMeters(double x1, double x2) {
|
2013-04-18 23:35:02 +02:00
|
|
|
// translate into meters
|
|
|
|
return (x1 - x2) * 0.011d;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
public static QuadPoint getProjectionPoint31(int px, int py, int st31x, int st31y, int end31x, int end31y) {
|
2013-10-28 22:46:29 +01:00
|
|
|
double projection = calculateProjection31TileMetric(st31x, st31y, end31x,
|
|
|
|
end31y, px, py);
|
2013-10-26 14:40:59 +02:00
|
|
|
double mDist = squareRootDist31(end31x, end31y, st31x,
|
|
|
|
st31y);
|
|
|
|
int pry = end31y;
|
2013-10-28 22:46:29 +01:00
|
|
|
int prx = end31x;
|
2013-10-26 14:40:59 +02:00
|
|
|
if (projection < 0) {
|
|
|
|
prx = st31x;
|
|
|
|
pry = st31y;
|
|
|
|
} else if (projection >= mDist * mDist) {
|
|
|
|
prx = end31x;
|
|
|
|
pry = end31y;
|
|
|
|
} else {
|
|
|
|
prx = (int) (st31x + (end31x - st31x)
|
|
|
|
* (projection / (mDist * mDist)));
|
|
|
|
pry = (int) (st31y + (end31y - st31y)
|
|
|
|
* (projection / (mDist * mDist)));
|
|
|
|
}
|
|
|
|
return new QuadPoint(prx, pry);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
|
|
|
|
2013-10-26 14:40:59 +02:00
|
|
|
public static double squareRootDist31(int x1, int y1, int x2, int y2) {
|
|
|
|
// translate into meters
|
|
|
|
double dy = MapUtils.convert31YToMeters(y1, y2);
|
|
|
|
double dx = MapUtils.convert31XToMeters(x1, x2);
|
|
|
|
return Math.sqrt(dx * dx + dy * dy);
|
|
|
|
// return measuredDist(x1, y1, x2, y2);
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-12-18 10:55:08 +01:00
|
|
|
public static double measuredDist31(int x1, int y1, int x2, int y2) {
|
|
|
|
return getDistance(MapUtils.get31LatitudeY(y1), MapUtils.get31LongitudeX(x1), MapUtils.get31LatitudeY(y2), MapUtils.get31LongitudeX(x2));
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-10-29 21:13:57 +01:00
|
|
|
public static double squareDist31TileMetric(int x1, int y1, int x2, int y2) {
|
|
|
|
// translate into meters
|
|
|
|
double dy = convert31YToMeters(y1, y2);
|
|
|
|
double dx = convert31XToMeters(x1, x2);
|
|
|
|
return dx * dx + dy * dy;
|
|
|
|
}
|
2016-05-27 10:50:24 +02:00
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
public static double calculateProjection31TileMetric(int xA, int yA, int xB, int yB, int xC, int yC) {
|
|
|
|
// Scalar multiplication between (AB, AC)
|
2013-10-26 14:40:59 +02:00
|
|
|
double multiple = MapUtils.convert31XToMeters(xB, xA) * MapUtils.convert31XToMeters(xC, xA) +
|
|
|
|
MapUtils.convert31YToMeters(yB, yA) * MapUtils.convert31YToMeters(yC, yA);
|
2013-04-18 23:35:02 +02:00
|
|
|
return multiple;
|
|
|
|
}
|
2013-09-29 16:49:17 +02:00
|
|
|
|
2016-01-23 10:27:53 +01:00
|
|
|
public static boolean rightSide(double lat, double lon,
|
2016-05-27 10:50:24 +02:00
|
|
|
double aLat, double aLon,
|
|
|
|
double bLat, double bLon) {
|
2016-01-23 10:27:53 +01:00
|
|
|
double ax = aLon - lon;
|
|
|
|
double ay = aLat - lat;
|
|
|
|
double bx = bLon - lon;
|
|
|
|
double by = bLat - lat;
|
|
|
|
double sa = ax * by - bx * ay;
|
|
|
|
return sa < 0;
|
|
|
|
}
|
2013-09-29 16:49:17 +02:00
|
|
|
|
2016-07-29 02:42:03 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public static long deinterleaveY(long coord) {
|
|
|
|
long x = 0;
|
|
|
|
for (byte b = 31; b >= 0; b--) {
|
|
|
|
x = (x << 1) | (1 & coord >> (b * 2));
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static long deinterleaveX(long coord) {
|
|
|
|
long x = 0;
|
|
|
|
for (byte b = 31; b >= 0; b--) {
|
|
|
|
x = (x << 1) | (1 & coord >> (b * 2 + 1));
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
2016-08-02 09:56:07 +02:00
|
|
|
|
|
|
|
public static QuadRect calculateLatLonBbox(double latitude, double longitude, int radiusMeters) {
|
|
|
|
int zoom = 16;
|
|
|
|
float coeff = (float) (radiusMeters / MapUtils.getTileDistanceWidth(zoom));
|
|
|
|
double tx = MapUtils.getTileNumberX(zoom, longitude);
|
|
|
|
double ty = MapUtils.getTileNumberY(zoom, latitude);
|
|
|
|
double topLeftX = Math.max(0, tx - coeff);
|
|
|
|
double topLeftY = Math.max(0, ty - coeff);
|
|
|
|
int max = (1 << zoom) - 1;
|
|
|
|
double bottomRightX = Math.min(max, tx + coeff);
|
|
|
|
double bottomRightY = Math.min(max, ty + coeff);
|
|
|
|
double pw = MapUtils.getPowZoom(31 - zoom);
|
|
|
|
QuadRect rect = new QuadRect(topLeftX * pw, topLeftY * pw, bottomRightX * pw, bottomRightY * pw);
|
|
|
|
rect.left = MapUtils.get31LongitudeX((int) rect.left);
|
|
|
|
rect.top = MapUtils.get31LatitudeY((int) rect.top);
|
|
|
|
rect.right = MapUtils.get31LongitudeX((int) rect.right);
|
|
|
|
rect.bottom = MapUtils.get31LatitudeY((int) rect.bottom);
|
|
|
|
return rect;
|
|
|
|
}
|
2016-07-29 02:42:03 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-04-18 23:35:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|