diff --git a/generator/generator_tests/osm_type_test.cpp b/generator/generator_tests/osm_type_test.cpp index ab0ebb4e0f..3383675c74 100644 --- a/generator/generator_tests/osm_type_test.cpp +++ b/generator/generator_tests/osm_type_test.cpp @@ -771,3 +771,40 @@ UNIT_TEST(OsmType_Entrance) TEST(params.IsTypeExist(GetType({"barrier", "entrance"})), (params)); } } + +UNIT_TEST(OsmType_Moscow) +{ + { + char const * arr[][2] = { + { "addr:country", "RU" }, + { "addr:region", "Москва" }, + { "admin_level", "2" }, + { "alt_name:vi", "Mạc Tư Khoa" }, + { "capital", "yes" }, + { "ele", "156" }, + { "int_name", "Moscow" }, + { "is_capital", "country" }, + { "ISO3166-2", "RU-MOW" }, + { "name", "Москва" }, + { "note", "эта точка должна быть здесь, в историческом центре Москвы" }, + { "official_status", "ru:город" }, + { "okato:user", "none" }, + { "place", "city" }, + { "population", "12108257" }, + { "population:date", "2014-01-01" }, + { "rank", "0" }, + { "wikipedia", "ru:Москва" }, + }; + + OsmElement e; + FillXmlElement(arr, ARRAY_SIZE(arr), &e); + + FeatureParams params; + ftype::GetNameAndType(&e, params); + + TEST_EQUAL(params.m_Types.size(), 1, (params)); + TEST(params.IsTypeExist(GetType({"place", "city", "capital", "2"})), (params)); + TEST(170 <= params.rank && params.rank <= 180, (params)); + TEST(!params.name.IsEmpty(), (params)); + } +} diff --git a/generator/osm2type.cpp b/generator/osm2type.cpp index 0aa0cfa00b..3e718d5ac4 100644 --- a/generator/osm2type.cpp +++ b/generator/osm2type.cpp @@ -3,12 +3,13 @@ #include "generator/osm_element.hpp" #include "indexer/classificator.hpp" +#include "indexer/feature_impl.hpp" #include "indexer/feature_visibility.hpp" + #include "geometry/mercator.hpp" #include "base/assert.hpp" #include "base/string_utils.hpp" -#include "base/math.hpp" #include "std/vector.hpp" #include "std/bind.hpp" @@ -197,7 +198,7 @@ namespace ftype { for (auto const & rule: rules) { - if (strcmp(e.key.data(), rule.key) != 0) + if (e.key != rule.key) continue; bool take = false; if (rule.value[0] == '*') @@ -539,18 +540,19 @@ namespace ftype { "population", "*", [¶ms](string & k, string & v) { // Get population rank. - // TODO: similar formula with indexer/feature.cpp, possible need refactoring uint64_t n; if (strings::to_uint64(v, n)) - params.rank = static_cast(log(double(n)) / log(1.1)); + params.rank = feature::PopulationToRank(n); k.clear(); v.clear(); - }}, + } + }, { "ref", "*", [¶ms](string & k, string & v) { // Get reference (we process road numbers only). params.ref = v; k.clear(); v.clear(); - }}, + } + }, { "layer", "*", [¶ms](string & k, string & v) { // Get layer. @@ -560,7 +562,8 @@ namespace ftype int8_t const bound = 10; params.layer = my::clamp(params.layer, -bound, bound); } - }}, + } + }, }); // Stage4: Match tags in classificator to find feature types. diff --git a/generator/search_index_builder.cpp b/generator/search_index_builder.cpp index c49aff6a8a..f129fe0ca5 100644 --- a/generator/search_index_builder.cpp +++ b/generator/search_index_builder.cpp @@ -7,6 +7,7 @@ #include "indexer/categories_holder.hpp" #include "indexer/classificator.hpp" #include "indexer/feature_algo.hpp" +#include "indexer/feature_impl.hpp" #include "indexer/feature_utils.hpp" #include "indexer/feature_visibility.hpp" #include "indexer/features_vector.hpp" @@ -217,7 +218,7 @@ struct ValueBuilder // get BEST geometry rect of feature v.m_pt = feature::GetCenter(ft); - v.m_rank = feature::GetSearchRank(types, v.m_pt, ft.GetPopulation()); + v.m_rank = feature::PopulationToRank(ft.GetPopulation()); } }; diff --git a/indexer/feature.cpp b/indexer/feature.cpp index a5c43525ee..00e88cfd2c 100644 --- a/indexer/feature.cpp +++ b/indexer/feature.cpp @@ -3,6 +3,7 @@ #include "indexer/classificator.hpp" #include "indexer/feature_algo.hpp" +#include "indexer/feature_impl.hpp" #include "indexer/feature_loader_base.hpp" #include "indexer/feature_visibility.hpp" #include "indexer/osm_editor.hpp" @@ -499,8 +500,7 @@ uint8_t FeatureType::GetRank() const uint32_t FeatureType::GetPopulation() const { - uint8_t const r = GetRank(); - return (r == 0 ? 1 : static_cast(pow(1.1, r))); + return feature::RankToPopulation(GetRank()); } string FeatureType::GetRoadNumber() const diff --git a/indexer/feature_impl.cpp b/indexer/feature_impl.cpp index b2d53f55e4..a20f19f03d 100644 --- a/indexer/feature_impl.cpp +++ b/indexer/feature_impl.cpp @@ -2,6 +2,7 @@ #include "base/string_utils.hpp" #include "base/logging.hpp" +#include "base/math.hpp" namespace feature @@ -71,4 +72,15 @@ bool IsHouseNumber(strings::UniString const & s) { return (!s.empty() && IsDigit(s[0])); } + +uint8_t PopulationToRank(uint64_t p) +{ + return static_cast(min(0xFF, my::rounds(log(double(p)) / log(1.1)))); } + +uint64_t RankToPopulation(uint8_t r) +{ + return (r == 0 ? 1 : static_cast(my::rounds(pow(1.1, r)))); +} + +} // namespace feature diff --git a/indexer/feature_impl.hpp b/indexer/feature_impl.hpp index 4b24519de8..ea1ba79950 100644 --- a/indexer/feature_impl.hpp +++ b/indexer/feature_impl.hpp @@ -35,4 +35,7 @@ namespace feature bool IsHouseNumber(string const & s); bool IsHouseNumber(strings::UniString const & s); bool IsHouseNumberDeepCheck(strings::UniString const & s); -} + + uint8_t PopulationToRank(uint64_t p); + uint64_t RankToPopulation(uint8_t r); +} // namespace feature diff --git a/indexer/feature_utils.cpp b/indexer/feature_utils.cpp index 7f33492efa..e5469c6e9d 100644 --- a/indexer/feature_utils.cpp +++ b/indexer/feature_utils.cpp @@ -78,69 +78,6 @@ public: return scale; } - uint8_t GetSearchRank(TypesHolder const & types, m2::PointD const & pt, uint32_t population) const - { - for (uint32_t t : types) - { - if (IsEqual(t, m_TypeSmallVillage)) - population = max(population, static_cast(100)); - else if (IsEqual(t, m_TypeVillage)) - population = max(population, static_cast(1000)); - else if (t == m_TypeTown || IsEqual(t, m_TypeCounty)) - population = max(population, static_cast(10000)); - else if (t == m_TypeState) - { - m2::RectD usaRects[] = - { - // Continental part of USA - m2::RectD(-125.73195962769162293, 25.168771674082393019, - -66.925073086214325713, 56.956377399113392812), - // Alaska - m2::RectD(-151.0, 63.0, -148.0, 66.0), - // Hawaii - m2::RectD(-179.3665041396082529, 17.740790096801504205, - -153.92127500280855656, 31.043358939740215874), - // Canada - m2::RectD(-141.00315086636985029, 45.927730040557435132, - -48.663019303849921471, 162.92387487639103938) - }; - - bool isUSA = false; - for (size_t i = 0; i < ARRAY_SIZE(usaRects); ++i) - if (usaRects[i].IsPointInside(pt)) - { - isUSA = true; - break; - } - - if (isUSA) - { - //LOG(LINFO, ("USA state population = ", population)); - // Get rank equal to city for USA's states. - population = max(population, static_cast(500000)); - } - else - { - //LOG(LINFO, ("Other state population = ", population)); - // Reduce rank for other states. - population = population / 10; - population = max(population, static_cast(10000)); - population = min(population, static_cast(500000)); - } - } - else if (t == m_TypeCity) - population = max(population, static_cast(500000)); - else if (t == m_TypeCityCapital) - population = max(population, static_cast(1000000)); - else if (t == m_TypeCountry) - population = max(population, static_cast(2000000)); - else if (t == m_TypeContinent) - population = max(population, static_cast(20000000)); - } - - return static_cast(log(double(population)) / log(1.1)); - } - private: static int GetDefaultScale() { return scales::GetUpperComfortScale(); } @@ -210,9 +147,4 @@ int GetFeatureViewportScale(TypesHolder const & types) return impl::GetFeatureEstimator().GetViewportScale(types); } -uint8_t GetSearchRank(TypesHolder const & types, m2::PointD const & pt, uint32_t population) -{ - return impl::GetFeatureEstimator().GetSearchRank(types, pt, population); -} - } // namespace feature diff --git a/indexer/feature_utils.hpp b/indexer/feature_utils.hpp index faa721fdd0..1cfa5a38ac 100644 --- a/indexer/feature_utils.hpp +++ b/indexer/feature_utils.hpp @@ -11,8 +11,4 @@ namespace feature /// Get viewport scale to show given feature. Used in search. int GetFeatureViewportScale(TypesHolder const & types); - - /// Get search rank for a feature. - /// Roughly, rank + 1 means that feature is 1.x times more popular. - uint8_t GetSearchRank(TypesHolder const & types, m2::PointD const & pt, uint32_t population); } // namespace feature diff --git a/indexer/rank_table.cpp b/indexer/rank_table.cpp index c83ba77281..d7bdd48fab 100644 --- a/indexer/rank_table.cpp +++ b/indexer/rank_table.cpp @@ -2,9 +2,10 @@ #include "indexer/data_header.hpp" #include "indexer/feature_algo.hpp" +#include "indexer/feature_impl.hpp" #include "indexer/feature_utils.hpp" #include "indexer/features_vector.hpp" -#include "indexer/types_skipper.hpp" +#include "indexer/ftypes_matcher.hpp" #include "platform/local_country_file.hpp" #include "platform/local_country_file_utils.hpp" @@ -246,18 +247,7 @@ unique_ptr LoadRankTable(unique_ptr && region) // Calculates search rank for a feature. uint8_t CalcSearchRank(FeatureType const & ft) { - static search::TypesSkipper skipIndex; - - feature::TypesHolder types(ft); - skipIndex.SkipTypes(types); - if (types.Empty()) - return 0; - - // Rank (and population) is used for point features only at this moment. - if (ft.GetFeatureType() == feature::GEOM_POINT) - return feature::GetSearchRank(types, ft.GetCenter(), ft.GetPopulation()); - else - return 0; + return feature::PopulationToRank(ftypes::GetPopulation(ft)); } // Creates rank table if it does not exists in |rcont| or has wrong