diff --git a/indexer/scales.hpp b/indexer/scales.hpp index 2c3398ccc1..476ea2e48d 100644 --- a/indexer/scales.hpp +++ b/indexer/scales.hpp @@ -14,6 +14,8 @@ namespace scales inline int GetUpperStyleScale() { return UPPER_STYLE_SCALE; } /// Upper scales for World visible styles and indexer buckets. inline int GetUpperWorldScale() { return 9; } + /// Upper scale level for countries. + inline int GetUpperCountryScale() { return GetUpperWorldScale() + 1; } /// Upper scale for user comfort view (e.g. location zoom). inline int GetUpperComfortScale() { return UPPER_STYLE_SCALE - 2; } /// Default navigation mode scale diff --git a/map/framework.cpp b/map/framework.cpp index 5516f6dd71..f6b49c44fc 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -1020,7 +1020,7 @@ void Framework::InitCountryInfoGetter() Platform const & platform = GetPlatform(); try { - m_infoGetter.reset(new storage::CountryInfoGetter(platform.GetReader(PACKED_POLYGONS_FILE), + m_infoGetter.reset(new storage::CountryInfoReader(platform.GetReader(PACKED_POLYGONS_FILE), platform.GetReader(COUNTRIES_FILE))); } catch (RootException const & e) diff --git a/routing/routing_integration_tests/routing_test_tools.cpp b/routing/routing_integration_tests/routing_test_tools.cpp index bb5dd8c3e6..a05eb48700 100644 --- a/routing/routing_integration_tests/routing_test_tools.cpp +++ b/routing/routing_integration_tests/routing_test_tools.cpp @@ -66,7 +66,7 @@ namespace integration unique_ptr CreateCountryInfoGetter() { Platform const & platform = GetPlatform(); - return make_unique(platform.GetReader(PACKED_POLYGONS_FILE), + return make_unique(platform.GetReader(PACKED_POLYGONS_FILE), platform.GetReader(COUNTRIES_FILE)); } diff --git a/search/retrieval.cpp b/search/retrieval.cpp index fadf76947c..f11974956a 100644 --- a/search/retrieval.cpp +++ b/search/retrieval.cpp @@ -27,7 +27,7 @@ double const kInfinity = numeric_limits::infinity(); // Maximum viewport scale level when we stop to expand viewport and // simply return all rest features from mwms. -double const kMaxViewportScaleLevel = 10.0; +double const kMaxViewportScaleLevel = scales::GetUpperCountryScale(); // Upper bound on a number of features when fast path is used. // Otherwise, slow path is used. diff --git a/search/search_integration_tests/retrieval_test.cpp b/search/search_integration_tests/retrieval_test.cpp index 2ffd5888aa..6398ce7025 100644 --- a/search/search_integration_tests/retrieval_test.cpp +++ b/search/search_integration_tests/retrieval_test.cpp @@ -14,6 +14,9 @@ #include "search/search_tests_support/test_search_engine.hpp" #include "search/search_tests_support/test_search_request.hpp" +#include "storage/country_decl.hpp" +#include "storage/country_info_getter.hpp" + #include "platform/local_country_file.hpp" #include "platform/local_country_file_utils.hpp" #include "platform/platform.hpp" @@ -163,6 +166,7 @@ bool MatchResults(Index const & index, vector> rules, for (auto const & u : unexpected) os << " " << u << endl; + LOG(LWARNING, (os.str())); return false; } @@ -465,7 +469,16 @@ UNIT_TEST(Retrieval_CafeMTV) builder.Add(*mtvCity); } - TestSearchEngine engine("en"); + m2::RectD const mskViewport(m2::PointD(0.99, -0.1), m2::PointD(1.01, 0.1)); + m2::RectD const mtvViewport(m2::PointD(-1.1, -0.1), m2::PointD(-0.99, 0.1)); + + // There are test requests involving locality search, thus it's + // better to mock information about countries. + vector countries; + countries.emplace_back(msk.GetCountryName(), mskViewport); + countries.emplace_back(mtv.GetCountryName(), mtvViewport); + + TestSearchEngine engine("en", make_unique(countries)); TEST_EQUAL(MwmSet::RegResult::Success, engine.RegisterMap(msk).second, ()); TEST_EQUAL(MwmSet::RegResult::Success, engine.RegisterMap(mtv).second, ()); TEST_EQUAL(MwmSet::RegResult::Success, engine.RegisterMap(testWorld).second, ()); @@ -474,11 +487,8 @@ UNIT_TEST(Retrieval_CafeMTV) auto const mtvId = engine.GetMwmIdByCountryFile(mtv.GetCountryFile()); auto const testWorldId = engine.GetMwmIdByCountryFile(testWorld.GetCountryFile()); - m2::RectD const moscowViewport(m2::PointD(0.99, -0.1), m2::PointD(1.01, 0.1)); - m2::RectD const mtvViewport(m2::PointD(-1.1, -0.1), m2::PointD(-0.99, 0.1)); - { - TestSearchRequest request(engine, "Moscow ", "en", search::SearchParams::ALL, moscowViewport); + TestSearchRequest request(engine, "Moscow ", "en", search::SearchParams::ALL, mskViewport); request.Wait(); initializer_list> mskCityAlts = { diff --git a/search/search_query.cpp b/search/search_query.cpp index c3e654b0ed..32fa7891f8 100644 --- a/search/search_query.cpp +++ b/search/search_query.cpp @@ -1301,37 +1301,15 @@ void Query::SearchAddress(Results & res) // Candidates for search around locality. Initially filled // with mwms containing city center. - TMWMVector candidateMwms; - auto localityMismatch = [&cityCenter, &rect, this](shared_ptr const & info) + TMWMVector localityMwms; + string const localityFile = m_infoGetter.GetRegionFile(cityCenter); + auto localityMismatch = [&localityFile](shared_ptr const & info) { - return !info->m_limitRect.IsIntersect(rect); + return info->GetCountryName() != localityFile; }; - remove_copy_if(mwmsInfo.begin(), mwmsInfo.end(), back_inserter(candidateMwms), + remove_copy_if(mwmsInfo.begin(), mwmsInfo.end(), back_inserter(localityMwms), localityMismatch); - - auto boundingBoxCmp = [](shared_ptr const & lhs, shared_ptr const & rhs) - { - auto const & lhsBox = lhs->m_limitRect; - double const lhsSize = max(lhsBox.SizeX(), lhsBox.SizeY()); - - auto & rhsBox = rhs->m_limitRect; - double const rhsSize = max(rhsBox.SizeX(), rhsBox.SizeY()); - - return lhsSize < rhsSize; - }; - - // Candidate mwm for search around locality. Among all - // candidates the one with smallest dimensions is - // selected. This hack is used here to prevent - // interferention from mwms whose bounding boxes are too - // pessimisic, say, from -180 to +180 in Mercator - // coordiantes. This is the case for Russia_Far_Eastern. - auto const candidateMwmIt = min_element(candidateMwms.begin(), candidateMwms.end(), boundingBoxCmp); - if (candidateMwmIt != candidateMwms.end()) - { - TMWMVector localityMwm = {*candidateMwmIt}; - SearchFeaturesInViewport(params, localityMwm, LOCALITY_V); - } + SearchFeaturesInViewport(params, localityMwms, LOCALITY_V); } else { diff --git a/search/search_tests_support/test_search_engine.cpp b/search/search_tests_support/test_search_engine.cpp index aba4f74b4a..5ba60e6e1d 100644 --- a/search/search_tests_support/test_search_engine.cpp +++ b/search/search_tests_support/test_search_engine.cpp @@ -7,6 +7,8 @@ #include "search/search_query_factory.hpp" #include "search/suggest.hpp" +#include "storage/country_info_getter.hpp" + #include "platform/platform.hpp" #include "defines.hpp" @@ -49,12 +51,24 @@ namespace tests_support { TestSearchEngine::TestSearchEngine(string const & locale) : m_platform(GetPlatform()) - , m_infoGetter(m_platform.GetReader(PACKED_POLYGONS_FILE), m_platform.GetReader(COUNTRIES_FILE)) - , m_engine(*this, m_platform.GetReader(SEARCH_CATEGORIES_FILE_NAME), m_infoGetter, locale, + , m_infoGetter(new storage::CountryInfoReader(m_platform.GetReader(PACKED_POLYGONS_FILE), + m_platform.GetReader(COUNTRIES_FILE))) + , m_engine(*this, m_platform.GetReader(SEARCH_CATEGORIES_FILE_NAME), *m_infoGetter, locale, make_unique()) { } +TestSearchEngine::TestSearchEngine(string const & locale, + unique_ptr && infoGetter) + : m_platform(GetPlatform()) + , m_infoGetter(move(infoGetter)) + , m_engine(*this, m_platform.GetReader(SEARCH_CATEGORIES_FILE_NAME), *m_infoGetter, locale, + make_unique()) +{ +} + +TestSearchEngine::~TestSearchEngine() {} + weak_ptr TestSearchEngine::Search(search::SearchParams const & params, m2::RectD const & viewport) { diff --git a/search/search_tests_support/test_search_engine.hpp b/search/search_tests_support/test_search_engine.hpp index 2ae9c0a1c0..b4f46f5b96 100644 --- a/search/search_tests_support/test_search_engine.hpp +++ b/search/search_tests_support/test_search_engine.hpp @@ -6,13 +6,16 @@ #include "search/search_engine.hpp" -#include "storage/country_info_getter.hpp" - #include "std/string.hpp" #include "std/weak_ptr.hpp" class Platform; +namespace storage +{ +class CountryInfoGetter; +} + namespace search { class SearchParams; @@ -23,13 +26,15 @@ class TestSearchEngine : public Index { public: TestSearchEngine(string const & locale); + TestSearchEngine(string const & locale, unique_ptr && infoGetter); + ~TestSearchEngine(); weak_ptr Search(search::SearchParams const & params, m2::RectD const & viewport); private: Platform & m_platform; - storage::CountryInfoGetter m_infoGetter; + unique_ptr m_infoGetter; search::Engine m_engine; }; } // namespace tests_support diff --git a/storage/country_info_getter.cpp b/storage/country_info_getter.cpp index ee4e340ae7..6536c745a3 100644 --- a/storage/country_info_getter.cpp +++ b/storage/country_info_getter.cpp @@ -45,17 +45,7 @@ private: }; } // namespace -CountryInfoGetter::CountryInfoGetter(ModelReaderPtr polyR, ModelReaderPtr countryR) - : m_reader(polyR), m_cache(3) -{ - ReaderSource src(m_reader.GetReader(PACKED_POLYGONS_INFO_TAG)); - rw::Read(src, m_countries); - - string buffer; - countryR.ReadAsString(buffer); - LoadCountryFile2CountryInfo(buffer, m_id2info); -} - +// CountryInfoGetter ------------------------------------------------------------------------------- string CountryInfoGetter::GetRegionFile(m2::PointD const & pt) const { IdType const id = FindFirstCountry(pt); @@ -114,7 +104,7 @@ bool CountryInfoGetter::IsBelongToRegions(m2::PointD const & pt, IdSet const & r { for (auto const & id : regions) { - if (m_countries[id].m_rect.IsPointInside(pt) && IsBelongToRegion(id, pt)) + if (m_countries[id].m_rect.IsPointInside(pt) && IsBelongToRegionImpl(id, pt)) return true; } return false; @@ -130,7 +120,39 @@ bool CountryInfoGetter::IsBelongToRegions(string const & fileName, IdSet const & return false; } -void CountryInfoGetter::ClearCaches() const +CountryInfoGetter::IdType CountryInfoGetter::FindFirstCountry(m2::PointD const & pt) const +{ + for (size_t id = 0; id < m_countries.size(); ++id) + { + if (m_countries[id].m_rect.IsPointInside(pt) && IsBelongToRegionImpl(id, pt)) + return id; + } + return kInvalidId; +} + +template +void CountryInfoGetter::ForEachCountry(string const & prefix, ToDo && toDo) const +{ + for (auto const & country : m_countries) + { + if (strings::StartsWith(country.m_name, prefix.c_str())) + toDo(country); + } +} + +// CountryInfoReader ------------------------------------------------------------------------------- +CountryInfoReader::CountryInfoReader(ModelReaderPtr polyR, ModelReaderPtr countryR) + : m_reader(polyR), m_cache(3) +{ + ReaderSource src(m_reader.GetReader(PACKED_POLYGONS_INFO_TAG)); + rw::Read(src, m_countries); + + string buffer; + countryR.ReadAsString(buffer); + LoadCountryFile2CountryInfo(buffer, m_id2info); +} + +void CountryInfoReader::ClearCachesImpl() const { lock_guard lock(m_cacheMutex); @@ -138,7 +160,7 @@ void CountryInfoGetter::ClearCaches() const m_cache.Reset(); } -bool CountryInfoGetter::IsBelongToRegion(size_t id, m2::PointD const & pt) const +bool CountryInfoReader::IsBelongToRegionImpl(size_t id, m2::PointD const & pt) const { lock_guard lock(m_cacheMutex); @@ -168,23 +190,23 @@ bool CountryInfoGetter::IsBelongToRegion(size_t id, m2::PointD const & pt) const return false; } -CountryInfoGetter::IdType CountryInfoGetter::FindFirstCountry(m2::PointD const & pt) const -{ - for (size_t id = 0; id < m_countries.size(); ++id) - { - if (m_countries[id].m_rect.IsPointInside(pt) && IsBelongToRegion(id, pt)) - return id; - } - return kInvalidId; -} - -template -void CountryInfoGetter::ForEachCountry(string const & prefix, ToDo && toDo) const +// CountryInfoGetterForTesting --------------------------------------------------------------------- +CountryInfoGetterForTesting::CountryInfoGetterForTesting(vector const & countries) { + m_countries.assign(countries.begin(), countries.end()); for (auto const & country : m_countries) { - if (strings::StartsWith(country.m_name, prefix.c_str())) - toDo(country); + string const & name = country.m_name; + m_id2info[name].m_name = name; } } + +void CountryInfoGetterForTesting::ClearCachesImpl() const {} + +bool CountryInfoGetterForTesting::IsBelongToRegionImpl(size_t id, + m2::PointD const & pt) const +{ + CHECK_LESS(id, m_countries.size(), ()); + return m_countries[id].m_rect.IsPointInside(pt); +} } // namespace storage diff --git a/storage/country_info_getter.hpp b/storage/country_info_getter.hpp index 9d1d6b3c51..865926ebba 100644 --- a/storage/country_info_getter.hpp +++ b/storage/country_info_getter.hpp @@ -23,7 +23,7 @@ public: using IdType = size_t; using IdSet = vector; - CountryInfoGetter(ModelReaderPtr polyR, ModelReaderPtr countryR); + virtual ~CountryInfoGetter() = default; // Returns country file name without an extension for a country |pt| // belongs to. If there are no such country, returns an empty @@ -58,11 +58,10 @@ public: bool IsBelongToRegions(string const & fileName, IdSet const & regions) const; // Clears regions cache. - void ClearCaches() const; + inline void ClearCaches() const { ClearCachesImpl(); } -private: - // Returns true when |pt| belongs to a country identified by |id|. - bool IsBelongToRegion(size_t id, m2::PointD const & pt) const; +protected: + CountryInfoGetter() = default; // Returns identifier of a first country containing |pt|. IdType FindFirstCountry(m2::PointD const & pt) const; @@ -71,15 +70,47 @@ private: template void ForEachCountry(string const & prefix, ToDo && toDo) const; + // Clears regions cache. + virtual void ClearCachesImpl() const = 0; + + // Returns true when |pt| belongs to a country identified by |id|. + virtual bool IsBelongToRegionImpl(size_t id, m2::PointD const & pt) const = 0; + + // List of all known countries. vector m_countries; // Maps country file name without an extension to a country info. map m_id2info; +}; + +// This class reads info about countries from polygons file and +// countries file and caches it. +class CountryInfoReader : public CountryInfoGetter +{ +public: + CountryInfoReader(ModelReaderPtr polyR, ModelReaderPtr countryR); + +protected: + // CountryInfoGetter overrides: + void ClearCachesImpl() const override; + bool IsBelongToRegionImpl(size_t id, m2::PointD const & pt) const override; - // Only cache and reader can be modified from different threads, so - // they're guarded by m_cacheMutex. FilesContainerR m_reader; mutable my::Cache> m_cache; mutable mutex m_cacheMutex; }; + +// This class allows users to get info about very simply rectangular +// countries, whose info can be described by CountryDef instances. +// It's needed for testing purposes only. +class CountryInfoGetterForTesting : public CountryInfoGetter +{ +public: + CountryInfoGetterForTesting(vector const & countries); + +protected: + // CountryInfoGetter overrides: + void ClearCachesImpl() const override; + bool IsBelongToRegionImpl(size_t id, m2::PointD const & pt) const override; +}; } // namespace storage diff --git a/storage/storage_tests/country_info_getter_test.cpp b/storage/storage_tests/country_info_getter_test.cpp index 663d3c51fb..efcbd25e55 100644 --- a/storage/storage_tests/country_info_getter_test.cpp +++ b/storage/storage_tests/country_info_getter_test.cpp @@ -19,7 +19,7 @@ namespace unique_ptr CreateCountryInfoGetter() { Platform & platform = GetPlatform(); - return make_unique(platform.GetReader(PACKED_POLYGONS_FILE), + return make_unique(platform.GetReader(PACKED_POLYGONS_FILE), platform.GetReader(COUNTRIES_FILE)); }