Introduced optional<string> as return type. Updated tests.

Signed-off-by: S. Kozyr <s.trump@gmail.com>
This commit is contained in:
Sergiy Kozyr 2023-05-30 10:29:17 +03:00
parent 6a23412a5e
commit 75b9b134eb
Signed by: strump
GPG key ID: C622E5563CAC205D
9 changed files with 266 additions and 379 deletions

View file

@ -962,9 +962,9 @@ Java_app_organicmaps_Framework_nativeFormatLatLon(JNIEnv * env, jclass, jdouble
case android::CoordinatesFormat::OSMLink: // Link to osm.org
return jni::ToJavaString(env, measurement_utils::FormatOsmLink(lat, lon, 14));
case android::CoordinatesFormat::UTM: // Universal Transverse Mercator
return jni::ToJavaString(env, utm_mgrs_utils::FormatUTM(lat, lon));
return jni::ToJavaString(env, utm_mgrs_utils::FormatUTM(lat, lon).value_or(nullptr));
case android::CoordinatesFormat::MGRS: // Military Grid Reference System
return jni::ToJavaString(env, utm_mgrs_utils::FormatMGRS(lat, lon, 5));
return jni::ToJavaString(env, utm_mgrs_utils::FormatMGRS(lat, lon, 5).value_or(nullptr));
}
}

View file

@ -99,6 +99,24 @@ std::string DebugPrintSequence(IterT beg, IterT end)
return out.str();
}
template <typename T>
std::string DebugPrint(std::optional<T> const & p)
{
if (p)
{
std::ostringstream out;
out << "optional(" << DebugPrint(p.value()) << ")";
return out.str();
}
else
return "nullopt";
}
std::string inline DebugPrint(std::nullopt_t const & p)
{
return "nullopt";
}
template <typename T, size_t N> inline std::string DebugPrint(T (&arr) [N])
{
return DebugPrintSequence(arr, arr + N);

View file

@ -6,63 +6,65 @@
using namespace utm_mgrs_utils;
#define OPT(a) std::make_optional(a)
UNIT_TEST(FormatUTM)
{
TEST_EQUAL(FormatUTM(42.0, -93.0), "15N 500000 4649776", ());
TEST_EQUAL(FormatUTM(31.77597, 35.23406), "36N 711554 3517777", ());
TEST_EQUAL(FormatUTM(-34.81449, -58.54016), "21S 359135 6146448", ());
TEST_EQUAL(FormatUTM(36.2361322, -115.0820944), "11N 672349 4011845", ());
TEST_EQUAL(FormatUTM(42.0, -93.0), OPT("15N 500000 4649776"), ());
TEST_EQUAL(FormatUTM(31.77597, 35.23406), OPT("36N 711554 3517777"), ());
TEST_EQUAL(FormatUTM(-34.81449, -58.54016), OPT("21S 359135 6146448"), ());
TEST_EQUAL(FormatUTM(36.2361322, -115.0820944), OPT("11N 672349 4011845"), ());
TEST_EQUAL(FormatUTM(50.77535, 6.08389), "32N 294409 5628898", ());
TEST_EQUAL(FormatUTM(40.71435, -74.00597), "18N 583960 4507523", ());
TEST_EQUAL(FormatUTM(-41.28646, 174.77624), "60S 313784 5427057", ());
TEST_EQUAL(FormatUTM(-33.92487, 18.42406), "34S 261878 6243186", ());
TEST_EQUAL(FormatUTM(-32.89018, -68.84405), "19S 514586 6360877", ());
TEST_EQUAL(FormatUTM(64.83778, -147.71639), "6N 466013 7190568", ());
TEST_EQUAL(FormatUTM(56.79680, -5.00601), "30N 377486 6296562", ());
TEST_EQUAL(FormatUTM(84, -5.00601), "30N 476594 9328501", ());
TEST_EQUAL(FormatUTM(50.77535, 6.08389), OPT("32N 294409 5628898"), ());
TEST_EQUAL(FormatUTM(40.71435, -74.00597), OPT("18N 583960 4507523"), ());
TEST_EQUAL(FormatUTM(-41.28646, 174.77624), OPT("60S 313784 5427057"), ());
TEST_EQUAL(FormatUTM(-33.92487, 18.42406), OPT("34S 261878 6243186"), ());
TEST_EQUAL(FormatUTM(-32.89018, -68.84405), OPT("19S 514586 6360877"), ());
TEST_EQUAL(FormatUTM(64.83778, -147.71639), OPT("6N 466013 7190568"), ());
TEST_EQUAL(FormatUTM(56.79680, -5.00601), OPT("30N 377486 6296562"), ());
TEST_EQUAL(FormatUTM(84, -5.00601), OPT("30N 476594 9328501"), ());
TEST_EQUAL(FormatUTM(12.016469, 188.0), std::nullopt, ());
}
UNIT_TEST(FormatMGRS)
{
TEST_EQUAL(FormatMGRS(42.0, -93.0, 5), "15T WG 00000 49776", ());
TEST_EQUAL(FormatMGRS(31.77597, 35.23406, 5), "36R YA 11554 17776", ());
TEST_EQUAL(FormatMGRS(-34.81449, -58.54016, 5), "21H UB 59135 46447", ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 5), "11S PA 72349 11844", ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 4), "11S PA 7234 1184", ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 3), "11S PA 723 118", ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 2), "11S PA 72 11", ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 1), "11S PA 7 1", ());
TEST_EQUAL(FormatMGRS(42.0, -93.0, 5), OPT("15T WG 00000 49776"), ());
TEST_EQUAL(FormatMGRS(31.77597, 35.23406, 5), OPT("36R YA 11554 17776"), ());
TEST_EQUAL(FormatMGRS(-34.81449, -58.54016, 5), OPT("21H UB 59135 46447"), ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 5), OPT("11S PA 72349 11844"), ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 4), OPT("11S PA 7234 1184"), ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 3), OPT("11S PA 723 118"), ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 2), OPT("11S PA 72 11"), ());
TEST_EQUAL(FormatMGRS(36.2361322, -115.0820944, 1), OPT("11S PA 7 1"), ());
TEST_EQUAL(FormatMGRS(84.644103, 3.000009, 5), "Latitude limit exceeded", ()); // Some converters generate string "Z AB 31142 05767"
TEST_EQUAL(FormatMGRS(-81.016469, 8.745519, 5), "Latitude limit exceeded", ()); // Some converters generate string "B BX 51947 87732"
TEST_EQUAL(FormatMGRS(12.016469, 188.0, 2), "Longitude limit exceeded", ());
TEST_EQUAL(FormatMGRS(84.644103, 3.000009, 5), std::nullopt, ()); // Some converters generate string "Z AB 31142 05767"
TEST_EQUAL(FormatMGRS(-81.016469, 8.745519, 5), std::nullopt, ()); // Some converters generate string "B BX 51947 87732"
TEST_EQUAL(FormatMGRS(12.016469, 188.0, 2), std::nullopt, ());
// Test data from https://s3.amazonaws.com/mgrs.io/mgrsToGeo_WE.txt
TEST_EQUAL(FormatMGRS(0.000005, 0.304981, 5), "31N BA 00000 00000", ());
TEST_EQUAL(FormatMGRS(4.518520, 48.296629, 5), "39N TF 00000 00000", ());
TEST_EQUAL(FormatMGRS(9.045438, 50.090125, 5), "39P VL 00000 00000", ());
TEST_EQUAL(FormatMGRS(63.115494, 91.017708, 5), "46V DR 00000 00000", ());
TEST_EQUAL(FormatMGRS(40.626644, 137.364687, 5), "53T QF 00000 00000", ());
TEST_EQUAL(FormatMGRS(0.000005, -4.797048, 5), "30N UF 00000 00000", ());
TEST_EQUAL(FormatMGRS(0.000005, -0.592324, 5), "30N YF 67993 00000", ());
TEST_EQUAL(FormatMGRS(0.000005, 6.592333, 5), "32N KF 32007 00000", ());
TEST_EQUAL(FormatMGRS(0.000005, 54.592333, 5), "40N BF 32007 00000", ());
TEST_EQUAL(FormatMGRS(76.463950, 58.675211, 5), "40X EK 43763 87577", ());
TEST_EQUAL(FormatMGRS(-49.578078, 85.150396, 5), "45F UF 66291 06635", ());
TEST_EQUAL(FormatMGRS(-45.089793, 132.812340, 5), "53G LL 27848 04746", ());
TEST_EQUAL(FormatMGRS(4.514602, 138.603457, 5), "54N TK 34069 99447", ());
TEST_EQUAL(FormatMGRS(-67.547521, -142.303616, 5), "07D DF 44443 06997", ());
TEST_EQUAL(FormatMGRS(-62.908852, -154.887008, 5), "05E ML 04130 23160", ());
TEST_EQUAL(FormatMGRS(4.514602, -144.603447, 5), "06N YK 65931 99447", ());
TEST_EQUAL(FormatMGRS(4.514602, -137.396543, 5), "08N KK 34069 99447", ());
TEST_EQUAL(FormatMGRS(9.028532, -144.637149, 5), "06P YQ 59760 98847", ());
TEST_EQUAL(FormatMGRS(54.109208, -60.059588, 5), "20U PE 92211 99669", ());
TEST_EQUAL(FormatMGRS(54.109208, -53.940397, 5), "22U CE 07789 99669", ());
TEST_EQUAL(FormatMGRS(-45.089793, -53.187660, 5), "22G CR 27848 04746", ());
TEST_EQUAL(FormatMGRS(-31.596034, -18.161583, 5), "27J YF 69322 00842", ());
TEST_EQUAL(FormatMGRS(-22.559756, -11.111475, 5), "29K KR 82882 03678", ());
TEST_EQUAL(FormatMGRS(0.000005, 0.592333, 5), "31N BA 32007 00000", ());
TEST_EQUAL(FormatMGRS(0.000005, 0.304981, 5), OPT("31N BA 00000 00000"), ());
TEST_EQUAL(FormatMGRS(4.518520, 48.296629, 5), OPT("39N TF 00000 00000"), ());
TEST_EQUAL(FormatMGRS(9.045438, 50.090125, 5), OPT("39P VL 00000 00000"), ());
TEST_EQUAL(FormatMGRS(63.115494, 91.017708, 5), OPT("46V DR 00000 00000"), ());
TEST_EQUAL(FormatMGRS(40.626644, 137.364687, 5), OPT("53T QF 00000 00000"), ());
TEST_EQUAL(FormatMGRS(0.000005, -4.797048, 5), OPT("30N UF 00000 00000"), ());
TEST_EQUAL(FormatMGRS(0.000005, -0.592324, 5), OPT("30N YF 67993 00000"), ());
TEST_EQUAL(FormatMGRS(0.000005, 6.592333, 5), OPT("32N KF 32007 00000"), ());
TEST_EQUAL(FormatMGRS(0.000005, 54.592333, 5), OPT("40N BF 32007 00000"), ());
TEST_EQUAL(FormatMGRS(76.463950, 58.675211, 5), OPT("40X EK 43763 87577"), ());
TEST_EQUAL(FormatMGRS(-49.578078, 85.150396, 5), OPT("45F UF 66291 06635"), ());
TEST_EQUAL(FormatMGRS(-45.089793, 132.812340, 5), OPT("53G LL 27848 04746"), ());
TEST_EQUAL(FormatMGRS(4.514602, 138.603457, 5), OPT("54N TK 34069 99447"), ());
TEST_EQUAL(FormatMGRS(-67.547521, -142.303616, 5), OPT("07D DF 44443 06997"), ());
TEST_EQUAL(FormatMGRS(-62.908852, -154.887008, 5), OPT("05E ML 04130 23160"), ());
TEST_EQUAL(FormatMGRS(4.514602, -144.603447, 5), OPT("06N YK 65931 99447"), ());
TEST_EQUAL(FormatMGRS(4.514602, -137.396543, 5), OPT("08N KK 34069 99447"), ());
TEST_EQUAL(FormatMGRS(9.028532, -144.637149, 5), OPT("06P YQ 59760 98847"), ());
TEST_EQUAL(FormatMGRS(54.109208, -60.059588, 5), OPT("20U PE 92211 99669"), ());
TEST_EQUAL(FormatMGRS(54.109208, -53.940397, 5), OPT("22U CE 07789 99669"), ());
TEST_EQUAL(FormatMGRS(-45.089793, -53.187660, 5), OPT("22G CR 27848 04746"), ());
TEST_EQUAL(FormatMGRS(-31.596034, -18.161583, 5), OPT("27J YF 69322 00842"), ());
TEST_EQUAL(FormatMGRS(-22.559756, -11.111475, 5), OPT("29K KR 82882 03678"), ());
TEST_EQUAL(FormatMGRS(0.000005, 0.592333, 5), OPT("31N BA 32007 00000"), ());
}

View file

@ -11,7 +11,8 @@ namespace utm_mgrs_utils
{
using namespace std;
using namespace math; // To use constexpr math::pi
using namespace base; // To use constexpr base::DegToRad and base::RadToDeg
using namespace base; // To use base::DegToRad and base::RadToDeg
using namespace ms; // To use ms::LatLon
typedef struct UTMPoint_Value
{
@ -46,13 +47,15 @@ constexpr double P3 = (21.0 / 16.0 * _E2 - 55.0 / 32.0 * _E4);
constexpr double P4 = (151.0 / 96.0 * _E3 - 417.0 / 128.0 * _E5);
constexpr double P5 = (1097.0 / 512.0 * _E4);
constexpr double kInvalidNorthing = -1.0;
const string ZONE_LETTERS = "CDEFGHJKLMNPQRSTUVWXX";
const int NUM_100K_SETS = 6;
const int SET_ORIGIN_COLUMN_LETTERS[] = { 'A', 'J', 'S', 'A', 'J', 'S' };
const int SET_ORIGIN_ROW_LETTERS[] = { 'A', 'F', 'A', 'F', 'A', 'F' };
const int SET_ORIGIN_COLUMN_LETTERS[] = { 'S', 'A', 'J', 'S', 'A', 'J' };
const int SET_ORIGIN_ROW_LETTERS[] = { 'F', 'A', 'F', 'A', 'F', 'A' };
// Returns angle in radians to be between -PI and PI.
// Returns angle in radians to be between -π and π.
double mod_angle(double value)
{
if (value < -pi)
@ -82,12 +85,12 @@ int latlon_to_zone_number(double lat, double lon)
return int((lon + 180.0) / 6.0) + 1;
}
char latitude_to_zone_letter(double lat)
optional<char> latitude_to_zone_letter(double lat)
{
if (-80.0 <= lat && lat <= 84.0)
return ZONE_LETTERS[int(lat + 80.0) >> 3];
else
return '?';
return nullopt;
}
int zone_number_to_central_longitude(int zone_number)
@ -149,22 +152,15 @@ string utm_to_str(UTMPoint point)
to_string(int(round(point.northing)));
}
int zone_to_100k(int i) {
int set = i % NUM_100K_SETS;
if (set == 0)
set = NUM_100K_SETS;
return set;
}
// Build 2 chars string with 100k square ID
// Build 2 chars string with MGRS 100k designator.
string get_100k_id(double easting, double northing, int zone_number)
{
int set = zone_to_100k(zone_number);
int set = zone_number % NUM_100K_SETS;
int setColumn = ((int) easting / 100000);
int setRow = ((int) northing / 100000) % 20;
int colOrigin = SET_ORIGIN_COLUMN_LETTERS[set - 1];
int rowOrigin = SET_ORIGIN_ROW_LETTERS[set - 1];
int colOrigin = SET_ORIGIN_COLUMN_LETTERS[set];
int rowOrigin = SET_ORIGIN_ROW_LETTERS[set];
int colInt = colOrigin + setColumn - 1;
int rowInt = rowOrigin + setRow;
@ -239,19 +235,19 @@ string utm_to_mgrs_str(UTMPoint point, int precision)
// Convert UTM parameters to lat,lon for WSG 84 ellipsoid.
// If UTM parameters are valid lat and lon references are used to output calculated coordinates.
// Otherwise function returns 'false'.
bool UTMtoLatLon(double easting, double northing, int zone_number, char zone_letter, double &lat, double &lon)
std::optional<ms::LatLon> UTMtoLatLon(double easting, double northing, int zone_number, char zone_letter)
{
if (zone_number < 1 || zone_number > 60)
return false;
return nullopt;
if (easting < 100000.0 || easting >= 1000000.0)
return false;
return nullopt;
if (northing < 0.0 || northing > 10000000.0)
return false;
return nullopt;
if (zone_letter<'C' || zone_letter>'X' || zone_letter == 'I' || zone_letter == 'O')
return false;
return nullopt;
bool northern = (zone_letter >= 'N');
double x = easting - 500000.0;
@ -305,10 +301,7 @@ bool UTMtoLatLon(double easting, double northing, int zone_number, char zone_let
longitude = mod_angle(longitude + DegToRad(zone_number_to_central_longitude(zone_number)));
lat = RadToDeg(latitude);
lon = RadToDeg(longitude);
return true;
return LatLon(RadToDeg(latitude), RadToDeg(longitude));
}
@ -316,7 +309,7 @@ bool UTMtoLatLon(double easting, double northing, int zone_number, char zone_let
* MGRS table set for the zone number, figure out the easting value that
* should be added to the other, secondary easting value.*/
double square_char_to_easting(char e, int set) {
int curCol = SET_ORIGIN_COLUMN_LETTERS[set - 1];
int curCol = SET_ORIGIN_COLUMN_LETTERS[set];
double eastingValue = 100000.0;
bool rewindMarker = false;
@ -330,7 +323,7 @@ double square_char_to_easting(char e, int set) {
if (curCol > 'Z')
{
if (rewindMarker)
return -1;
return kInvalidNorthing;
curCol = 'A';
rewindMarker = true;
}
@ -355,12 +348,12 @@ double square_char_to_easting(char e, int set) {
* @param set
* the MGRS table set number, which is dependent on the UTM zone
* number. */
double square_char_to_northing(char n, int set) {
double square_char_to_northing(char n, int set)
{
if (n > 'V')
return -1;
return kInvalidNorthing;
int curRow = SET_ORIGIN_ROW_LETTERS[set - 1];
int curRow = SET_ORIGIN_ROW_LETTERS[set];
double northingValue = 0.0;
bool rewindMarker = false;
@ -371,12 +364,10 @@ double square_char_to_northing(char n, int set) {
curRow++;
if (curRow == 'O')
curRow++;
// fixing a bug making whole application hang in this loop
// when 'n' is a wrong character
if (curRow > 'V')
{
if (rewindMarker) // making sure that this loop ends
return -1;
if (rewindMarker) // Making sure that this loop ends even if n has invalid value.
return kInvalidNorthing;
curRow = 'A';
rewindMarker = true;
}
@ -387,11 +378,7 @@ double square_char_to_northing(char n, int set) {
}
/* The function getMinNorthing returns the minimum northing value of a MGRS
* zone.
*
* portted from Geotrans' c Latitude_Band_Value structure table. zoneLetter
* : MGRS zone (input). */
// Get minimum northing value of a MGRS zone.
double zone_to_min_northing(char zoneLetter)
{
double northing;
@ -458,41 +445,41 @@ double zone_to_min_northing(char zoneLetter)
northing = 7900000.0;
break;
default:
northing = -1.0;
northing = kInvalidNorthing;
}
return northing;
}
// Convert MGRS parameters to UTM parameters and then use UTM to lat,lon conversion.
bool MGRStoLatLon(double easting, double northing, int zone_code, char zone_letter, char square_code[2], double &lat, double &lon)
std::optional<ms::LatLon> MGRStoLatLon(double easting, double northing, int zone_code, char zone_letter, char square_code[2])
{
// Convert easting and northing according to zone_code and square_code
if (zone_code < 1 || zone_code > 60)
return false;
return nullopt;
if (zone_letter <= 'B' || zone_letter >= 'Y' || zone_letter == 'I' || zone_letter == 'O')
return false;
return nullopt;
int set = zone_to_100k(zone_code);
int set = zone_code % NUM_100K_SETS;
char char1 = square_code[0];
char char2 = square_code[1];
if (char1 < 'A' || char2 < 'A' || char1 > 'Z' || char2 > 'Z' || char1 == 'I' || char2 == 'I' || char1 == 'O' || char2 == 'O')
return false;
return nullopt;
float east100k = square_char_to_easting(char1, set);
if (east100k < 0)
return false;
if (east100k == kInvalidNorthing)
return nullopt;
float north100k = square_char_to_northing(char2, set);
if (north100k < 0)
return false;
if (north100k == kInvalidNorthing)
return nullopt;
double minNorthing = zone_to_min_northing(zone_letter);
if (minNorthing < 0)
return false;
if (minNorthing == kInvalidNorthing)
return nullopt;
while (north100k < minNorthing)
north100k += 2000000.0;
@ -500,11 +487,11 @@ bool MGRStoLatLon(double easting, double northing, int zone_code, char zone_lett
easting += east100k;
northing += north100k;
return UTMtoLatLon(easting, northing, zone_code, zone_letter, lat, lon);
return UTMtoLatLon(easting, northing, zone_code, zone_letter);
}
// Convert lat,lon for WSG 84 ellipsoid to MGRS string.
string FormatMGRS(double lat, double lon, int precision)
// Convert lat,lon for WGS84 ellipsoid to MGRS string.
optional<string> FormatMGRS(double lat, double lon, int precision)
{
if (precision > 5)
precision = 5;
@ -512,24 +499,30 @@ string FormatMGRS(double lat, double lon, int precision)
precision = 1;
if (lat <= -80 || lat > 84)
return "Latitude limit exceeded";
return nullopt; // Latitude limit exceeded.
if (lon <= -180 || lon > 180)
return "Longitude limit exceeded";
return nullopt; // Longitude limit exceeded.
UTMPoint mgrsp = latlon_to_utm(lat, lon);
// Need to add this to set the right letter for the latitude.
mgrsp.zone_letter = latitude_to_zone_letter(lat);
return utm_to_mgrs_str(mgrsp, precision);
// Need to set the right letter for the latitude.
auto maybeZone = latitude_to_zone_letter(lat);
if (maybeZone)
{
mgrsp.zone_letter = maybeZone.value();
return utm_to_mgrs_str(mgrsp, precision);
}
return nullopt;
}
// Convert lat,lon for WSG 84 ellipsoid to UTM string.
string FormatUTM(double lat, double lon)
// Convert lat,lon for WGS84 ellipsoid to UTM string.
optional<string> FormatUTM(double lat, double lon)
{
if (lat <= -80 || lat > 84)
return "Latitude limit exceeded";
return nullopt; // Latitude limit exceeded.
if (lon <= -180 || lon > 180)
return "Longitude limit exceeded";
return nullopt; // Longitude limit exceeded.
UTMPoint utm = latlon_to_utm(lat, lon);
return utm_to_str(utm);

View file

@ -2,10 +2,12 @@
#include <string>
#include "geometry/latlon.hpp"
namespace utm_mgrs_utils
{
// Convert from Lat and Lon coordinates in WGS 84 projection to UTM string
std::string FormatUTM(double lat, double lon);
std::optional<std::string> FormatUTM(double lat, double lon);
/* Convert from Lat and Lon coordinates in WGS 84 projection to MGRS square string
* Parameter prec should be a number from 1 to 5.
@ -15,12 +17,12 @@ std::string FormatUTM(double lat, double lon);
* prec = 4 gives MGRS coordinates "11S PA 7234 1184"
* prec = 5 gives MGRS coordinates "11S PA 72349 11844" (highest precision)
*/
std::string FormatMGRS(double lat, double lon, int prec);
std::optional<std::string> FormatMGRS(double lat, double lon, int prec);
// Covevrt UTM coordinates to Lat Lon. If UTM parameters are invalid function returns false
bool UTMtoLatLon(double easting, double northing, int zone_code, char zone_letter, double &lat, double &lon);
std::optional<ms::LatLon> UTMtoLatLon(double easting, double northing, int zone_code, char zone_letter);
// Covevrt MGRS coordinates to Lat Lon. If parameters are invalid function returns false
bool MGRStoLatLon(double easting, double northing, int zone_code, char zone_letter, char square_code[2], double &lat, double &lon);
std::optional<ms::LatLon> MGRStoLatLon(double easting, double northing, int zone_code, char zone_letter, char square_code[2]);
} // namespace utm_mgrs_utils

View file

@ -617,16 +617,20 @@ bool Processor::SearchCoordinates()
results.emplace_back(lat, lon);
}
if (MatchUTMCoords(m_query, lat, lon))
optional<ms::LatLon> maybeLatLon = MatchUTMCoords(m_query);
if (maybeLatLon)
{
coords_found = true;
results.emplace_back(lat, lon);
auto latLon = maybeLatLon.value();
results.emplace_back(latLon.m_lat, latLon.m_lon);
}
if (MatchMGRSCoords(m_query, lat, lon))
maybeLatLon = MatchMGRSCoords(m_query);
if (maybeLatLon)
{
coords_found = true;
results.emplace_back(lat, lon);
auto latLon = maybeLatLon.value();
results.emplace_back(latLon.m_lat, latLon.m_lon);
}
istringstream iss(m_query);

View file

@ -13,281 +13,147 @@ using namespace search;
// We expect the results to be quite precise.
double const kEps = 1e-5;
void TestAlmostEqual(double actual, double expected)
void TestAlmostEqual(std::optional<ms::LatLon> maybeLatLon, double expectedLat, double expectedLon)
{
TEST(base::AlmostEqualAbsOrRel(actual, expected, kEps), (actual, expected));
TEST(maybeLatLon.has_value(), ());
auto const actualLat = maybeLatLon.value().m_lat;
TEST(base::AlmostEqualAbsOrRel(actualLat, expectedLat, kEps), ("Lat is not close", actualLat, expectedLat));
auto const actualLon = maybeLatLon.value().m_lon;
TEST(base::AlmostEqualAbsOrRel(actualLon, expectedLon, kEps), ("Lon is not close", actualLon, expectedLon));
}
UNIT_TEST(MatchUTMCoords)
{
double lat, lon;
// Extra spaces shouldn't break format
TEST(MatchUTMCoords("15 N 500000 4649776", lat, lon), ());
TEST(MatchUTMCoords("15 N 500000 4649776", lat, lon), ());
TEST(MatchUTMCoords("15 N 500000 4649776").has_value(), ());
TEST(MatchUTMCoords("15 N 500000 4649776").has_value(), ());
TEST(MatchUTMCoords("15N 500000 4649776", lat, lon), ());
TestAlmostEqual(lat, 42.0);
TestAlmostEqual(lon, -93.0);
TEST(MatchUTMCoords("15 N 500000 4649776", lat, lon), ());
TestAlmostEqual(lat, 42.0);
TestAlmostEqual(lon, -93.0);
TEST(MatchUTMCoords("32U 294409 5628898", lat, lon), ());
TestAlmostEqual(lat, 50.77535);
TestAlmostEqual(lon, 6.08389);
TEST(MatchUTMCoords("32 U 294409 5628898", lat, lon), ());
TestAlmostEqual(lat, 50.77535);
TestAlmostEqual(lon, 6.08389);
TEST(MatchUTMCoords("30X 476594 9328501", lat, lon), ());
TestAlmostEqual(lat, 84.0);
TestAlmostEqual(lon, -5.00601);
TEST(MatchUTMCoords("30 N 476594 9328501", lat, lon), ());
TestAlmostEqual(lat, 84.0);
TestAlmostEqual(lon, -5.00601);
TestAlmostEqual(MatchUTMCoords("15N 500000 4649776"), 42.0, -93.0);
TestAlmostEqual(MatchUTMCoords("15 N 500000 4649776"), 42.0, -93.0);
TestAlmostEqual(MatchUTMCoords("32U 294409 5628898"), 50.77535, 6.08389);
TestAlmostEqual(MatchUTMCoords("32 U 294409 5628898"), 50.77535, 6.08389);
TestAlmostEqual(MatchUTMCoords("30X 476594 9328501"), 84.0, -5.00601);
TestAlmostEqual(MatchUTMCoords("30 N 476594 9328501"), 84.0, -5.00601);
}
UNIT_TEST(MatchUTMCoords_False)
{
double lat, lon;
TEST(!MatchUTMCoords("2 1st", lat, lon), ());
TEST(!MatchUTMCoords("15N5000004649776", lat, lon), ());
TEST(!MatchUTMCoords("2 1st").has_value(), ());
TEST(!MatchUTMCoords("15N5000004649776").has_value(), ());
// Wrong zone number (first two digits)
TEST(!MatchUTMCoords("0X 476594 9328501", lat, lon), ());
TEST(!MatchUTMCoords("0 X 476594 9328501", lat, lon), ());
TEST(!MatchUTMCoords("61N 294409 5628898", lat, lon), ());
TEST(!MatchUTMCoords("61 N 294409 5628898", lat, lon), ());
TEST(!MatchUTMCoords("0X 476594 9328501").has_value(), ());
TEST(!MatchUTMCoords("0 X 476594 9328501").has_value(), ());
TEST(!MatchUTMCoords("61N 294409 5628898").has_value(), ());
TEST(!MatchUTMCoords("61 N 294409 5628898").has_value(), ());
// Wrong zone letter
TEST(!MatchUTMCoords("25I 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("25 I 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("25O 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("25 O 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("5A 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("5 A 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("7B 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("7 B 500000 4649776", lat, lon), ());
TEST(!MatchUTMCoords("25I 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("25 I 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("25O 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("25 O 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("5A 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("5 A 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("7B 500000 4649776").has_value(), ());
TEST(!MatchUTMCoords("7 B 500000 4649776").has_value(), ());
// easting out of range (must be between 100,000 m and 999,999 m)
TEST(!MatchUTMCoords("19S 999 6360877", lat, lon), ());
TEST(!MatchUTMCoords("19S 99999 6360877", lat, lon), ());
TEST(!MatchUTMCoords("19S 1000000 6360877", lat, lon), ());
TEST(!MatchUTMCoords("19S 2000000 6360877", lat, lon), ());
TEST(!MatchUTMCoords("19S 999 6360877").has_value(), ());
TEST(!MatchUTMCoords("19S 99999 6360877").has_value(), ());
TEST(!MatchUTMCoords("19S 1000000 6360877").has_value(), ());
TEST(!MatchUTMCoords("19S 2000000 6360877").has_value(), ());
// northing out of range (must be between 0 m and 10,000,000 m)
TEST(!MatchUTMCoords("30N 476594 10000001", lat, lon), ());
TEST(!MatchUTMCoords("30N 476594 20000000", lat, lon), ());
TEST(!MatchUTMCoords("30N 476594 10000001").has_value(), ());
TEST(!MatchUTMCoords("30N 476594 20000000").has_value(), ());
}
UNIT_TEST(MatchMGRSCoords_parsing)
{
double lat, lon;
TEST(MatchMGRSCoords("30N YF 67993 00000", lat, lon), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30NYF 67993 00000", lat, lon), ());
TEST(MatchMGRSCoords("30NYF 67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30NYF 67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30NYF67993 00000", lat, lon), ());
TEST(MatchMGRSCoords("30NYF67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30NYF67993 00000 ", lat, lon), ());
TEST(MatchMGRSCoords("30NYF6799300000", lat, lon), ());
TEST(MatchMGRSCoords("30NYF6799300000 ", lat, lon), ());
TEST(MatchMGRSCoords("30N YF 67993 00000").has_value(), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30N YF 67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30NYF 67993 00000").has_value(), ());
TEST(MatchMGRSCoords("30NYF 67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30NYF 67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30NYF67993 00000").has_value(), ());
TEST(MatchMGRSCoords("30NYF67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30NYF67993 00000 ").has_value(), ());
TEST(MatchMGRSCoords("30NYF6799300000").has_value(), ());
TEST(MatchMGRSCoords("30NYF6799300000 ").has_value(), ());
// Wrong number of digits
TEST(!MatchMGRSCoords("30NYF 679930000 ", lat, lon), ());
TEST(!MatchMGRSCoords("30NYF 679930000", lat, lon), ());
TEST(!MatchMGRSCoords("30NYF 679930000 ").has_value(), ());
TEST(!MatchMGRSCoords("30NYF 679930000").has_value(), ());
TEST(!MatchMGRSCoords("30N YF 693 23020", lat, lon), ());
TEST(!MatchMGRSCoords("30N YF 693 23 ", lat, lon), ());
TEST(!MatchMGRSCoords("30N YF 693 23020").has_value(), ());
TEST(!MatchMGRSCoords("30N YF 693 23 ").has_value(), ());
// Invalid zone
TEST(!MatchMGRSCoords("30 FF 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30A YF 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z YF 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z F 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z 3F 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z K? 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z IB 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30Z DO 693 230", lat, lon), ());
TEST(!MatchMGRSCoords("30 FF 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30A YF 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z YF 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z F 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z 3F 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z K? 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z IB 693 230").has_value(), ());
TEST(!MatchMGRSCoords("30Z DO 693 230").has_value(), ());
// Wrong easting or northing
TEST(!MatchMGRSCoords("30NYF 679_3 00000", lat, lon), ());
TEST(!MatchMGRSCoords("30NYF 6930&000", lat, lon), ());
TEST(!MatchMGRSCoords("30NYF 679_3 00000").has_value(), ());
TEST(!MatchMGRSCoords("30NYF 6930&000").has_value(), ());
}
UNIT_TEST(MatchMGRSCoords_convert)
{
double lat, lon;
TestAlmostEqual(MatchMGRSCoords("30N YF 67993 00000"), 0.000000, -0.592330);
TestAlmostEqual(MatchMGRSCoords("31N BA 00000 00000"), 0.000000, 0.304980);
TestAlmostEqual(MatchMGRSCoords("32R LR 00000 00000"), 27.107980, 6.982490);
TestAlmostEqual(MatchMGRSCoords("32R MR 00000 00000"), 27.118850, 7.991060);
TestAlmostEqual(MatchMGRSCoords("33R TK 05023 99880"), 27.089890, 12.025310);
TestAlmostEqual(MatchMGRSCoords("31R GQ 69322 99158"), 31.596040, 5.838410);
TestAlmostEqual(MatchMGRSCoords("35L KF 50481 01847"), -13.541120, 24.694510);
TestAlmostEqual(MatchMGRSCoords("33L YL 59760 01153"), -9.028520, 17.362850);
TestAlmostEqual(MatchMGRSCoords("36G VR 27441 12148"), -45.040410, 32.078730);
TestAlmostEqual(MatchMGRSCoords("41R KQ 30678 99158"), 31.596040, 60.161590);
TestAlmostEqual(MatchMGRSCoords("43F CF 66291 06635"), -49.578080, 73.150400);
TestAlmostEqual(MatchMGRSCoords("43F DF 65832 14591"), -49.520340, 74.527920);
TestAlmostEqual(MatchMGRSCoords("41G NL 72559 12148"), -45.040410, 63.921270);
TestAlmostEqual(MatchMGRSCoords("41G PL 72152 04746"), -45.089800, 65.187670);
TestAlmostEqual(MatchMGRSCoords("42G UR 00000 00000"), -45.125150, 66.456880);
TestAlmostEqual(MatchMGRSCoords("42G VR 00000 00000"), -45.146390, 67.727970);
TestAlmostEqual(MatchMGRSCoords("42G WR 00000 00000"), -45.153480, 69.000000);
TestAlmostEqual(MatchMGRSCoords("42G XR 00000 00000"), -45.146390, 70.272030);
TestAlmostEqual(MatchMGRSCoords("42G YR 00000 00000"), -45.125150, 71.543120);
TestAlmostEqual(MatchMGRSCoords("43G CL 27848 04746"), -45.089800, 72.812330);
TestAlmostEqual(MatchMGRSCoords("43G DL 27441 12148"), -45.040410, 74.078730);
TestAlmostEqual(MatchMGRSCoords("41G PR 08066 09951"), -40.554160, 64.276370);
TestAlmostEqual(MatchMGRSCoords("41G QR 07714 03147"), -40.596410, 65.454770);
TestAlmostEqual(MatchMGRSCoords("42G UA 00000 00000"), -40.626640, 66.635320);
TestAlmostEqual(MatchMGRSCoords("43L GF 49519 01847"), -13.541120, 77.305490);
TestAlmostEqual(MatchMGRSCoords("45L VL 00000 00000"), -9.045430, 86.090120);
TestAlmostEqual(MatchMGRSCoords("50Q KE 64726 97325"), 18.051740, 114.777360);
TestAlmostEqual(MatchMGRSCoords("53F LF 66291 06635"), -49.578080, 133.150400);
TestAlmostEqual(MatchMGRSCoords("60F VL 65832 14591"), -49.520340, 176.527920);
TEST(MatchMGRSCoords("30N YF 67993 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, -0.592330);
TEST(MatchMGRSCoords("31N BA 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, 0.304980);
TEST(MatchMGRSCoords("32R LR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 27.107980);
TestAlmostEqual(lon, 6.982490);
TEST(MatchMGRSCoords("32R MR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 27.118850);
TestAlmostEqual(lon, 7.991060);
TEST(MatchMGRSCoords("33R TK 05023 99880", lat, lon), ());
TestAlmostEqual(lat, 27.089890);
TestAlmostEqual(lon, 12.025310);
TEST(MatchMGRSCoords("31R GQ 69322 99158", lat, lon), ());
TestAlmostEqual(lat, 31.596040);
TestAlmostEqual(lon, 5.838410);
TEST(MatchMGRSCoords("35L KF 50481 01847", lat, lon), ());
TestAlmostEqual(lat, -13.541120);
TestAlmostEqual(lon, 24.694510);
TEST(MatchMGRSCoords("33L YL 59760 01153", lat, lon), ());
TestAlmostEqual(lat, -9.028520);
TestAlmostEqual(lon, 17.362850);
TEST(MatchMGRSCoords("36G VR 27441 12148", lat, lon), ());
TestAlmostEqual(lat, -45.040410);
TestAlmostEqual(lon, 32.078730);
TEST(MatchMGRSCoords("41R KQ 30678 99158", lat, lon), ());
TestAlmostEqual(lat, 31.596040);
TestAlmostEqual(lon, 60.161590);
TEST(MatchMGRSCoords("43F CF 66291 06635", lat, lon), ());
TestAlmostEqual(lat, -49.578080);
TestAlmostEqual(lon, 73.150400);
TEST(MatchMGRSCoords("43F DF 65832 14591", lat, lon), ());
TestAlmostEqual(lat, -49.520340);
TestAlmostEqual(lon, 74.527920);
TEST(MatchMGRSCoords("41G NL 72559 12148", lat, lon), ());
TestAlmostEqual(lat, -45.040410);
TestAlmostEqual(lon, 63.921270);
TEST(MatchMGRSCoords("41G PL 72152 04746", lat, lon), ());
TestAlmostEqual(lat, -45.089800);
TestAlmostEqual(lon, 65.187670);
TEST(MatchMGRSCoords("42G UR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -45.125150);
TestAlmostEqual(lon, 66.456880);
TEST(MatchMGRSCoords("42G VR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -45.146390);
TestAlmostEqual(lon, 67.727970);
TEST(MatchMGRSCoords("42G WR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -45.153480);
TestAlmostEqual(lon, 69.000000);
TEST(MatchMGRSCoords("42G XR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -45.146390);
TestAlmostEqual(lon, 70.272030);
TEST(MatchMGRSCoords("42G YR 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -45.125150);
TestAlmostEqual(lon, 71.543120);
TEST(MatchMGRSCoords("43G CL 27848 04746", lat, lon), ());
TestAlmostEqual(lat, -45.089800);
TestAlmostEqual(lon, 72.812330);
TEST(MatchMGRSCoords("43G DL 27441 12148", lat, lon), ());
TestAlmostEqual(lat, -45.040410);
TestAlmostEqual(lon, 74.078730);
TEST(MatchMGRSCoords("41G PR 08066 09951", lat, lon), ());
TestAlmostEqual(lat, -40.554160);
TestAlmostEqual(lon, 64.276370);
TEST(MatchMGRSCoords("41G QR 07714 03147", lat, lon), ());
TestAlmostEqual(lat, -40.596410);
TestAlmostEqual(lon, 65.454770);
TEST(MatchMGRSCoords("42G UA 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -40.626640);
TestAlmostEqual(lon, 66.635320);
TEST(MatchMGRSCoords("43L GF 49519 01847", lat, lon), ());
TestAlmostEqual(lat, -13.541120);
TestAlmostEqual(lon, 77.305490);
TEST(MatchMGRSCoords("45L VL 00000 00000", lat, lon), ());
TestAlmostEqual(lat, -9.045430);
TestAlmostEqual(lon, 86.090120);
TEST(MatchMGRSCoords("50Q KE 64726 97325", lat, lon), ());
TestAlmostEqual(lat, 18.051740);
TestAlmostEqual(lon, 114.777360);
TEST(MatchMGRSCoords("53F LF 66291 06635", lat, lon), ());
TestAlmostEqual(lat, -49.578080);
TestAlmostEqual(lon, 133.150400);
TEST(MatchMGRSCoords("60F VL 65832 14591", lat, lon), ());
TestAlmostEqual(lat, -49.520340);
TestAlmostEqual(lon, 176.527920);
TEST(MatchMGRSCoords("04X ER 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 81.060880);
TestAlmostEqual(lon, -159.000000);
TEST(MatchMGRSCoords("05X MK 95564 95053", lat, lon), ());
TestAlmostEqual(lat, 81.016470);
TestAlmostEqual(lon, -153.254480);
TEST(MatchMGRSCoords("08X ME 93476 90354", lat, lon), ());
TestAlmostEqual(lat, 72.012660);
TestAlmostEqual(lon, -135.189280);
TEST(MatchMGRSCoords("12H TF 59828 01847", lat, lon), ());
TestAlmostEqual(lat, -36.098350);
TestAlmostEqual(lon, -113.667820);
TEST(MatchMGRSCoords("15N YA 67993 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, -90.592330);
TEST(MatchMGRSCoords("16N BF 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, -89.695020);
TEST(MatchMGRSCoords("16N CF 00000 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, -88.797050);
TEST(MatchMGRSCoords("21L TL 40240 01153", lat, lon), ());
TestAlmostEqual(lat, -9.028520);
TestAlmostEqual(lon, -59.362850);
TEST(MatchMGRSCoords("24P YV 49519 98153", lat, lon), ());
TestAlmostEqual(lat, 13.541120);
TestAlmostEqual(lon, -36.694510);
TEST(MatchMGRSCoords("31K BA 64726 02675", lat, lon), ());
TestAlmostEqual(lat, -18.051740);
TestAlmostEqual(lon, 0.777360);
TEST(MatchMGRSCoords("31N BA 32007 00000", lat, lon), ());
TestAlmostEqual(lat, 0.000000);
TestAlmostEqual(lon, 0.592330);
TestAlmostEqual(MatchMGRSCoords("04X ER 00000 00000"), 81.060880, -159.000000);
TestAlmostEqual(MatchMGRSCoords("05X MK 95564 95053"), 81.016470, -153.254480);
TestAlmostEqual(MatchMGRSCoords("08X ME 93476 90354"), 72.012660, -135.189280);
TestAlmostEqual(MatchMGRSCoords("12H TF 59828 01847"), -36.098350, -113.667820);
TestAlmostEqual(MatchMGRSCoords("15N YA 67993 00000"), 0.000000, -90.592330);
TestAlmostEqual(MatchMGRSCoords("16N BF 00000 00000"), 0.000000, -89.695020);
TestAlmostEqual(MatchMGRSCoords("16N CF 00000 00000"), 0.000000, -88.797050);
TestAlmostEqual(MatchMGRSCoords("21L TL 40240 01153"), -9.028520, -59.362850);
TestAlmostEqual(MatchMGRSCoords("24P YV 49519 98153"), 13.541120, -36.694510);
TestAlmostEqual(MatchMGRSCoords("31K BA 64726 02675"), -18.051740, 0.777360);
TestAlmostEqual(MatchMGRSCoords("31N BA 32007 00000"), 0.000000, 0.592330);
}

View file

@ -26,7 +26,7 @@ size_t MatchLong(string const & query, long &value, size_t startPos);
// Parse UTM format "(\d\d)\s?(\W)\s+(\d+)\s+(\d+)" and converts it to lat,lon.
// Return true if parsed successfully or false otherwise.
// See utm_mgrs_coords_match_test.cpp for sample UTM strings
bool MatchUTMCoords(string const & query, double & lat, double & lon)
std::optional<ms::LatLon> MatchUTMCoords(string const & query)
{
int zone_code;
char zone_letter;
@ -34,21 +34,21 @@ bool MatchUTMCoords(string const & query, double & lat, double & lon)
size_t pos = MatchZoneCode(query, zone_code);
if (pos == string::npos)
return false;
return nullopt;
pos = MatchZoneLetter(query, zone_letter, pos);
if (pos == string::npos)
return false;
return nullopt;
pos = MatchLong(query, easting, pos);
if (pos == string::npos)
return false;
return nullopt;
pos = MatchLong(query, northing, pos);
if (pos == string::npos)
return false;
return nullopt;
return utm_mgrs_utils::UTMtoLatLon((double)easting, (double)northing, zone_code, zone_letter, lat, lon);
return utm_mgrs_utils::UTMtoLatLon((double)easting, (double)northing, zone_code, zone_letter);
}
// Matches 2 digits zone code. Returns end position of matched chars or string::npos if no match.
@ -112,7 +112,7 @@ size_t MatchLong(string const & query, long &value, size_t startPos)
// Parse MGRS format "(\d\d\W)\s*(\W\W)\s*(\d+)\s*(\d+)" and converts it to lat,lon.
// Returns true if parsed successfully or false otherwise.
// See utm_mgrs_coords_match_test.cpp for sample MGRS strings
bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
std::optional<ms::LatLon> MatchMGRSCoords(std::string const & query)
{
long zone_code;
char zone_letter;
@ -124,7 +124,7 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
strings::SimpleTokenizer it(query, " \t\r");
if (!it)
return false;
return nullopt;
auto token = std::string(*it);
// Parse 2 digit zone code and 1 char zone letter
@ -133,24 +133,24 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
char dig1 = token[0];
char dig2 = token[1];
if (dig1 < '0' || dig1 > '9' || dig2 < '0' || dig2 > '9')
return false;
return nullopt;
zone_code = (dig1 - '0') * 10 + (dig2 - '0');
if (zone_code<1 || zone_code > 60)
return false;
return nullopt;
zone_letter = token[2];
token = token.substr(3);
}
else
return false;
return nullopt;
// Read next token if needed.
if (token.size() == 0)
{
++it;
if (!it)
return false;
return nullopt;
token = std::string(*it);
}
@ -167,7 +167,7 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
{
++it;
if (!it)
return false;
return nullopt;
token = std::string(*it);
}
@ -184,7 +184,7 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
{
// eastingStr contains both easting and northing. Let's split
if (eastingStr.size()%2 != 0)
return false;
return nullopt;
int eastingSize = eastingStr.size()/2;
northingStr = eastingStr.substr(eastingSize);
@ -192,10 +192,10 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
}
if (eastingStr.size() != northingStr.size() || eastingStr.size()>5 || northingStr.size()>5)
return false;
return nullopt;
if (!strings::to_int64(eastingStr, easting))
return false;
return nullopt;
if (eastingStr.size() < 5)
{
int decShift = 5 - eastingStr.size();
@ -203,14 +203,14 @@ bool MatchMGRSCoords(std::string const & query, double & lat, double & lon)
}
if (!strings::to_int64(northingStr, northing))
return false;
return nullopt;
if (northingStr.size() < 5)
{
int decShift = 5 - northingStr.size();
northing *= base::PowUint(10L, decShift);
}
return utm_mgrs_utils::MGRStoLatLon((double)easting, (double)northing, zone_code, zone_letter, square_code, lat, lon);
return utm_mgrs_utils::MGRStoLatLon((double)easting, (double)northing, zone_code, zone_letter, square_code);
}
}

View file

@ -2,9 +2,11 @@
#include <string>
#include "geometry/latlon.hpp"
namespace search
{
// Parses input query for UTM and MGRS coordinate formats.
bool MatchUTMCoords(std::string const & query, double & lat, double & lon);
bool MatchMGRSCoords(std::string const & query, double & lat, double & lon);
std::optional<ms::LatLon> MatchUTMCoords(std::string const & query);
std::optional<ms::LatLon> MatchMGRSCoords(std::string const & query);
} // namespace search