-package tim.prune.function.browser;
-
-import java.text.DecimalFormat;
-import java.text.NumberFormat;
-import java.util.Locale;
-
-import tim.prune.I18nManager;
-import tim.prune.data.DataPoint;
-import tim.prune.data.DoubleRange;
-import tim.prune.data.TrackInfo;
-
-/**
- * Class to manage the generation of map urls
- * for display in an external browser
- */
-public abstract class UrlGenerator
-{
- /** Number formatter for five dp */
- private static final NumberFormat FIVE_DP = NumberFormat.getNumberInstance(Locale.UK);
- // Select the UK locale for this formatter so that decimal point is always used (not comma)
- static {
- if (FIVE_DP instanceof DecimalFormat) ((DecimalFormat) FIVE_DP).applyPattern("0.00000");
- }
-
- public enum WebService
- {
- MAP_SOURCE_GOOGLE, /* Google maps */
- MAP_SOURCE_OSM, /* OpenStreetMap */
- MAP_SOURCE_MAPQUEST, /* Mapquest */
- MAP_SOURCE_YAHOO, /* Yahoo */
- MAP_SOURCE_BING, /* Bing */
- MAP_SOURCE_PEAKFINDER, /* PeakFinder */
- MAP_SOURCE_GEOHACK, /* Geohack */
- MAP_SOURCE_INLINESKATE, /* Inlinemap.net */
- MAP_SOURCE_GRAPHHOPPER /* Routing with GraphHopper */
- }
-
- /**
- * Generate a URL for the given source and track info
- * @param inSource source to use, from the enum in UrlGenerator
- * @param inTrackInfo track info
- * @return url for map
- */
- public static String generateUrl(WebService inSource, TrackInfo inTrackInfo)
- {
- switch (inSource)
- {
- case MAP_SOURCE_GOOGLE:
- return generateGoogleUrl(inTrackInfo);
- case MAP_SOURCE_MAPQUEST:
- return generateMapquestUrl(inTrackInfo);
- case MAP_SOURCE_YAHOO:
- return generateYahooUrl(inTrackInfo);
- case MAP_SOURCE_BING:
- return generateBingUrl(inTrackInfo);
- case MAP_SOURCE_PEAKFINDER:
- case MAP_SOURCE_GEOHACK:
- case MAP_SOURCE_INLINESKATE:
- return generateUrlForPoint(inSource, inTrackInfo);
- case MAP_SOURCE_GRAPHHOPPER:
- return generateGraphHopperUrl(inTrackInfo);
- case MAP_SOURCE_OSM:
- default:
- return generateOpenStreetMapUrl(inTrackInfo);
- }
- }
-
- /**
- * Generate a url for Google maps
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateGoogleUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- double medianLat = getMedianValue(inTrackInfo.getTrack().getLatRange());
- double medianLon = getMedianValue(inTrackInfo.getTrack().getLonRange());
- double latSpan = getSpan(inTrackInfo.getTrack().getLatRange());
- double lonSpan = getSpan(inTrackInfo.getTrack().getLonRange());
- // Build basic url with centre position and span
- String url = "http://" + I18nManager.getText("url.googlemaps")
- + "/?ll=" + FIVE_DP.format(medianLat) + "," + FIVE_DP.format(medianLon)
- + "&spn=" + FIVE_DP.format(latSpan) + "," + FIVE_DP.format(lonSpan);
- DataPoint currPoint = inTrackInfo.getCurrentPoint();
- // Add selected point, if any
- if (currPoint != null) {
- url = url + "&q=" + FIVE_DP.format(currPoint.getLatitude().getDouble()) + ","
- + FIVE_DP.format(currPoint.getLongitude().getDouble());
- if (currPoint.getWaypointName() != null) {
- url = url + "(" + currPoint.getWaypointName() + ")";
- }
- }
- return url;
- }
-
- /**
- * Generate a url for Mapquest maps
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateMapquestUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- double medianLat = getMedianValue(inTrackInfo.getTrack().getLatRange());
- double medianLon = getMedianValue(inTrackInfo.getTrack().getLonRange());
- // Build basic url with centre position
- String url = "http://atlas.mapquest.com/maps/map.adp?latlongtype=decimal&latitude="
- + FIVE_DP.format(medianLat) + "&longitude=" + FIVE_DP.format(medianLon);
- return url;
- }
-
-
- /**
- * Generate a url for Yahoo maps
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateYahooUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- double medianLat = getMedianValue(inTrackInfo.getTrack().getLatRange());
- double medianLon = getMedianValue(inTrackInfo.getTrack().getLonRange());
- // Build basic url with centre position
- String url = "http://maps.yahoo.com/#lat=" + FIVE_DP.format(medianLat)
- + "&lon=" + FIVE_DP.format(medianLon) + "&zoom=13";
- return url;
- }
-
- /**
- * Generate a url for Bing maps
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateBingUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- double medianLat = getMedianValue(inTrackInfo.getTrack().getLatRange());
- double medianLon = getMedianValue(inTrackInfo.getTrack().getLonRange());
- // Build basic url with centre position
- String latStr = FIVE_DP.format(medianLat);
- String lonStr = FIVE_DP.format(medianLon);
- String url = "http://bing.com/maps/default.aspx?cp=" + latStr + "~" + lonStr
- + "&where1=" + latStr + "%2C%20" + lonStr;
- return url;
- }
-
- /**
- * Generate a url for routing with GraphHopper
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateGraphHopperUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo != null && inTrackInfo.getTrack() != null && inTrackInfo.getTrack().getNumPoints() >= 2)
- {
- if (inTrackInfo.getTrack().getNumPoints() == 2)
- {
- // Use first and last point of track
- return generateGraphHopperUrl(inTrackInfo.getTrack().getPoint(0),
- inTrackInfo.getTrack().getPoint(1));
- }
- else if (inTrackInfo.getSelection().hasRangeSelected())
- {
- // Use first and last point of selection
- final int startIndex = inTrackInfo.getSelection().getStart();
- final int endIndex = inTrackInfo.getSelection().getEnd();
- return generateGraphHopperUrl(inTrackInfo.getTrack().getPoint(startIndex),
- inTrackInfo.getTrack().getPoint(endIndex));
- }
- }
- return null;
- }
-
- /**
- * Generate a url for routing with GraphHopper
- * @param inStartPoint start point of routing
- * @param inEndPoint end point of routing
- * @return URL
- */
- private static String generateGraphHopperUrl(DataPoint inStartPoint, DataPoint inEndPoint)
- {
- final String url = "https://graphhopper.com/maps/"
- + "?point=" + FIVE_DP.format(inStartPoint.getLatitude().getDouble())
- + "%2C" + FIVE_DP.format(inStartPoint.getLongitude().getDouble())
- + "&point=" + FIVE_DP.format(inEndPoint.getLatitude().getDouble())
- + "%2C" + FIVE_DP.format(inEndPoint.getLongitude().getDouble())
- + "&locale=" + I18nManager.getText("wikipedia.lang")
- + "&elevation=true&weighting=fastest";
- return url;
- }
-
- /**
- * Generate a url for Open Street Map
- * @param inTrackInfo track information
- * @return URL
- */
- private static String generateOpenStreetMapUrl(TrackInfo inTrackInfo)
- {
- // Check if any data to display
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- DoubleRange latRange = inTrackInfo.getTrack().getLatRange();
- DoubleRange lonRange = inTrackInfo.getTrack().getLonRange();
- // Build basic url using min and max lat and long
- String url = "http://openstreetmap.org/?minlat=" + FIVE_DP.format(latRange.getMinimum())
- + "&maxlat=" + FIVE_DP.format(latRange.getMaximum())
- + "&minlon=" + FIVE_DP.format(lonRange.getMinimum()) + "&maxlon=" + FIVE_DP.format(lonRange.getMaximum());
- DataPoint currPoint = inTrackInfo.getCurrentPoint();
- // Add selected point, if any (no way to add point name?)
- if (currPoint != null) {
- url = url + "&mlat=" + FIVE_DP.format(currPoint.getLatitude().getDouble())
- + "&mlon=" + FIVE_DP.format(currPoint.getLongitude().getDouble());
- }
- return url;
- }
-
- /**
- * Generate a URL which only needs the current point
- * This is just a helper method to simplify the calls to the service-specific methods
- * @param inSource service to call
- * @param inTrackInfo track info
- * @return URL if available, or null
- */
- private static String generateUrlForPoint(WebService inService, TrackInfo inTrackInfo)
- {
- if (inTrackInfo == null || inTrackInfo.getTrack() == null || inTrackInfo.getTrack().getNumPoints() < 1)
- {
- return null;
- }
- // Need a current point
- DataPoint currPoint = inTrackInfo.getCurrentPoint();
- if (currPoint == null)
- {
- return null;
- }
- switch (inService)
- {
- case MAP_SOURCE_PEAKFINDER:
- return generatePeakfinderUrl(currPoint);
- case MAP_SOURCE_GEOHACK:
- return generateGeohackUrl(currPoint);
- case MAP_SOURCE_INLINESKATE:
- return generateInlinemapUrl(currPoint);
- default:
- return null;
- }
- }
-
-
- /**
- * Generate a url for PeakFinder
- * @param inPoint current point, not null
- * @return URL
- */
- private static String generatePeakfinderUrl(DataPoint inPoint)
- {
- return "https://www.peakfinder.org/?lat=" + FIVE_DP.format(inPoint.getLatitude().getDouble())
- + "&lng=" + FIVE_DP.format(inPoint.getLongitude().getDouble());
- }
-
- /**
- * Generate a url for Geohack
- * @param inPoint current point, not null
- * @return URL
- */
- private static String generateGeohackUrl(DataPoint inPoint)
- {
- return "https://tools.wmflabs.org/geohack/geohack.php?params=" + FIVE_DP.format(inPoint.getLatitude().getDouble())
- + "_N_" + FIVE_DP.format(inPoint.getLongitude().getDouble()) + "_E";
- // Note: Could use absolute values and S, W but this seems to work
- }
-
- /**
- * Generate a url for Inlinemap.net
- * @param inPoint current point, not null
- * @return URL
- */
- private static String generateInlinemapUrl(DataPoint inPoint)
- {
- return "http://www.inlinemap.net/en/?tab=new#/z14/" + FIVE_DP.format(inPoint.getLatitude().getDouble())
- + "," + FIVE_DP.format(inPoint.getLongitude().getDouble()) + "/terrain";
- }
-
- /**
- * Get the median value from the given lat/long range
- * @param inRange range of values
- * @return median value
- */
- private static double getMedianValue(DoubleRange inRange)
- {
- return (inRange.getMaximum() + inRange.getMinimum()) / 2.0;
- }
-
- /**
- * Get the span of the given lat/long range
- * @param inRange range of values
- * @return span
- */
- private static double getSpan(DoubleRange inRange)
- {
- return inRange.getMaximum() - inRange.getMinimum();
- }
-}