diff --git a/generator/collector_city_area.cpp b/generator/collector_city_area.cpp index 1b5e8d22ba..7b266166ff 100644 --- a/generator/collector_city_area.cpp +++ b/generator/collector_city_area.cpp @@ -31,7 +31,8 @@ std::shared_ptr CityAreaCollector::Clone( void CityAreaCollector::CollectFeature(FeatureBuilder const & feature, OsmElement const &) { - if (!(feature.IsArea() && ftypes::IsCityTownOrVillage(feature.GetTypes()))) + auto const & isCityTownOrVillage = ftypes::IsCityTownOrVillageChecker::Instance(); + if (!(feature.IsArea() && isCityTownOrVillage(feature.GetTypes()))) return; auto copy = feature; diff --git a/generator/generator_integration_tests/features_tests.cpp b/generator/generator_integration_tests/features_tests.cpp index 6988d42178..60238d1827 100644 --- a/generator/generator_integration_tests/features_tests.cpp +++ b/generator/generator_integration_tests/features_tests.cpp @@ -427,7 +427,8 @@ private: if (poiChecker(fb.GetTypes())) ++actual.m_poi; - if (ftypes::IsCityTownOrVillage(fb.GetTypes())) + auto const & isCityTownOrVillage = ftypes::IsCityTownOrVillageChecker::Instance(); + if (isCityTownOrVillage(fb.GetTypes())) ++actual.m_cityTownOrVillage; auto static const & bookingChecker = ftypes::IsBookingHotelChecker::Instance(); diff --git a/generator/generator_tests_support/test_mwm_builder.cpp b/generator/generator_tests_support/test_mwm_builder.cpp index eaa0392280..598d4c4b73 100644 --- a/generator/generator_tests_support/test_mwm_builder.cpp +++ b/generator/generator_tests_support/test_mwm_builder.cpp @@ -87,7 +87,8 @@ bool TestMwmBuilder::Add(FeatureBuilder & fb) { CHECK(m_collector, ("It's not possible to add features after call to Finish().")); - if (ftypes::IsCityTownOrVillage(fb.GetTypes()) && fb.GetGeomType() == GeomType::Area) + auto const & isCityTownOrVillage = ftypes::IsCityTownOrVillageChecker::Instance(); + if (isCityTownOrVillage(fb.GetTypes()) && fb.GetGeomType() == GeomType::Area) { auto const & metadata = fb.GetMetadata(); uint64_t testId; diff --git a/generator/place_processor.cpp b/generator/place_processor.cpp index e436e25a98..b3b04f71ef 100644 --- a/generator/place_processor.cpp +++ b/generator/place_processor.cpp @@ -159,12 +159,13 @@ void PlaceProcessor::FillTable(FeaturePlaces::const_iterator start, FeaturePlace { CHECK(m_boundariesTable, ()); base::GeoObjectId lastId; + auto const & isCityTownOrVillage = ftypes::IsCityTownOrVillageChecker::Instance(); for (auto outerIt = start; outerIt != end; ++outerIt) { auto const & fbs = outerIt->GetFbs(); for (auto const & fb : fbs) { - if (!(fb.IsArea() && ftypes::IsCityTownOrVillage(fb.GetTypes()))) + if (!(fb.IsArea() && isCityTownOrVillage(fb.GetTypes()))) continue; auto const id = fb.GetLastOsmId(); diff --git a/indexer/ftypes_matcher.cpp b/indexer/ftypes_matcher.cpp index e828e31f94..78e9a6f773 100644 --- a/indexer/ftypes_matcher.cpp +++ b/indexer/ftypes_matcher.cpp @@ -713,6 +713,32 @@ LocalityType IsLocalityChecker::GetType(FeatureType & f) const return GetType(types); } +IsCountryChecker::IsCountryChecker() +{ + Classificator const & c = classif(); + m_types.push_back(c.GetTypeByPath({"place", "country"})); +} + +IsStateChecker::IsStateChecker() +{ + Classificator const & c = classif(); + m_types.push_back(c.GetTypeByPath({"place", "state"})); +} + +IsCityTownOrVillageChecker::IsCityTownOrVillageChecker() +{ + vector> const types = { + {"place", "city"}, + {"place", "town"}, + {"place", "village"}, + {"place", "hamlet"} + }; + + Classificator const & c = classif(); + for (auto const & t : types) + m_types.push_back(c.GetTypeByPath({t.first, t.second})); +} + uint64_t GetPopulation(FeatureType & ft) { uint64_t population = ft.GetPopulation(); diff --git a/indexer/ftypes_matcher.hpp b/indexer/ftypes_matcher.hpp index 43fb6d4df8..e4c858fff1 100644 --- a/indexer/ftypes_matcher.hpp +++ b/indexer/ftypes_matcher.hpp @@ -430,15 +430,27 @@ public: DECLARE_CHECKER_INSTANCE(IsLocalityChecker); }; -template -bool IsCityTownOrVillage(Types const & types) + +class IsCountryChecker : public BaseChecker { - feature::TypesHolder h; - for (auto const t : types) - h.Add(t); - auto const type = IsLocalityChecker::Instance().GetType(h); - return type == LocalityType ::City || type == LocalityType ::Town || type == LocalityType ::Village; -} + IsCountryChecker(); +public: + DECLARE_CHECKER_INSTANCE(IsCountryChecker); +}; + +class IsStateChecker : public BaseChecker +{ + IsStateChecker(); +public: + DECLARE_CHECKER_INSTANCE(IsStateChecker); +}; + +class IsCityTownOrVillageChecker : public BaseChecker +{ + IsCityTownOrVillageChecker(); +public: + DECLARE_CHECKER_INSTANCE(IsCityTownOrVillageChecker); +}; /// @name Get city radius and population. /// @param r Radius in meters. diff --git a/search/categories_cache.cpp b/search/categories_cache.cpp index 3552e95d31..f93d608324 100644 --- a/search/categories_cache.cpp +++ b/search/categories_cache.cpp @@ -72,9 +72,21 @@ VillagesCache::VillagesCache(base::Cancellable const & cancellable) { } -// LocalitiesCache ---------------------------------------------------------------------------------- -LocalitiesCache::LocalitiesCache(base::Cancellable const & cancellable) - : CategoriesCache(ftypes::IsLocalityChecker::Instance(), cancellable) +// CountriesCache ---------------------------------------------------------------------------------- +CountriesCache::CountriesCache(base::Cancellable const & cancellable) + : CategoriesCache(ftypes::IsCountryChecker::Instance(), cancellable) +{ +} + +// StatesCache ------------------------------------------------------------------------------------- +StatesCache::StatesCache(base::Cancellable const & cancellable) + : CategoriesCache(ftypes::IsStateChecker::Instance(), cancellable) +{ +} + +// CitiesTownsOrVillagesCache ---------------------------------------------------------------------- +CitiesTownsOrVillagesCache::CitiesTownsOrVillagesCache(base::Cancellable const & cancellable) + : CategoriesCache(ftypes::IsCityTownOrVillageChecker::Instance(), cancellable) { } diff --git a/search/categories_cache.hpp b/search/categories_cache.hpp index c181243ef8..b2d6f699c3 100644 --- a/search/categories_cache.hpp +++ b/search/categories_cache.hpp @@ -64,10 +64,24 @@ public: VillagesCache(base::Cancellable const & cancellable); }; -class LocalitiesCache : public CategoriesCache +class CountriesCache : public CategoriesCache { public: - LocalitiesCache(base::Cancellable const & cancellable); + CountriesCache(base::Cancellable const & cancellable); +}; + +class StatesCache : public CategoriesCache +{ +public: + StatesCache(base::Cancellable const & cancellable); +}; + +// Used for cities/towns/villages from world. Currently we do not have villages in World.mwm but +// it may be good to put some important villages to it: mountain/beach resorts. +class CitiesTownsOrVillagesCache : public CategoriesCache +{ +public: + CitiesTownsOrVillagesCache(base::Cancellable const & cancellable); }; class HotelsCache : public CategoriesCache diff --git a/search/geocoder.cpp b/search/geocoder.cpp index ab4447d998..5db712206b 100644 --- a/search/geocoder.cpp +++ b/search/geocoder.cpp @@ -61,20 +61,14 @@ namespace search { namespace { -size_t constexpr kMaxNumCities = 5; -size_t constexpr kMaxNumStates = 5; +size_t constexpr kMaxNumCities = 10; +size_t constexpr kMaxNumStates = 10; size_t constexpr kMaxNumVillages = 5; -size_t constexpr kMaxNumCountries = 5; +size_t constexpr kMaxNumCountries = 10; double constexpr kMaxViewportRadiusM = 50.0 * 1000; double constexpr kMaxPostcodeRadiusM = 1000; double constexpr kMaxSuburbRadiusM = 2000; -// This constant limits number of localities that will be extracted -// from World map. Villages are not counted here as they're not -// included into World map. -// @vng Set this value to possible maximum. -size_t const kMaxNumLocalities = LocalityScorer::kDefaultReadLimit; - size_t constexpr kPivotRectsCacheSize = 10; size_t constexpr kPostcodesRectsCacheSize = 10; size_t constexpr kSuburbsRectsCacheSize = 10; @@ -299,19 +293,34 @@ bool Geocoder::ExtendedMwmInfos::ExtendedMwmInfo::operator<( return m_distance < rhs.m_distance; } +// Geocoder::LocalitiesCaches ---------------------------------------------------------------------- +Geocoder::LocalitiesCaches::LocalitiesCaches(base::Cancellable const & cancellable) + : m_countries(cancellable) + , m_states(cancellable) + , m_citiesTownsOrVillages(cancellable) + , m_villages(cancellable) +{ +} + +void Geocoder::LocalitiesCaches::Clear() +{ + m_countries.Clear(); + m_states.Clear(); + m_citiesTownsOrVillages.Clear(); + m_villages.Clear(); +} + // Geocoder::Geocoder ------------------------------------------------------------------------------ Geocoder::Geocoder(DataSource const & dataSource, storage::CountryInfoGetter const & infoGetter, CategoriesHolder const & categories, CitiesBoundariesTable const & citiesBoundaries, PreRanker & preRanker, - VillagesCache & villagesCache, LocalitiesCache & localitiesCache, - base::Cancellable const & cancellable) + LocalitiesCaches & localitiesCaches, base::Cancellable const & cancellable) : m_dataSource(dataSource) , m_infoGetter(infoGetter) , m_categories(categories) , m_streetsCache(cancellable) , m_suburbsCache(cancellable) - , m_villagesCache(villagesCache) - , m_localitiesCache(localitiesCache) + , m_localitiesCaches(localitiesCaches) , m_hotelsCache(cancellable) , m_foodCache(cancellable) , m_hotelsFilter(m_hotelsCache) @@ -582,7 +591,7 @@ void Geocoder::GoImpl(vector> const & infos, bool inViewport features = features.Intersect(viewportCBV); } - ctx.m_villages = m_villagesCache.Get(*m_context); + ctx.m_villages = m_localitiesCaches.m_villages.Get(*m_context); auto citiesFromWorld = m_cities; FillVillageLocalities(ctx); @@ -680,102 +689,106 @@ void Geocoder::FillLocalityCandidates(BaseContext const & ctx, CBV const & filte void Geocoder::CacheWorldLocalities() { if (auto context = GetWorldContext(m_dataSource)) - UNUSED_VALUE(m_localitiesCache.Get(*context)); + { + // Get only world localities + UNUSED_VALUE(m_localitiesCaches.m_countries.Get(*context)); + UNUSED_VALUE(m_localitiesCaches.m_states.Get(*context)); + UNUSED_VALUE(m_localitiesCaches.m_citiesTownsOrVillages.Get(*context)); + } } void Geocoder::FillLocalitiesTable(BaseContext const & ctx) { + auto addRegionMaps = [&](FeatureType & ft, Locality const & l, Region::Type type) { + if (ft.GetGeomType() != feature::GeomType::Point) + return; + + string affiliation; + if (!GetAffiliationName(ft, affiliation)) + return; + + Region region(l, type); + region.m_center = ft.GetCenter(); + + ft.GetName(StringUtf8Multilang::kDefaultCode, region.m_defaultName); + LOG(LDEBUG, ("Region =", region.m_defaultName)); + + m_infoGetter.GetMatchedRegions(affiliation, region.m_ids); + m_regions[type][l.m_tokenRange].push_back(region); + }; + vector preLocalities; - CBV filter = m_localitiesCache.Get(*m_context); - FillLocalityCandidates(ctx, filter, kMaxNumLocalities, preLocalities); - - size_t numCities = 0; - size_t numStates = 0; - size_t numCountries = 0; + CBV filter = m_localitiesCaches.m_countries.Get(*m_context); + FillLocalityCandidates(ctx, filter, kMaxNumCountries, preLocalities); for (auto & l : preLocalities) { auto ft = m_context->GetFeature(l.m_featureId); if (!ft) + { + LOG(LWARNING, ("Failed to get country from world", l.m_featureId)); continue; + } - auto addRegionMaps = [&](size_t maxCount, Region::Type type, size_t & count) { - if (count < maxCount && ft->GetGeomType() == feature::GeomType::Point) + addRegionMaps(*ft, l, Region::TYPE_COUNTRY); + } + + filter = m_localitiesCaches.m_states.Get(*m_context); + FillLocalityCandidates(ctx, filter, kMaxNumStates, preLocalities); + for (auto & l : preLocalities) + { + auto ft = m_context->GetFeature(l.m_featureId); + if (!ft) + { + LOG(LWARNING, ("Failed to get state from world", l.m_featureId)); + continue; + } + + addRegionMaps(*ft, l, Region::TYPE_STATE); + } + + filter = m_localitiesCaches.m_citiesTownsOrVillages.Get(*m_context); + FillLocalityCandidates(ctx, filter, kMaxNumCities, preLocalities); + for (auto & l : preLocalities) + { + auto ft = m_context->GetFeature(l.m_featureId); + if (!ft) + { + LOG(LWARNING, ("Failed to get city from world", l.m_featureId)); + continue; + } + + if (ft->GetGeomType() == feature::GeomType::Point) + { + City city(l, Model::TYPE_CITY); + + CitiesBoundariesTable::Boundaries boundaries; + bool haveBoundary = false; + if (m_citiesBoundaries.Get(ft->GetID(), boundaries)) { - string affiliation; - if (!GetAffiliationName(*ft, affiliation)) - return; - - Region region(l, type); - region.m_center = ft->GetCenter(); - - ft->GetName(StringUtf8Multilang::kDefaultCode, region.m_defaultName); - LOG(LDEBUG, ("Region =", region.m_defaultName)); - - m_infoGetter.GetMatchedRegions(affiliation, region.m_ids); - if (region.m_ids.empty()) - { - LOG(LWARNING, - ("Maps not found for region:", region.m_defaultName, "affiliation:", affiliation)); - } - - ++count; - m_regions[type][l.m_tokenRange].push_back(region); + city.m_rect = boundaries.GetLimitRect(); + if (city.m_rect.IsValid()) + haveBoundary = true; } - }; - switch (m_model.GetType(*ft)) - { - case Model::TYPE_CITY: - { - if (numCities < kMaxNumCities && ft->GetGeomType() == feature::GeomType::Point) + if (!haveBoundary) { - ++numCities; - - City city(l, Model::TYPE_CITY); - - CitiesBoundariesTable::Boundaries boundaries; - bool haveBoundary = false; - if (m_citiesBoundaries.Get(ft->GetID(), boundaries)) - { - city.m_rect = boundaries.GetLimitRect(); - if (city.m_rect.IsValid()) - haveBoundary = true; - } - - if (!haveBoundary) - { - auto const center = feature::GetCenter(*ft); - auto const population = ftypes::GetPopulation(*ft); - auto const radius = ftypes::GetRadiusByPopulation(population); - city.m_rect = mercator::RectByCenterXYAndSizeInMeters(center, radius); - } + auto const center = feature::GetCenter(*ft); + auto const population = ftypes::GetPopulation(*ft); + auto const radius = ftypes::GetRadiusByPopulation(population); + city.m_rect = mercator::RectByCenterXYAndSizeInMeters(center, radius); + } #if defined(DEBUG) - ft->GetName(StringUtf8Multilang::kDefaultCode, city.m_defaultName); - LOG(LINFO, - ("City =", city.m_defaultName, "rect =", city.m_rect, "rect source:", haveBoundary ? "table" : "population", - "sizeX =", - mercator::DistanceOnEarth(city.m_rect.LeftTop(), city.m_rect.RightTop()), - "sizeY =", - mercator::DistanceOnEarth(city.m_rect.LeftTop(), city.m_rect.LeftBottom()))); + ft->GetName(StringUtf8Multilang::kDefaultCode, city.m_defaultName); + LOG(LINFO, + ("City =", city.m_defaultName, "rect =", city.m_rect, + "rect source:", haveBoundary ? "table" : "population", + "sizeX =", mercator::DistanceOnEarth(city.m_rect.LeftTop(), city.m_rect.RightTop()), + "sizeY =", mercator::DistanceOnEarth(city.m_rect.LeftTop(), city.m_rect.LeftBottom()))); #endif - m_cities[city.m_tokenRange].push_back(city); - } - break; - } - case Model::TYPE_STATE: - { - addRegionMaps(kMaxNumStates, Region::TYPE_STATE, numStates); - break; - } - case Model::TYPE_COUNTRY: - { - addRegionMaps(kMaxNumCountries, Region::TYPE_COUNTRY, numCountries); - break; - } - default: break; + m_cities[city.m_tokenRange].push_back(city); } } } diff --git a/search/geocoder.hpp b/search/geocoder.hpp index b02ab4e923..3c43a9afbf 100644 --- a/search/geocoder.hpp +++ b/search/geocoder.hpp @@ -96,9 +96,20 @@ public: double m_villageSearchRadiusM = 0.0; }; + struct LocalitiesCaches + { + LocalitiesCaches(base::Cancellable const & cancellable); + void Clear(); + + CountriesCache m_countries; + StatesCache m_states; + CitiesTownsOrVillagesCache m_citiesTownsOrVillages; + VillagesCache m_villages; + }; + Geocoder(DataSource const & dataSource, storage::CountryInfoGetter const & infoGetter, CategoriesHolder const & categories, CitiesBoundariesTable const & citiesBoundaries, - PreRanker & preRanker, VillagesCache & villagesCache, LocalitiesCache & localitiesCache, + PreRanker & preRanker, LocalitiesCaches & localitiesCaches, base::Cancellable const & cancellable); ~Geocoder(); @@ -306,8 +317,7 @@ private: StreetsCache m_streetsCache; SuburbsCache m_suburbsCache; - VillagesCache & m_villagesCache; - LocalitiesCache & m_localitiesCache; + LocalitiesCaches & m_localitiesCaches; HotelsCache m_hotelsCache; FoodCache m_foodCache; hotels_filter::HotelsFilter m_hotelsFilter; diff --git a/search/processor.cpp b/search/processor.cpp index 04d9efa78e..8eb673a880 100644 --- a/search/processor.cpp +++ b/search/processor.cpp @@ -160,15 +160,14 @@ Processor::Processor(DataSource const & dataSource, CategoriesHolder const & cat : m_categories(categories) , m_infoGetter(infoGetter) , m_dataSource(dataSource) - , m_villagesCache(static_cast(*this)) - , m_localitiesCache(static_cast(*this)) + , m_localitiesCaches(static_cast(*this)) , m_citiesBoundaries(m_dataSource) , m_keywordsScorer(LanguageTier::LANGUAGE_TIER_COUNT) , m_ranker(m_dataSource, m_citiesBoundaries, infoGetter, m_keywordsScorer, m_emitter, categories, - suggests, m_villagesCache, static_cast(*this)) + suggests, m_localitiesCaches.m_villages, static_cast(*this)) , m_preRanker(m_dataSource, m_ranker) , m_geocoder(m_dataSource, infoGetter, categories, m_citiesBoundaries, m_preRanker, - m_villagesCache, m_localitiesCache, static_cast(*this)) + m_localitiesCaches, static_cast(*this)) , m_bookmarksProcessor(m_emitter, static_cast(*this)) { // Current and input langs are to be set later. @@ -794,8 +793,7 @@ void Processor::InitEmitter(SearchParams const & searchParams) void Processor::ClearCaches() { m_geocoder.ClearCaches(); - m_villagesCache.Clear(); - m_localitiesCache.Clear(); + m_localitiesCaches.Clear(); m_preRanker.ClearCaches(); m_ranker.ClearCaches(); m_viewport.MakeEmpty(); diff --git a/search/processor.hpp b/search/processor.hpp index 0a77d7466e..82d5b1c310 100644 --- a/search/processor.hpp +++ b/search/processor.hpp @@ -147,8 +147,7 @@ protected: DataSource const & m_dataSource; - VillagesCache m_villagesCache; - LocalitiesCache m_localitiesCache; + Geocoder::LocalitiesCaches m_localitiesCaches; CitiesBoundariesTable m_citiesBoundaries; KeywordLangMatcher m_keywordsScorer;