From b87699ccc8b9ee23690b1ded15a96378497fe7e8 Mon Sep 17 00:00:00 2001 From: Alex Zolotarev Date: Thu, 17 Mar 2016 13:28:02 +0300 Subject: [PATCH] [search] Print street, house number for buildings in search results. Also print addresses. --- .../jni/com/mapswithme/maps/SearchEngine.cpp | 15 +---- .../Search/TableView/MWMSearchCommonCell.mm | 3 +- map/framework.cpp | 36 ---------- map/framework.hpp | 1 - search/intermediate_result.cpp | 66 ++++++++++++------- search/intermediate_result.hpp | 14 ++-- search/result.cpp | 34 ++-------- search/result.hpp | 13 +--- search/search_query.cpp | 18 ++--- search/search_query.hpp | 3 + 10 files changed, 67 insertions(+), 136 deletions(-) diff --git a/android/jni/com/mapswithme/maps/SearchEngine.cpp b/android/jni/com/mapswithme/maps/SearchEngine.cpp index b78ddcc14a..877b302dc9 100644 --- a/android/jni/com/mapswithme/maps/SearchEngine.cpp +++ b/android/jni/com/mapswithme/maps/SearchEngine.cpp @@ -39,7 +39,7 @@ jobject ToJavaResult(Result & result, bool hasPosition, double lat, double lon) ::Framework * fr = g_framework->NativeFramework(); jni::TScopedLocalIntArrayRef ranges(env, env->NewIntArray(result.GetHighlightRangesCount() * 2)); - jint * rawArr = env->GetIntArrayElements(ranges.get(), nullptr); + jint * rawArr = env->GetIntArrayElements(ranges, nullptr); for (int i = 0; i < result.GetHighlightRangesCount(); i++) { auto const & range = result.GetHighlightRange(i); @@ -69,21 +69,12 @@ jobject ToJavaResult(Result & result, bool hasPosition, double lat, double lon) return ret; } - search::AddressInfo info; - if (result.GetResultType() == Result::RESULT_FEATURE) - { - fr->LoadSearchResultMetadata(result); - info = fr->GetFeatureAddressInfo(result.GetFeatureID()); - } - else if (result.HasPoint()) - info = fr->GetAddressInfoAtPoint(result.GetFeatureCenter()); - jni::TScopedLocalRef featureType(env, jni::ToJavaString(env, result.GetFeatureType())); - jni::TScopedLocalRef region(env, jni::ToJavaString(env, info.FormatAddress(search::AddressInfo::SEARCH_RESULT))); + jni::TScopedLocalRef address(env, jni::ToJavaString(env, result.GetAddress())); jni::TScopedLocalRef dist(env, jni::ToJavaString(env, distance)); jni::TScopedLocalRef cuisine(env, jni::ToJavaString(env, result.GetCuisine())); jni::TScopedLocalRef desc(env, env->NewObject(g_descriptionClass, g_descriptionConstructor, - featureType.get(), region.get(), + featureType.get(), address.get(), dist.get(), cuisine.get(), result.GetStarsCount(), static_cast(result.IsOpenNow()))); diff --git a/iphone/Maps/Classes/CustomViews/MapViewControls/Search/TableView/MWMSearchCommonCell.mm b/iphone/Maps/Classes/CustomViews/MapViewControls/Search/TableView/MWMSearchCommonCell.mm index 67ee74c947..8428e8d3a3 100644 --- a/iphone/Maps/Classes/CustomViews/MapViewControls/Search/TableView/MWMSearchCommonCell.mm +++ b/iphone/Maps/Classes/CustomViews/MapViewControls/Search/TableView/MWMSearchCommonCell.mm @@ -29,8 +29,7 @@ { [super config:result]; self.typeLabel.text = @(result.GetFeatureType().c_str()).capitalizedString; - search::AddressInfo const info = GetFramework().GetSearchResultAddress(result); - self.locationLabel.text = @(info.FormatAddress(search::AddressInfo::SEARCH_RESULT).c_str()); + self.locationLabel.text = @(result.GetAddress().c_str()); [self.locationLabel sizeToFit]; if (!forHeight) diff --git a/map/framework.cpp b/map/framework.cpp index 23a0336878..51b611ef7f 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -1286,42 +1286,6 @@ size_t Framework::ShowSearchResults(search::Results const & results) return count; } -search::AddressInfo Framework::GetSearchResultAddress(search::Result const & res) const -{ - search::AddressInfo info; - if (res.IsSuggest()) - return info; - - string const & type = res.GetFeatureType(); - if (!type.empty()) - info.m_types.push_back(type); - - // Assign name if it's not equal with type. - string const & name = res.GetString(); - if (name != type) - info.m_name = name; - - info.m_city = res.GetAddress(); - - /// @todo Optimize here according to the fact that feature is - /// already read in many cases during search results processing. - auto const & id = res.GetFeatureID(); - if (id.IsValid()) - { - Index::FeaturesLoaderGuard loader(m_model.GetIndex(), id.m_mwmId); - FeatureType ft; - loader.GetFeatureByIndex(id.m_index, ft); - if (ft.GetFeatureType() == feature::GEOM_LINE || - !ftypes::IsAddressObjectChecker::Instance()(ft)) - { - return info; - } - } - - info = GetAddressInfoAtPoint(res.GetFeatureCenter()); - return info; -} - void Framework::FillSearchResultsMarks(search::Results const & results) { UserMarkControllerGuard guard(m_bmManager, UserMarkType::SEARCH_MARK); diff --git a/map/framework.hpp b/map/framework.hpp index 6129a4df4d..a617b44cb7 100644 --- a/map/framework.hpp +++ b/map/framework.hpp @@ -399,7 +399,6 @@ public: void ShowSearchResult(search::Result const & res); size_t ShowSearchResults(search::Results const & results); - search::AddressInfo GetSearchResultAddress(search::Result const & res) const; void StartInteractiveSearch(search::SearchParams const & params); bool IsInteractiveSearchActive() const { return !m_lastInteractiveSearchParams.m_query.empty(); } diff --git a/search/intermediate_result.cpp b/search/intermediate_result.cpp index bbdfac0cd6..bf20a2fb3c 100644 --- a/search/intermediate_result.cpp +++ b/search/intermediate_result.cpp @@ -1,5 +1,6 @@ #include "intermediate_result.hpp" #include "geometry_utils.hpp" +#include "reverse_geocoder.hpp" #include "storage/country_info_getter.hpp" @@ -91,7 +92,7 @@ PreResult2::PreResult2(FeatureType const & f, PreResult1 const * p, m2::PointD c : m_id(f.GetID()), m_types(f), m_str(displayName), - m_resultType(RESULT_FEATURE), + m_resultType(ftypes::IsBuildingChecker::Instance()(m_types) ? RESULT_BUILDING : RESULT_FEATURE), m_geomType(f.GetFeatureType()) { ASSERT(m_id.IsValid(), ()); @@ -115,14 +116,6 @@ PreResult2::PreResult2(double lat, double lon) m_region.SetParams(string(), MercatorBounds::FromLatLon(lat, lon)); } -PreResult2::PreResult2(m2::PointD const & pt, string const & str, uint32_t type) - : m_str(str), m_resultType(RESULT_BUILDING) -{ - m_region.SetParams(string(), pt); - - m_types.Assign(type); -} - namespace { class SkipRegionInfo @@ -168,40 +161,63 @@ string PreResult2::GetRegionName(storage::CountryInfoGetter const & infoGetter, return info.m_name; } +namespace +{ +// TODO: Format street and house number according to local country's rules. +string FormatStreetAndHouse(ReverseGeocoder::Address const & addr) +{ + ASSERT_GREATER_OR_EQUAL(addr.GetDistance(), 0, ()); + return addr.GetStreetName() + ", " + addr.GetHouseNumber(); +} +// TODO: Share common formatting code for search results and place page. +string FormatFullAddress(ReverseGeocoder::Address const & addr, string const & region) +{ + // TODO: Print "near" for not exact addresses. + string res; + //if (addr.GetDistance() >= 0 && addr.GetDistance() < 50.0) + if (addr.GetDistance() == 0) + res = FormatStreetAndHouse(addr); + else + return region; + return res + ", " + region; +} +} // namespace + Result PreResult2::GenerateFinalResult(storage::CountryInfoGetter const & infoGetter, CategoriesHolder const * pCat, - set const * pTypes, int8_t locale) const + set const * pTypes, int8_t locale, + ReverseGeocoder const & coder) const { + ReverseGeocoder::Address addr; + coder.GetNearbyAddress(GetCenter(), addr); + + // Insert exact address (street and house number) instead of empty result name. + string name; + if (m_str.empty() && addr.GetDistance() == 0) + name = FormatStreetAndHouse(addr); + else + name = m_str; + uint32_t const type = GetBestType(pTypes); - string const regionName = GetRegionName(infoGetter, type); + // TODO: GetRegionName should return City, State, Country instead of Country, State, City. + string const address = FormatFullAddress(addr, GetRegionName(infoGetter, type)); switch (m_resultType) { case RESULT_FEATURE: - return Result(m_id, GetCenter(), m_str, regionName, pCat->GetReadableFeatureType(type, locale) + case RESULT_BUILDING: + return Result(m_id, GetCenter(), name, address, pCat->GetReadableFeatureType(type, locale) #ifdef DEBUG + ' ' + strings::to_string(static_cast(m_info.m_rank)) #endif , type, m_metadata); - case RESULT_BUILDING: - return Result(GetCenter(), m_str, regionName, pCat->GetReadableFeatureType(type, locale)); - default: ASSERT_EQUAL(m_resultType, RESULT_LATLON, ()); - return Result(GetCenter(), m_str, regionName, string()); + return Result(GetCenter(), m_str, address); } } -Result PreResult2::GeneratePointResult(storage::CountryInfoGetter const & infoGetter, - CategoriesHolder const * pCat, - set const * pTypes, int8_t locale) const -{ - uint32_t const type = GetBestType(pTypes); - return Result(m_id, GetCenter(), m_str, GetRegionName(infoGetter, type), - pCat->GetReadableFeatureType(type, locale)); -} - // static bool PreResult2::LessRank(PreResult2 const & r1, PreResult2 const & r2) { diff --git a/search/intermediate_result.hpp b/search/intermediate_result.hpp index 6f88f16b24..5fe416505b 100644 --- a/search/intermediate_result.hpp +++ b/search/intermediate_result.hpp @@ -17,6 +17,7 @@ struct CountryInfo; namespace search { +class ReverseGeocoder; namespace impl { /// First pass results class. Objects are creating during search in trie. @@ -60,19 +61,16 @@ public: { RESULT_LATLON, RESULT_FEATURE, - RESULT_BUILDING + RESULT_BUILDING //!< Buildings are not filtered out in duplicates filter. }; - /// For RESULT_FEATURE. + /// For RESULT_FEATURE and RESULT_BUILDING. PreResult2(FeatureType const & f, PreResult1 const * p, m2::PointD const & pivot, string const & displayName, string const & fileName); /// For RESULT_LATLON. PreResult2(double lat, double lon); - /// For RESULT_BUILDING. - PreResult2(m2::PointD const & pt, string const & str, uint32_t type); - inline search::v2::RankingInfo const & GetRankingInfo() const { return m_info; } template @@ -87,11 +85,7 @@ public: /// @param[in] lang Current system language. Result GenerateFinalResult(storage::CountryInfoGetter const & infoGetter, CategoriesHolder const * pCat, set const * pTypes, - int8_t locale) const; - - Result GeneratePointResult(storage::CountryInfoGetter const & infoGetter, - CategoriesHolder const * pCat, - set const * pTypes, int8_t locale) const; + int8_t locale, ReverseGeocoder const & coder) const; static bool LessRank(PreResult2 const & r1, PreResult2 const & r2); static bool LessDistance(PreResult2 const & r1, PreResult2 const & r2); diff --git a/search/result.cpp b/search/result.cpp index fc986236a4..acdd8fd7fd 100644 --- a/search/result.cpp +++ b/search/result.cpp @@ -9,29 +9,16 @@ Result::Result(FeatureID const & id, m2::PointD const & pt, string const & str, Metadata const & meta) : m_id(id) , m_center(pt) - , m_str(str) + , m_str(str.empty() ? type : str) //!< Features with empty names can be found after suggestion. , m_address(address) , m_type(type) , m_featureType(featureType) , m_metadata(meta) { - Init(true /* metadataInitialized */); } -Result::Result(FeatureID const & id, m2::PointD const & pt, string const & str, - string const & address, string const & type) - : m_id(id) - , m_center(pt) - , m_str(str) - , m_address(address) - , m_type(type) -{ - Init(false /* metadataInitialized */); -} - -Result::Result(m2::PointD const & pt, string const & str, string const & address, - string const & type) - : m_center(pt), m_str(str), m_address(address), m_type(type) +Result::Result(m2::PointD const & pt, string const & latlon, string const & address) + : m_center(pt), m_str(latlon), m_address(address) { } @@ -52,15 +39,6 @@ Result::Result(Result const & res, string const & suggest) { } -void Result::Init(bool metadataInitialized) -{ - // Features with empty names can be found after suggestion. - if (m_str.empty()) - m_str = m_type; - - m_metadata.m_isInitialized = metadataInitialized; -} - Result::ResultType Result::GetResultType() const { bool const idValid = m_id.IsValid(); @@ -144,9 +122,9 @@ pair const & Result::GetHighlightRange(size_t idx) const void Result::AppendCity(string const & name) { - // No need to store mwm file name if we have valid city name. - if (!name.empty()) - m_address = name; + // Append only if city is absent in region (mwm) name. + if (m_address.find(name) == string::npos) + m_address = name + ", " + m_address; } string Result::ToStringForStats() const diff --git a/search/result.hpp b/search/result.hpp index 4da4d7956e..d0cec78b53 100644 --- a/search/result.hpp +++ b/search/result.hpp @@ -41,15 +41,10 @@ public: /// For RESULT_FEATURE. Result(FeatureID const & id, m2::PointD const & pt, string const & str, string const & address, - string const & type, uint32_t featureType, Metadata const & meta); + string const & type, uint32_t featureType, Metadata const & meta = {}); - /// Used for generation viewport results. - Result(FeatureID const & id, m2::PointD const & pt, string const & str, - string const & address, string const & type); - - /// @param[in] type Empty string - RESULT_LATLON, building address otherwise. - Result(m2::PointD const & pt, string const & str, - string const & address, string const & type); + /// For RESULT_LATLON. + Result(m2::PointD const & pt, string const & latlon, string const & address); /// For RESULT_SUGGESTION_PURE. Result(string const & str, string const & suggest); @@ -112,8 +107,6 @@ public: string ToStringForStats() const; private: - void Init(bool metadataInitialized); - FeatureID m_id; m2::PointD m_center; string m_str, m_address, m_type; diff --git a/search/search_query.cpp b/search/search_query.cpp index 98e2b09039..4ed4fe5d8c 100644 --- a/search/search_query.cpp +++ b/search/search_query.cpp @@ -23,6 +23,7 @@ #include "indexer/feature.hpp" #include "indexer/feature_algo.hpp" #include "indexer/feature_covering.hpp" +#include "indexer/feature_data.hpp" #include "indexer/feature_impl.hpp" #include "indexer/features_vector.hpp" #include "indexer/index.hpp" @@ -184,6 +185,7 @@ Query::Query(Index & index, CategoriesHolder const & categories, vector , m_suggestsEnabled(true) , m_viewportSearch(false) , m_keepHouseNumberInQuery(false) + , m_reverseGeocoder(index) { // Initialize keywords scorer. // Note! This order should match the indexes arrays above. @@ -513,9 +515,8 @@ void Query::FlushViewportResults(v2::Geocoder::Params const & params, Results & { if (IsCancelled()) break; - res.AddResultNoChecks((*(indV[i])) - .GeneratePointResult(m_infoGetter, &m_categories, &m_prefferedTypes, - m_currentLocaleCode)); + res.AddResultNoChecks((*(indV[i])).GenerateFinalResult(m_infoGetter, &m_categories, + &m_prefferedTypes, m_currentLocaleCode, m_reverseGeocoder)); } } @@ -578,14 +579,6 @@ class PreResult2Maker m_query.GetBestMatchName(f, name); - // It's invalid for a building to have an empty name if it has a - // house number - it will be merged with other buildings in a - // MakePreResult2(). To prevent this, house number is used as a - // building name here, if the latter is empty. - auto const & checker = ftypes::IsBuildingChecker::Instance(); - if (checker(f) && name.empty()) - name = f.GetHouseNumber(); - // country (region) name is a file name if feature isn't from World.mwm if (m_pFV->IsWorld()) country.clear(); @@ -1039,7 +1032,8 @@ public: Result Query::MakeResult(impl::PreResult2 const & r) const { Result res = - r.GenerateFinalResult(m_infoGetter, &m_categories, &m_prefferedTypes, m_currentLocaleCode); + r.GenerateFinalResult(m_infoGetter, &m_categories, &m_prefferedTypes, m_currentLocaleCode, + m_reverseGeocoder); MakeResultHighlight(res); #ifdef FIND_LOCALITY_TEST diff --git a/search/search_query.hpp b/search/search_query.hpp index 16ff5ef2be..7064a39968 100644 --- a/search/search_query.hpp +++ b/search/search_query.hpp @@ -2,6 +2,7 @@ #include "search/intermediate_result.hpp" #include "search/keyword_lang_matcher.hpp" #include "search/mode.hpp" +#include "search/reverse_geocoder.hpp" #include "search/search_trie.hpp" #include "search/suggest.hpp" #include "search/v2/geocoder.hpp" @@ -286,6 +287,8 @@ protected: bool m_viewportSearch; bool m_keepHouseNumberInQuery; //@} + + search::ReverseGeocoder const m_reverseGeocoder; }; } // namespace search