From 0db9a51e168c88881836e6ee3e40a9400af4e4f7 Mon Sep 17 00:00:00 2001 From: vng Date: Fri, 14 Mar 2014 14:21:54 +0300 Subject: [PATCH] [search] Match coordinates like DDD.MM.SS and variations. --- search/latlon_match.cpp | 169 +++++++++++++++++++++- search/latlon_match.hpp | 2 + search/search_query.cpp | 3 +- search/search_tests/latlon_match_test.cpp | 42 +++++- 4 files changed, 207 insertions(+), 9 deletions(-) diff --git a/search/latlon_match.cpp b/search/latlon_match.cpp index 6fc0054d82..876c3d84b0 100644 --- a/search/latlon_match.cpp +++ b/search/latlon_match.cpp @@ -1,24 +1,36 @@ #include "latlon_match.hpp" -#include "../std/algorithm.hpp" + +#include "../indexer/mercator.hpp" + +#include "../std/array.hpp" #include "../std/cmath.hpp" #include "../std/cstdlib.hpp" + +namespace search +{ + namespace { +template void SkipSpaces(CharT * & s) +{ + while (*s && (*s == ' ' || *s == '\t')) + ++s; +} + template void Skip(CharT * & s) { - // 0xC2 - commond Unicode first byte for ANSI (not ASCII!) symbols. - // 0xC2,0xB0 - degree symbol. - while (*s == ' ' || *s == ',' || *s == ';' || *s == ':' || *s == '.' || *s == '(' || *s == ')' || - *s == char(0xB0) || *s == char(0xC2)) + while (*s && (*s == ' ' || *s == '\t' || *s == ',' || *s == ';' || + *s == ':' || *s == '.' || *s == '(' || *s == ')' || + *s == 'N' || *s == 'E' || *s == 'n' || *s == 'e')) ++s; } } // unnamed namespace -bool search::MatchLatLon(string const & str, double & latRes, double & lonRes, - double & precisionLat, double & precisionLon) +bool MatchLatLon(string const & str, double & latRes, double & lonRes, + double & precisionLat, double & precisionLon) { char const * s = str.c_str(); @@ -68,3 +80,146 @@ bool search::MatchLatLon(string const & str, double & latRes, double & lonRes, precisionLon = 0.5 * pow(10.0, -max(0, min(8, lonDigits))); return true; } + +namespace +{ + +int Match1Byte(char const * & s) +{ + uint8_t const ch = static_cast(*s++); + switch (ch) + { + case 0xB0: // ° + return 0; + default: + return -1; + } +} + +int Match2Bytes(char const * & s) +{ + uint8_t ch = static_cast(*s++); + if (ch != 0x80) + return -1; + + ch = static_cast(*s++); + switch (ch) + { + case 0x99: // ’ + return 1; + case 0x9D: // ” + return 2; + case 0xB2: // ′ + return 1; + case 0xB3: // ″ + return 2; + default: + return -1; + } +} + +int GetDMSIndex(char const * & s) +{ + uint8_t const ch = static_cast(*s++); + switch (ch) + { + // UTF8 control symbols + case 0xC2: + return Match1Byte(s); + case 0xE2: + return Match2Bytes(s); + + case '*': return 0; + case '\'': return 1; + case '\"': return 2; + default: return -1; + } +} + +} + +bool MatchLatLonDegree(string const & query, double & lat, double & lon) +{ + // should be default initialization (0, false) + array, 6> v; + + int base = 0; + + char const * s = query.c_str(); + while (true) + { + char const * s1 = s; + Skip(s); + if (!*s) + { + // End of the string - check matching. + break; + } + + char * s2; + double const x = strtod(s, &s2); + if (s == s2) + { + // Invalid token + if (s == s1) + { + // Return error if there are no any delimiters + return false; + } + else + { + // Check matching if token is delimited. + break; + } + } + + s = s2; + SkipSpaces(s); + + int const i = GetDMSIndex(s); + switch (i) + { + case -1: // expect valid control symbol + return false; + + case 0: // degree + if (v[base].second) + { + if (base == 0) + base += 3; + else + { + // repeated value + return false; + } + } + break; + + default: // minutes or seconds + if (x < 0.0 || v[base + i].second || !v[base].second) + return false; + } + + v[base + i].first = x; + v[base + i].second = true; + } + + if (!v[0].second || !v[3].second) + { + // degree should exist for both coordinates + return false; + } + + lat = fabs(v[0].first) + v[1].first / 60.0 + v[2].first / 3600.0; + if (v[0].first < 0.0) lat = -lat; + + lon = fabs(v[3].first) + v[4].first / 60.0 + v[5].first / 3600.0; + if (v[3].first < 0.0) lon = -lon; + + if (lon > 180.0) lon -= 360.0; + if (lon < -180.0) lon += 360.0; + + return MercatorBounds::ValidLat(lat) && MercatorBounds::ValidLon(lon); +} + +} // search diff --git a/search/latlon_match.hpp b/search/latlon_match.hpp index d9a688259f..c62bc424b4 100644 --- a/search/latlon_match.hpp +++ b/search/latlon_match.hpp @@ -9,4 +9,6 @@ namespace search bool MatchLatLon(string const & query, double & lat, double & lon, double & precisionLat, double & precisionLon); +bool MatchLatLonDegree(string const & query, double & lat, double & lon); + } // namespace search diff --git a/search/search_query.cpp b/search/search_query.cpp index 073c099b27..df61ac4036 100644 --- a/search/search_query.cpp +++ b/search/search_query.cpp @@ -287,7 +287,8 @@ void Query::SetQuery(string const & query) void Query::SearchCoordinates(string const & query, Results & res) const { double lat, lon, latPrec, lonPrec; - if (search::MatchLatLon(query, lat, lon, latPrec, lonPrec)) + if (MatchLatLon(query, lat, lon, latPrec, lonPrec) || + MatchLatLonDegree(query, lat, lon)) { //double const precision = 5.0 * max(0.0001, min(latPrec, lonPrec)); // Min 55 meters res.AddResult(MakeResult(impl::PreResult2(GetViewport(), m_position, lat, lon))); diff --git a/search/search_tests/latlon_match_test.cpp b/search/search_tests/latlon_match_test.cpp index 73cb4e44fb..ded099e972 100644 --- a/search/search_tests/latlon_match_test.cpp +++ b/search/search_tests/latlon_match_test.cpp @@ -36,7 +36,7 @@ void TestLatLonFailed(string const & s) } // unnamed namespace -UNIT_TEST(LatLonMatch) +UNIT_TEST(LatLon_Match) { TestLatLonFailed(""); TestLatLonFailed("0.0"); @@ -65,3 +65,43 @@ UNIT_TEST(LatLonMatch) TEST_EQUAL(R(0.0, -160.0, 0.05, 0.5), TestLatLonMatchSuccessful("0.0 200"), ()); TEST_EQUAL(R(0.0, 0.0, 0.05, 0.5), TestLatLonMatchSuccessful("0.0 360"), ()); } + +UNIT_TEST(LatLon_Degree_Match) +{ + using namespace search; + double lat, lon; + + TEST(MatchLatLonDegree("0*30\', 1*0\'30\"", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, 0.5, ()); + TEST_ALMOST_EQUAL(lon, 1.00833333333333, ()); + + TEST(MatchLatLonDegree("50 *, 40 *", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, 50.0, ()); + TEST_ALMOST_EQUAL(lon, 40.0, ()); + + TEST(!MatchLatLonDegree("50* 40*, 30*", lat, lon), ()); + + TEST(MatchLatLonDegree("(-50°30\'30\" -49°59\'59\"", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, -50.50833333333333, ()); + TEST_ALMOST_EQUAL(lon, -49.99972222222222, ()); + + TEST(!MatchLatLonDegree("50°, 30\"", lat, lon), ()); + TEST(!MatchLatLonDegree("50\', -50°", lat, lon), ()); + TEST(!MatchLatLonDegree("-90*50\'50\", -50°", lat, lon), ()); + + TEST(MatchLatLonDegree("(-89*, 360*)", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, -89.0, ()); + TEST_ALMOST_EQUAL(lon, 0.0, ()); + + TEST(MatchLatLonDegree("-89*15.5\' N; 120*30\'50.5\" e", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, -89.25833333333333, ()); + TEST_ALMOST_EQUAL(lon, 120.51402777777778, ()); + + TEST(MatchLatLonDegree("N55°45′20.99″ E37°37′03.62″", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, 55.755830555555556, ()); + TEST_ALMOST_EQUAL(lon, 37.617672222222222, ()); + + TEST(MatchLatLonDegree("55°45’20.9916\"N, 37°37’3.6228\"E hsdfjgkdsjbv", lat, lon), ()); + TEST_ALMOST_EQUAL(lat, 55.755831, ()); + TEST_ALMOST_EQUAL(lon, 37.617673, ()); +}