diff --git a/android/jni/com/mapswithme/maps/Framework.cpp b/android/jni/com/mapswithme/maps/Framework.cpp index dc3fa8855c..05ea5ffdb7 100644 --- a/android/jni/com/mapswithme/maps/Framework.cpp +++ b/android/jni/com/mapswithme/maps/Framework.cpp @@ -51,6 +51,8 @@ #include "base/math.hpp" #include "base/sunrise_sunset.hpp" +#include "3party/open-location-code/openlocationcode.h" + #include #include #include @@ -918,29 +920,20 @@ Java_com_mapswithme_maps_Framework_nativeGetDistanceAndAzimuthFromLatLon( } JNIEXPORT jobject JNICALL -Java_com_mapswithme_maps_Framework_nativeFormatLatLon(JNIEnv * env, jclass, jdouble lat, jdouble lon, jboolean useDMSFormat) +Java_com_mapswithme_maps_Framework_nativeFormatLatLon(JNIEnv * env, jclass, jdouble lat, jdouble lon, int coordsFormat) { - return jni::ToJavaString( - env, (useDMSFormat ? measurement_utils::FormatLatLonAsDMS(lat, lon, 2) - : measurement_utils::FormatLatLon(lat, lon, true /* withSemicolon */, 6))); -} - -JNIEXPORT jobjectArray JNICALL -Java_com_mapswithme_maps_Framework_nativeFormatLatLonToArr(JNIEnv * env, jclass, jdouble lat, jdouble lon, jboolean useDMSFormat) -{ - string slat, slon; - if (useDMSFormat) - measurement_utils::FormatLatLonAsDMS(lat, lon, slat, slon, 2); - else - measurement_utils::FormatLatLon(lat, lon, slat, slon, 6); - - static jclass const klass = jni::GetGlobalClassRef(env, "java/lang/String"); - jobjectArray arr = env->NewObjectArray(2, klass, 0); - - env->SetObjectArrayElement(arr, 0, jni::ToJavaString(env, slat)); - env->SetObjectArrayElement(arr, 1, jni::ToJavaString(env, slon)); - - return arr; + switch (coordsFormat) // See CoordinatesFormat enum for all possible values. + { + default: + case 0: // DMS, comma separated + return jni::ToJavaString(env, measurement_utils::FormatLatLonAsDMS(lat, lon, true /*withComma*/, 2)); + case 1: // Decimal, comma separated + return jni::ToJavaString(env, measurement_utils::FormatLatLon(lat, lon, true /* withComma */, 6)); + case 2: // Open location code, long format + return jni::ToJavaString(env, openlocationcode::Encode({lat, lon})); + case 3: // Link to osm.org + return jni::ToJavaString(env, measurement_utils::FormatOsmLink(lat, lon, 14)); + } } JNIEXPORT jobject JNICALL diff --git a/android/src/com/mapswithme/maps/Framework.java b/android/src/com/mapswithme/maps/Framework.java index 41ed092ee6..be1c945b84 100644 --- a/android/src/com/mapswithme/maps/Framework.java +++ b/android/src/com/mapswithme/maps/Framework.java @@ -207,10 +207,7 @@ public class Framework public static native DistanceAndAzimut nativeGetDistanceAndAzimuthFromLatLon(double dstLat, double dstLon, double srcLat, double srcLon, double north); - public static native String nativeFormatLatLon(double lat, double lon, boolean useDmsFormat); - - @Size(2) - public static native String[] nativeFormatLatLonToArr(double lat, double lon, boolean useDmsFormat); + public static native String nativeFormatLatLon(double lat, double lon, int coordFormat); public static native String nativeFormatAltitude(double alt); diff --git a/android/src/com/mapswithme/maps/routing/RoutingController.java b/android/src/com/mapswithme/maps/routing/RoutingController.java index bcf4044af5..81113a60c2 100644 --- a/android/src/com/mapswithme/maps/routing/RoutingController.java +++ b/android/src/com/mapswithme/maps/routing/RoutingController.java @@ -23,6 +23,7 @@ import com.mapswithme.maps.base.Initializable; import com.mapswithme.maps.bookmarks.data.FeatureId; import com.mapswithme.maps.bookmarks.data.MapObject; import com.mapswithme.maps.location.LocationHelper; +import com.mapswithme.maps.widget.placepage.CoordinatesFormat; import com.mapswithme.util.Config; import com.mapswithme.util.StringUtils; import com.mapswithme.util.Utils; @@ -933,7 +934,7 @@ public class RoutingController implements Initializable } else { - title = Framework.nativeFormatLatLon(point.getLat(), point.getLon(), false /* useDmsFormat */); + title = Framework.nativeFormatLatLon(point.getLat(), point.getLon(), CoordinatesFormat.LatLonDecimal.getId()); } } return new Pair<>(title, subtitle); diff --git a/android/src/com/mapswithme/maps/widget/placepage/CoordinatesFormat.java b/android/src/com/mapswithme/maps/widget/placepage/CoordinatesFormat.java new file mode 100644 index 0000000000..533bc37818 --- /dev/null +++ b/android/src/com/mapswithme/maps/widget/placepage/CoordinatesFormat.java @@ -0,0 +1,32 @@ +package com.mapswithme.maps.widget.placepage; + +public enum CoordinatesFormat +{ + LatLonDMS(0), // Latitude, Longitude in degrees minutes seconds format, comma separated + LatLonDecimal(1), // Latitude, Longitude in decimal format, comma separated + OLCFull(2), // Open location code, full format + OSMLink(3); // Link to the OSM. E.g. https://osm.org/go/xcXjyqQlq-?m= + + private final int id; + + CoordinatesFormat(int id) + { + this.id = id; + } + + public int getId() + { + return id; + } + + public static CoordinatesFormat fromId(int id) + { + for (CoordinatesFormat cursor: CoordinatesFormat.values()) + { + if (cursor.id == id) + return cursor; + } + + throw new IllegalArgumentException(); + } +} diff --git a/android/src/com/mapswithme/maps/widget/placepage/PlacePageView.java b/android/src/com/mapswithme/maps/widget/placepage/PlacePageView.java index ea3120717d..7f77583958 100644 --- a/android/src/com/mapswithme/maps/widget/placepage/PlacePageView.java +++ b/android/src/com/mapswithme/maps/widget/placepage/PlacePageView.java @@ -67,6 +67,7 @@ import com.mapswithme.util.log.Logger; import com.mapswithme.util.log.LoggerFactory; import java.util.ArrayList; +import java.util.Arrays; import java.util.Calendar; import java.util.List; import java.util.Locale; @@ -81,6 +82,11 @@ public class PlacePageView extends NestedScrollViewClickFixed private static final Logger LOGGER = LoggerFactory.INSTANCE.getLogger(LoggerFactory.Type.MISC); private static final String TAG = PlacePageView.class.getSimpleName(); private static final String PREF_USE_DMS = "use_dms"; + private static final List visibleCoordsFormat = + Arrays.asList(CoordinatesFormat.LatLonDMS, + CoordinatesFormat.LatLonDecimal, + CoordinatesFormat.OLCFull, + CoordinatesFormat.OSMLink); private boolean mIsDocked; private boolean mIsFloating; @@ -166,7 +172,7 @@ public class PlacePageView extends NestedScrollViewClickFixed // Data @Nullable private MapObject mMapObject; - private boolean mIsLatLonDms; + private CoordinatesFormat mCoordsFormat = CoordinatesFormat.LatLonDecimal; // Downloader`s stuff private DownloaderStatusIcon mDownloaderIcon; @@ -283,7 +289,7 @@ public class PlacePageView extends NestedScrollViewClickFixed public PlacePageView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs); - mIsLatLonDms = MwmApplication.prefs(context).getBoolean(PREF_USE_DMS, false); + mCoordsFormat = CoordinatesFormat.fromId(MwmApplication.prefs(context).getInt(PREF_USE_DMS, CoordinatesFormat.LatLonDecimal.getId())); init(attrs, defStyleAttr); } @@ -1156,9 +1162,8 @@ public class PlacePageView extends NestedScrollViewClickFixed { final double lat = mapObject.getLat(); final double lon = mapObject.getLon(); - final String[] latLon = Framework.nativeFormatLatLonToArr(lat, lon, mIsLatLonDms); - if (latLon.length == 2) - mTvLatlon.setText(String.format(Locale.US, "%1$s, %2$s", latLon[0], latLon[1])); + final String latLon = Framework.nativeFormatLatLon(lat, lon, mCoordsFormat.getId()); + mTvLatlon.setText(latLon); } private static void refreshMetadataOrHide(String metadata, View metaLayout, TextView metaTv) @@ -1224,8 +1229,9 @@ public class PlacePageView extends NestedScrollViewClickFixed addPlace(); break; case R.id.ll__place_latlon: - mIsLatLonDms = !mIsLatLonDms; - MwmApplication.prefs(getContext()).edit().putBoolean(PREF_USE_DMS, mIsLatLonDms).apply(); + final int formatIndex = visibleCoordsFormat.indexOf(mCoordsFormat); + mCoordsFormat = visibleCoordsFormat.get((formatIndex + 1) % visibleCoordsFormat.size()); + MwmApplication.prefs(getContext()).edit().putInt(PREF_USE_DMS, mCoordsFormat.getId()).apply(); if (mMapObject == null) { LOGGER.e(TAG, "A LatLon cannot be refreshed, mMapObject is null"); @@ -1305,8 +1311,8 @@ public class PlacePageView extends NestedScrollViewClickFixed } final double lat = mMapObject.getLat(); final double lon = mMapObject.getLon(); - items.add(Framework.nativeFormatLatLon(lat, lon, false)); - items.add(Framework.nativeFormatLatLon(lat, lon, true)); + for(CoordinatesFormat format: visibleCoordsFormat) + items.add(Framework.nativeFormatLatLon(lat, lon, format.getId())); break; case R.id.ll__place_website: items.add(mTvWebsite.getText().toString()); diff --git a/map/place_page_info.cpp b/map/place_page_info.cpp index bbbd5dc3c1..bc2be5dab1 100644 --- a/map/place_page_info.cpp +++ b/map/place_page_info.cpp @@ -214,7 +214,7 @@ void Info::SetCustomNameWithCoordinates(m2::PointD const & mercator, std::string m_uiTitle = name; m_uiSubtitle = measurement_utils::FormatLatLon( mercator::YToLat(mercator.y), mercator::XToLon(mercator.x), - true /* withSemicolon */); + true /* withComma */); } m_customName = name; } @@ -279,7 +279,7 @@ std::string Info::GetFormattedCoordinate(bool isDMS) const { auto const & ll = GetLatLon(); return isDMS ? measurement_utils::FormatLatLon(ll.m_lat, ll.m_lon, true) - : measurement_utils::FormatLatLonAsDMS(ll.m_lat, ll.m_lon, 2); + : measurement_utils::FormatLatLonAsDMS(ll.m_lat, ll.m_lon, false, 2); } void Info::SetRoadType(RoadWarningMarkType type, std::string const & localizedType, std::string const & distance) diff --git a/platform/measurement_utils.cpp b/platform/measurement_utils.cpp index c61661b8e4..fe6b2647ba 100644 --- a/platform/measurement_utils.cpp +++ b/platform/measurement_utils.cpp @@ -143,9 +143,9 @@ string FormatLatLonAsDMSImpl(double value, char positive, char negative, int dac return sstream.str(); } -string FormatLatLonAsDMS(double lat, double lon, int dac) +string FormatLatLonAsDMS(double lat, double lon, bool withComma, int dac) { - return (FormatLatLonAsDMSImpl(lat, 'N', 'S', dac) + " " + + return (FormatLatLonAsDMSImpl(lat, 'N', 'S', dac) + (withComma ? ", " : " ") + FormatLatLonAsDMSImpl(lon, 'E', 'W', dac)); } @@ -172,9 +172,9 @@ string FormatLatLon(double lat, double lon, int dac) return to_string_dac(lat, dac) + " " + to_string_dac(lon, dac); } -string FormatLatLon(double lat, double lon, bool withSemicolon, int dac) +string FormatLatLon(double lat, double lon, bool withComma, int dac) { - return to_string_dac(lat, dac) + (withSemicolon ? ", " : " ") + to_string_dac(lon, dac); + return to_string_dac(lat, dac) + (withComma ? ", " : " ") + to_string_dac(lon, dac); } void FormatLatLon(double lat, double lon, string & latText, string & lonText, int dac) @@ -245,6 +245,40 @@ string FormatSpeedUnits(Units units) UNREACHABLE(); } +// Interleaves the bits of two 32-bit numbers. +// The result is known as a Morton code. +long interleave_bits(long x, long y) +{ + long c = 0; + for(int i=31; i>=0; i--) + { + c = (c << 1) | ((x >> i) & 1); + c = (c << 1) | ((y >> i) & 1); + } + + return c; +} + +string FormatOsmLink(double lat, double lon, int zoom) +{ + long x = round((lon + 180.0) / 360.0 * (1L<<32)); + long y = round((lat + 90.0) / 180.0 * (1L<<32)); + long code = interleave_bits(x, y); + string encodedCoords = ""; + char char_array[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_~"; + int i; + + for (i = 0; i < ceil((zoom + 8) / 3.0); ++i) { + int digit = (code >> (58 - (6 * i))) & 0x3f; + encodedCoords += char_array[digit]; + } + + for (i = 0; i < ((zoom + 8) % 3); ++i) + encodedCoords += "-"; + + return "https://osm.org/go/" + encodedCoords + "?m="; +} + bool OSMDistanceToMeters(string const & osmRawValue, double & outMeters) { using strings::is_finite; diff --git a/platform/measurement_utils.hpp b/platform/measurement_utils.hpp index c48af4d6d5..aaf00d1fbb 100644 --- a/platform/measurement_utils.hpp +++ b/platform/measurement_utils.hpp @@ -50,7 +50,7 @@ std::string FormatSpeedUnits(Units units); /// @param[in] dac Digits after comma in seconds. /// Use dac == 3 for our common conversions to DMS. -std::string FormatLatLonAsDMS(double lat, double lon, int dac = 3); +std::string FormatLatLonAsDMS(double lat, double lon, bool withComma, int dac = 3); void FormatLatLonAsDMS(double lat, double lon, std::string & latText, std::string & lonText, int dac = 3); std::string FormatMercatorAsDMS(m2::PointD const & mercator, int dac = 3); @@ -59,12 +59,14 @@ void FormatMercatorAsDMS(m2::PointD const & mercator, std::string & lat, std::st /// Default dac == 6 for the simple decimal formatting. std::string FormatLatLon(double lat, double lon, int dac = 6); -std::string FormatLatLon(double lat, double lon, bool withSemicolon, int dac = 6); +std::string FormatLatLon(double lat, double lon, bool withComma, int dac = 6); void FormatLatLon(double lat, double lon, std::string & latText, std::string & lonText, int dac = 6); std::string FormatMercator(m2::PointD const & mercator, int dac = 6); void FormatMercator(m2::PointD const & mercator, std::string & lat, std::string & lon, int dac = 6); +std::string FormatOsmLink(double lat, double lon, int zoom); + /// Converts OSM distance (height, ele etc.) to meters. /// @returns false if fails. bool OSMDistanceToMeters(std::string const & osmRawValue, double & outMeters); diff --git a/platform/platform_tests/measurement_tests.cpp b/platform/platform_tests/measurement_tests.cpp index 0998b2c9c5..a8005668ca 100644 --- a/platform/platform_tests/measurement_tests.cpp +++ b/platform/platform_tests/measurement_tests.cpp @@ -65,29 +65,39 @@ UNIT_TEST(Measurement_Smoke) UNIT_TEST(LatLonToDMS_Origin) { - TEST_EQUAL(FormatLatLonAsDMS(0, 0), "00°00′00″ 00°00′00″", ()); - TEST_EQUAL(FormatLatLonAsDMS(0, 0, 3), "00°00′00″ 00°00′00″", ()); + TEST_EQUAL(FormatLatLonAsDMS(0, 0, false), "00°00′00″ 00°00′00″", ()); + TEST_EQUAL(FormatLatLonAsDMS(0, 0, false, 3), "00°00′00″ 00°00′00″", ()); } UNIT_TEST(LatLonToDMS_Rounding) { // Here and after data is from Wiki: http://bit.ly/datafotformatingtest // Boston - TEST_EQUAL(FormatLatLonAsDMS(42.358056, -71.063611, 0), "42°21′29″N 71°03′49″W", ()); + TEST_EQUAL(FormatLatLonAsDMS(42.358056, -71.063611, false, 0), "42°21′29″N 71°03′49″W", ()); // Minsk - TEST_EQUAL(FormatLatLonAsDMS(53.916667, 27.55, 0), "53°55′00″N 27°33′00″E", ()); + TEST_EQUAL(FormatLatLonAsDMS(53.916667, 27.55, false, 0), "53°55′00″N 27°33′00″E", ()); // Rio - TEST_EQUAL(FormatLatLonAsDMS(-22.908333, -43.196389, 0), "22°54′30″S 43°11′47″W", ()); + TEST_EQUAL(FormatLatLonAsDMS(-22.908333, -43.196389, false, 0), "22°54′30″S 43°11′47″W", ()); } UNIT_TEST(LatLonToDMS_NoRounding) { // Paris - TEST_EQUAL(FormatLatLonAsDMS(48.8567, 2.3508, 2), "48°51′24.12″N 02°21′02.88″E", ()); + TEST_EQUAL(FormatLatLonAsDMS(48.8567, 2.3508, false, 2), "48°51′24.12″N 02°21′02.88″E", ()); // Vatican - TEST_EQUAL(FormatLatLonAsDMS(41.904, 12.453, 2), "41°54′14.4″N 12°27′10.8″E", ()); + TEST_EQUAL(FormatLatLonAsDMS(41.904, 12.453, false, 2), "41°54′14.4″N 12°27′10.8″E", ()); - TEST_EQUAL(FormatLatLonAsDMS(21.981112, -159.371112, 2), "21°58′52″N 159°22′16″W", ()); + TEST_EQUAL(FormatLatLonAsDMS(21.981112, -159.371112, false, 2), "21°58′52″N 159°22′16″W", ()); +} + +UNIT_TEST(FormatOsmLink) +{ + //Zero point + TEST_EQUAL(FormatOsmLink(0, 0, 5), "https://osm.org/go/wAAAA-?m=", ()); + //Eifel tower + TEST_EQUAL(FormatOsmLink(48.85825, 2.29450, 15), "https://osm.org/go/0BOdUs9e--?m=", ()); + //Buenos Aires + TEST_EQUAL(FormatOsmLink(-34.6061, -58.4360, 10), "https://osm.org/go/Mnx6SB?m=", ()); } UNIT_TEST(FormatAltitude) diff --git a/qt/bookmark_dialog.cpp b/qt/bookmark_dialog.cpp index 1bbd7fbad2..6bd892df83 100644 --- a/qt/bookmark_dialog.cpp +++ b/qt/bookmark_dialog.cpp @@ -286,7 +286,7 @@ void BookmarkDialog::FillTree() { name = measurement_utils::FormatLatLon(mercator::YToLat(bookmark->GetPivot().y), mercator::XToLon(bookmark->GetPivot().x), - true /* withSemicolon */); + true /* withComma */); } auto bookmarkItem = CreateTreeItem(name + " (Bookmark)", categoryItem); bookmarkItem->setTextColor(0, Qt::blue);