diff --git a/coding/serdes_json.hpp b/coding/serdes_json.hpp index e6af971567..abd5fd3786 100644 --- a/coding/serdes_json.hpp +++ b/coding/serdes_json.hpp @@ -170,6 +170,12 @@ public: }); } + template + void operator()(Optional const & opt, Optional const &, char const * name = nullptr) + { + (*this)(opt, name); + } + protected: template void NewScopeWith(base::JSONPtr json_object, char const * name, Fn && fn) @@ -370,6 +376,19 @@ public: RestoreContext(outerContext); } + template + void operator()(Optional & opt, Optional const & defaultValue, char const * name = nullptr) + { + auto json = base::GetJSONOptionalField(m_json, name); + if (!json) + { + opt = defaultValue; + return; + } + + (*this)(opt, name); + } + protected: json_t * SaveContext(char const * name = nullptr) { diff --git a/map/CMakeLists.txt b/map/CMakeLists.txt index ad66e7f521..8a2aaa9dc1 100644 --- a/map/CMakeLists.txt +++ b/map/CMakeLists.txt @@ -79,6 +79,9 @@ set( mwm_url.hpp notifications/notification_manager.cpp notifications/notification_manager.hpp + notifications/notification_manager_delegate.cpp + notifications/notification_manager_delegate.hpp + notifications/notification_queue.cpp notifications/notification_queue.hpp notifications/notification_queue_serdes.cpp notifications/notification_queue_serdes.hpp diff --git a/map/framework.cpp b/map/framework.cpp index 79dca6fbe2..e716160c06 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -6,6 +6,7 @@ #include "map/ge0_parser.hpp" #include "map/geourl_process.hpp" #include "map/gps_tracker.hpp" +#include "map/notifications/notification_manager_delegate.hpp" #include "map/taxi_delegate.hpp" #include "map/user_mark.hpp" #include "map/utils.hpp" @@ -380,6 +381,9 @@ void Framework::Migrate(bool keepDownloaded) InitDiscoveryManager(); InitTaxiEngine(); RegisterAllMaps(); + m_notificationManager.SetDelegate( + std::make_unique(m_model.GetDataSource(), + *m_cityFinder, *m_ugcApi)); m_trafficManager.SetCurrentDataVersion(GetStorage().GetCurrentDataVersion()); if (m_drapeEngine && m_isRenderingEnabled) @@ -424,7 +428,6 @@ Framework::Framework(FrameworkParams const & params) , m_descriptionsLoader(std::make_unique(m_model.GetDataSource())) , m_purchase(std::make_unique()) , m_tipsApi(static_cast(*this)) - , m_notificationManager(static_cast(*this)) { CHECK(IsLittleEndian(), ("Only little-endian architectures are supported.")); @@ -547,6 +550,9 @@ Framework::Framework(FrameworkParams const & params) InitTransliteration(); LOG(LDEBUG, ("Transliterators initialized")); + m_notificationManager.SetDelegate( + std::make_unique(m_model.GetDataSource(), + *m_cityFinder, *m_ugcApi)); m_notificationManager.Load(); m_notificationManager.TrimExpired(); @@ -807,18 +813,7 @@ void Framework::ResetBookmarkInfo(Bookmark const & bmk, place_page::Info & info) search::ReverseGeocoder::Address Framework::GetAddressAtPoint(m2::PointD const & pt) const { - double const kDistanceThresholdMeters = 0.5; - - search::ReverseGeocoder const coder(m_model.GetDataSource()); - search::ReverseGeocoder::Address address; - coder.GetNearbyAddress(pt, address); - - // We do not init nearby address info for points that are located - // outside of the nearby building. - if (address.GetDistance() < kDistanceThresholdMeters) - return address; - - return {}; + return utils::GetAddressAtPoint(m_model.GetDataSource(), pt); } void Framework::FillFeatureInfo(FeatureID const & fid, place_page::Info & info) const diff --git a/map/framework.hpp b/map/framework.hpp index e0718e814e..3e5dc98c6c 100644 --- a/map/framework.hpp +++ b/map/framework.hpp @@ -145,7 +145,6 @@ struct FrameworkParams class Framework : public SearchAPI::Delegate, public RoutingManager::Delegate, public TipsApi::Delegate, - public notifications::NotificationManager::Delegate, private power_management::PowerManager::Subscriber { DISALLOW_COPY(Framework); @@ -263,8 +262,7 @@ public: booking::Api const * GetBookingApi(platform::NetworkPolicy const & policy) const; taxi::Engine * GetTaxiEngine(platform::NetworkPolicy const & policy); locals::Api * GetLocalsApi(platform::NetworkPolicy const & policy); - // NotificationManager::Delegate override. - ugc::Api * GetUGCApi() override { return m_ugcApi.get(); } + ugc::Api * GetUGCApi() { return m_ugcApi.get(); } ugc::Api const * GetUGCApi() const { return m_ugcApi.get(); } df::DrapeApi & GetDrapeApi() { return m_drapeApi; } diff --git a/map/map_tests/notification_tests.cpp b/map/map_tests/notification_tests.cpp index 53a551d159..a990150847 100644 --- a/map/map_tests/notification_tests.cpp +++ b/map/map_tests/notification_tests.cpp @@ -13,15 +13,30 @@ #include using namespace notifications; +using namespace std::chrono; namespace notifications { class NotificationManagerForTesting : public NotificationManager { public: - explicit NotificationManagerForTesting(NotificationManager::Delegate & delegate) - : NotificationManager(delegate) + class NotificationManagerDelegate : public NotificationManager::Delegate { + public: + ugc::Api & GetUGCApi() override + { + UNREACHABLE(); + } + + string GetAddress(m2::PointD const & pt) override + { + return {}; + } + }; + + NotificationManagerForTesting() + { + SetDelegate(std::make_unique()); } Queue & GetEditableQueue() { return m_queue; } @@ -30,6 +45,11 @@ public: { ProcessUgcRateCandidates(poi); } + + static void SetCreatedTime(NotificationCandidate & dst, Time time) + { + dst.m_created = time; + } }; } // namespace notifications @@ -44,54 +64,35 @@ public: } }; -class DelegateForTesting : public NotificationManager::Delegate -{ -public: - // NotificationManager::Delegate overrides: - ugc::Api * GetUGCApi() override - { - return nullptr; - } -}; - Queue MakeDefaultQueueForTesting() { Queue queue; { - NotificationCandidate notification; - notification.m_type = NotificationCandidate::Type::UgcReview; + eye::MapObject mapObject; + mapObject.SetBestType("cafe"); + mapObject.SetPos({15.686299, 73.704084}); + mapObject.SetReadableName("Baba"); - notification.m_mapObject = std::make_unique(); - notification.m_mapObject->SetBestType("cafe"); - notification.m_mapObject->SetPos({15.686299, 73.704084}); - notification.m_mapObject->SetReadableName("Baba"); - - queue.m_candidates.emplace_back(std::move(notification)); + queue.m_candidates.emplace_back(mapObject, ""); } { - NotificationCandidate notification; - notification.m_type = NotificationCandidate::Type::UgcReview; + eye::MapObject mapObject; + mapObject.SetBestType("shop"); + mapObject.SetPos({12.923975, 100.776627}); + mapObject.SetReadableName("7eleven"); - notification.m_mapObject = std::make_unique(); - notification.m_mapObject->SetBestType("shop"); - notification.m_mapObject->SetPos({12.923975, 100.776627}); - notification.m_mapObject->SetReadableName("7eleven"); - - queue.m_candidates.emplace_back(std::move(notification)); + queue.m_candidates.emplace_back(mapObject, ""); } { - NotificationCandidate notification; - notification.m_type = NotificationCandidate::Type::UgcReview; + eye::MapObject mapObject; + mapObject.SetBestType("viewpoint"); + mapObject.SetPos({-45.943995, 167.619933}); + mapObject.SetReadableName("Waiau"); - notification.m_mapObject = std::make_unique(); - notification.m_mapObject->SetBestType("viewpoint"); - notification.m_mapObject->SetPos({-45.943995, 167.619933}); - notification.m_mapObject->SetReadableName("Waiau"); - - queue.m_candidates.emplace_back(std::move(notification)); + queue.m_candidates.emplace_back(mapObject, ""); } return queue; @@ -107,11 +108,10 @@ void CompareWithDefaultQueue(Queue const & lhs) { auto const & lhsItem = lhs.m_candidates[i]; auto const & rhsItem = rhs.m_candidates[i]; - TEST_EQUAL(lhsItem.m_type, rhsItem.m_type, ()); - TEST(lhsItem.m_mapObject, ()); - TEST_EQUAL(lhsItem.m_mapObject->GetBestType(), rhsItem.m_mapObject->GetBestType(), ()); - TEST_EQUAL(lhsItem.m_mapObject->GetReadableName(), rhsItem.m_mapObject->GetReadableName(), ()); - TEST_EQUAL(lhsItem.m_mapObject->GetPos(), lhsItem.m_mapObject->GetPos(), ()); + TEST_EQUAL(lhsItem.GetType(), rhsItem.GetType(), ()); + TEST_EQUAL(lhsItem.GetBestFeatureType(), rhsItem.GetBestFeatureType(), ()); + TEST_EQUAL(lhsItem.GetReadableName(), rhsItem.GetReadableName(), ()); + TEST_EQUAL(lhsItem.GetPos(), lhsItem.GetPos(), ()); } } @@ -144,8 +144,7 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_QueueSaveLoadTest) UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckRouteToInSameGeoTrigger) { - DelegateForTesting delegate; - NotificationManagerForTesting notificationManager(delegate); + NotificationManagerForTesting notificationManager; eye::MapObject mapObject; mapObject.SetPos(MercatorBounds::FromLatLon({59.909299, 10.769807})); @@ -160,12 +159,14 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckRouteToInSam notificationManager.OnMapObjectEvent(mapObject); TEST_EQUAL(notificationManager.GetEditableQueue().m_candidates.size(), 1, ()); - notificationManager.GetEditableQueue().m_candidates[0].m_created = event.m_eventTime; + + auto & candidate = notificationManager.GetEditableQueue().m_candidates[0]; + NotificationManagerForTesting::SetCreatedTime(candidate, event.m_eventTime); auto result = notificationManager.GetNotification(); TEST(result.is_initialized(), ()); - TEST_EQUAL(result.get().m_type, NotificationCandidate::Type::UgcReview, ()); + TEST_EQUAL(result.get().GetType(), NotificationCandidate::Type::UgcReview, ()); result = notificationManager.GetNotification(); TEST(!result.is_initialized(), ()); @@ -173,8 +174,7 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckRouteToInSam UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckUgcNotSavedTrigger) { - DelegateForTesting delegate; - NotificationManagerForTesting notificationManager(delegate); + NotificationManagerForTesting notificationManager; eye::MapObject mapObject; mapObject.SetPos(MercatorBounds::FromLatLon({59.909299, 10.769807})); @@ -207,13 +207,13 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckUgcNotSavedT TEST(!result.is_initialized(), ()); - notificationManager.GetEditableQueue().m_candidates[0].m_created = - notifications::Clock::now() - std::chrono::hours(25); + auto & candidate = notificationManager.GetEditableQueue().m_candidates[0]; + NotificationManagerForTesting::SetCreatedTime(candidate, Clock::now() - hours(25)); result = notificationManager.GetNotification(); TEST(result.is_initialized(), ()); - TEST_EQUAL(result.get().m_type, NotificationCandidate::Type::UgcReview, ()); + TEST_EQUAL(result.get().GetType(), NotificationCandidate::Type::UgcReview, ()); result = notificationManager.GetNotification(); TEST(!result.is_initialized(), ()); @@ -242,8 +242,7 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckUgcNotSavedT UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckPlannedTripTrigger) { - DelegateForTesting delegate; - NotificationManagerForTesting notificationManager(delegate); + NotificationManagerForTesting notificationManager; eye::MapObject mapObject; mapObject.SetPos(MercatorBounds::FromLatLon({59.909299, 10.769807})); @@ -287,13 +286,13 @@ UNIT_CLASS_TEST(ScopedNotificationsQueue, Notifications_UgcRateCheckPlannedTripT TEST(!result.is_initialized(), ()); - notificationManager.GetEditableQueue().m_candidates[0].m_created = - notifications::Clock::now() - std::chrono::hours(25); + auto & candidate = notificationManager.GetEditableQueue().m_candidates[0]; + NotificationManagerForTesting::SetCreatedTime(candidate, Clock::now() - hours(25)); result = notificationManager.GetNotification(); TEST(result.is_initialized(), ()); - TEST_EQUAL(result.get().m_type, NotificationCandidate::Type::UgcReview, ()); + TEST_EQUAL(result.get().GetType(), NotificationCandidate::Type::UgcReview, ()); result = notificationManager.GetNotification(); TEST(!result.is_initialized(), ()); diff --git a/map/notifications/notification_manager.cpp b/map/notifications/notification_manager.cpp index ce3ad195da..c0d1c6a9b9 100644 --- a/map/notifications/notification_manager.cpp +++ b/map/notifications/notification_manager.cpp @@ -104,9 +104,10 @@ bool CheckPlannedTripTrigger(eye::MapObject const & poi) namespace notifications { -NotificationManager::NotificationManager(NotificationManager::Delegate & delegate) - : m_delegate(delegate) + +void NotificationManager::SetDelegate(std::unique_ptr delegate) { + m_delegate = std::move(delegate); } void NotificationManager::Load() @@ -136,10 +137,10 @@ void NotificationManager::TrimExpired() candidates.erase(std::remove_if(candidates.begin(), candidates.end(), [](auto const & item) { - if (item.m_used.time_since_epoch().count() != 0) - return Clock::now() - item.m_used >= eye::Eye::GetMapObjectEventsExpirePeriod(); + if (item.IsUsed()) + return Clock::now() - item.GetLastUsedTime() >= eye::Eye::GetMapObjectEventsExpirePeriod(); - return Clock::now() - item.m_created >= kCandidatesExpirePeriod; + return Clock::now() - item.GetCreatedTime() >= kCandidatesExpirePeriod; }), candidates.end()); if (sizeBefore != candidates.size()) @@ -161,7 +162,7 @@ boost::optional NotificationManager::GetNotification() if (it == candidates.end()) return {}; - it->m_used = Clock::now(); + it->MarkAsUsed(); m_queue.m_lastNotificationProvidedTime = Clock::now(); VERIFY(Save(), ()); @@ -176,15 +177,13 @@ size_t NotificationManager::GetCandidatesCount() const void NotificationManager::OnMapObjectEvent(eye::MapObject const & poi) { - CHECK(m_delegate.GetUGCApi(), ()); CHECK_GREATER(poi.GetEvents().size(), 0, ()); - auto const bestType = classif().GetTypeByReadableObjectName(poi.GetBestType()); - if (poi.GetEvents().back().m_type == eye::MapObject::Event::Type::UgcSaved) return ProcessUgcRateCandidates(poi); - m_delegate.GetUGCApi()->HasUGCForPlace(bestType, poi.GetPos(), [this, poi] (bool result) + auto const bestType = classif().GetTypeByReadableObjectName(poi.GetBestType()); + m_delegate->GetUGCApi().HasUGCForPlace(bestType, poi.GetPos(), [this, poi] (bool result) { if (!result) ProcessUgcRateCandidates(poi); @@ -202,16 +201,16 @@ void NotificationManager::ProcessUgcRateCandidates(eye::MapObject const & poi) { CHECK_GREATER(poi.GetEvents().size(), 0, ()); + if (poi.IsEmpty()) + return; + auto it = m_queue.m_candidates.begin(); for (; it != m_queue.m_candidates.end(); ++it) { - if (it->m_type != NotificationCandidate::Type::UgcReview || !it->m_mapObject || - it->m_used.time_since_epoch().count() != 0) - { + if (it->GetType() != NotificationCandidate::Type::UgcReview || it->IsUsed()) continue; - } - if (it->m_mapObject->AlmostEquals(poi)) + if (it->IsSameMapObject(poi)) { if (poi.GetEvents().back().m_type == eye::MapObject::Event::Type::UgcSaved) { @@ -229,12 +228,7 @@ void NotificationManager::ProcessUgcRateCandidates(eye::MapObject const & poi) if (CheckUgcNotSavedTrigger(poi) || CheckRouteToInSameGeoTrigger(poi) || CheckPlannedTripTrigger(poi)) { - NotificationCandidate candidate; - candidate.m_type = NotificationCandidate::Type::UgcReview; - candidate.m_created = Clock::now(); - candidate.m_mapObject = std::make_shared(poi); - candidate.m_mapObject->GetEditableEvents().clear(); - m_queue.m_candidates.emplace_back(std::move(candidate)); + m_queue.m_candidates.emplace_back(poi, m_delegate->GetAddress(poi.GetPos())); VERIFY(Save(), ()); } @@ -245,9 +239,8 @@ Candidates::iterator NotificationManager::GetUgcRateCandidate() auto it = m_queue.m_candidates.begin(); for (; it != m_queue.m_candidates.end(); ++it) { - if (it->m_used.time_since_epoch().count() == 0 && - it->m_type == NotificationCandidate::Type::UgcReview && - Clock::now() - it->m_created >= kMinTimeSinceLastEventForUgcRate) + if (!it->IsUsed() && it->GetType() == NotificationCandidate::Type::UgcReview && + Clock::now() - it->GetCreatedTime() >= kMinTimeSinceLastEventForUgcRate) { return it; } diff --git a/map/notifications/notification_manager.hpp b/map/notifications/notification_manager.hpp index abed317b9a..bad68ea5f0 100644 --- a/map/notifications/notification_manager.hpp +++ b/map/notifications/notification_manager.hpp @@ -7,9 +7,11 @@ #include "metrics/eye.hpp" #include +#include #include #include +#include namespace notifications { @@ -22,10 +24,11 @@ public: { public: virtual ~Delegate() = default; - virtual ugc::Api * GetUGCApi() = 0; + virtual ugc::Api & GetUGCApi() = 0; + virtual std::string GetAddress(m2::PointD const & pt) = 0; }; - explicit NotificationManager(Delegate & delegate); + void SetDelegate(std::unique_ptr delegate); void Load(); void TrimExpired(); @@ -41,17 +44,18 @@ private: void ProcessUgcRateCandidates(eye::MapObject const & poi); Candidates::iterator GetUgcRateCandidate(); - Delegate & m_delegate; + std::unique_ptr m_delegate; // Notification candidates queue. Queue m_queue; }; } // namespace notifications + namespace lightweight { class NotificationManager { public: - NotificationManager() : m_manager(m_delegate) { m_manager.Load(); } + NotificationManager() { m_manager.Load(); } boost::optional GetNotification() { @@ -65,17 +69,6 @@ public: } private: - class EmptyDelegate : public notifications::NotificationManager::Delegate - { - public: - // NotificationManager::Delegate overrides: - ugc::Api * GetUGCApi() override - { - return nullptr; - } - }; - - EmptyDelegate m_delegate; notifications::NotificationManager m_manager; }; } // namespace lightweight diff --git a/map/notifications/notification_manager_delegate.cpp b/map/notifications/notification_manager_delegate.cpp new file mode 100644 index 0000000000..8af7de69ee --- /dev/null +++ b/map/notifications/notification_manager_delegate.cpp @@ -0,0 +1,43 @@ +#include "notification_manager_delegate.hpp" + +#include "ugc/api.hpp" + +#include "map/utils.hpp" + +#include "search/city_finder.hpp" + +#include "indexer/feature_decl.hpp" + +#include "platform/preferred_languages.hpp" + +#include "coding/string_utf8_multilang.hpp" + +namespace notifications +{ +NotificationManagerDelegate::NotificationManagerDelegate(DataSource const & dataSource, + search::CityFinder & cityFinder, + ugc::Api & ugcApi) + : m_dataSource(dataSource), m_cityFinder(cityFinder), m_ugcApi(ugcApi) +{ +} + +ugc::Api & NotificationManagerDelegate::GetUGCApi() +{ + return m_ugcApi; +} + +string NotificationManagerDelegate::GetAddress(m2::PointD const & pt) +{ + auto const address = utils::GetAddressAtPoint(m_dataSource, pt).FormatAddress(); + auto const langIndex = StringUtf8Multilang::GetLangIndex(languages::GetCurrentNorm()); + auto const city = m_cityFinder.GetCityName(pt, langIndex); + + if (address.empty()) + return city; + + if (city.empty()) + return address; + + return city + " ," + address; +} +} // namespace notifications diff --git a/map/notifications/notification_manager_delegate.hpp b/map/notifications/notification_manager_delegate.hpp new file mode 100644 index 0000000000..684a50d6ff --- /dev/null +++ b/map/notifications/notification_manager_delegate.hpp @@ -0,0 +1,36 @@ +#pragma once + +#include "map/notifications/notification_manager.hpp" + +#include "geometry/point2d.hpp" + +class DataSource; + +namespace search +{ +class CityFinder; +} + +namespace ugc +{ +class Api; +} + +namespace notifications +{ +class NotificationManagerDelegate : public NotificationManager::Delegate +{ +public: + NotificationManagerDelegate(DataSource const & dataSource, search::CityFinder & cityFinder, + ugc::Api & ugcApi); + + // NotificationManager::Delegate overrides: + ugc::Api & GetUGCApi() override; + string GetAddress(m2::PointD const & pt) override; + +private: + DataSource const & m_dataSource; + search::CityFinder & m_cityFinder; + ugc::Api & m_ugcApi; +}; +} diff --git a/map/notifications/notification_queue.cpp b/map/notifications/notification_queue.cpp new file mode 100644 index 0000000000..9d8a88576f --- /dev/null +++ b/map/notifications/notification_queue.cpp @@ -0,0 +1,87 @@ +#include "map/notifications/notification_queue.hpp" + +#include "base/assert.hpp" + +namespace notifications +{ +NotificationCandidate::NotificationCandidate(eye::MapObject const & poi, + std::string const & address) +: m_type(NotificationCandidate::Type::UgcReview) +, m_created(Clock::now()) +, m_mapObject(std::make_shared(poi)) +, m_address(address) +{ + CHECK(!poi.IsEmpty(), ()); + + m_mapObject->GetEditableEvents().clear(); +} + +NotificationCandidate::Type NotificationCandidate::GetType() const +{ + return m_type; +} + +Time NotificationCandidate::GetCreatedTime() const +{ + return m_created; +} + +Time NotificationCandidate::GetLastUsedTime() const +{ + return m_used; +} + +bool NotificationCandidate::IsUsed() const +{ + return m_used.time_since_epoch().count() != 0; +} + +void NotificationCandidate::MarkAsUsed() +{ + CHECK_EQUAL(m_used.time_since_epoch().count(), 0, ()); + + m_used = Clock::now(); +} + +bool NotificationCandidate::IsSameMapObject(eye::MapObject const & rhs) const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_mapObject->AlmostEquals(rhs); +} + +std::string const & NotificationCandidate::GetBestFeatureType() const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_mapObject->GetBestType(); +} + +m2::PointD const & NotificationCandidate::GetPos() const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_mapObject->GetPos(); +} + +std::string const & NotificationCandidate::GetDefaultName() const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_mapObject->GetDefaultName(); +} + +std::string const & NotificationCandidate::GetReadableName() const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_mapObject->GetReadableName(); +} + +std::string const & NotificationCandidate::GetAddress() const +{ + CHECK_EQUAL(m_type, NotificationCandidate::Type::UgcReview, ()); + + return m_address; +} +} // namespace notifications diff --git a/map/notifications/notification_queue.hpp b/map/notifications/notification_queue.hpp index cb4e9e170b..6165e8edc2 100644 --- a/map/notifications/notification_queue.hpp +++ b/map/notifications/notification_queue.hpp @@ -11,8 +11,11 @@ namespace notifications using Clock = std::chrono::system_clock; using Time = Clock::time_point; -struct NotificationCandidate +class NotificationCandidate { +public: + friend class NotificationManagerForTesting; + enum class Type : uint8_t { UgcAuth = 0, @@ -20,12 +23,35 @@ struct NotificationCandidate }; DECLARE_VISITOR(visitor(m_type, "type"), visitor(m_created, "created_time"), - visitor(m_used, "used"), visitor(m_mapObject, "object")); + visitor(m_used, "used"), visitor(m_mapObject, "object"), + visitor(m_address, std::string(""), "address")); + NotificationCandidate() = default; + // Constructs candidate with type Type::UgcReview. + NotificationCandidate(eye::MapObject const & poi, std::string const & address); + + Type GetType() const; + Time GetCreatedTime() const; + Time GetLastUsedTime() const; + bool IsUsed() const; + void MarkAsUsed(); + + // Methods for Type::UgcReview type. + // It is possible to use inheritance, but our serialization/deserialization + // mechanism is not support it. + bool IsSameMapObject(eye::MapObject const & rhs) const; + std::string const & GetBestFeatureType() const; + m2::PointD const & GetPos() const; + std::string const & GetDefaultName() const; + std::string const & GetReadableName() const; + std::string const & GetAddress() const; + +private: Type m_type; Time m_created; Time m_used; std::shared_ptr m_mapObject; + std::string m_address; }; using Candidates = std::deque; diff --git a/map/utils.cpp b/map/utils.cpp index bdd9152b53..010ae2d087 100644 --- a/map/utils.cpp +++ b/map/utils.cpp @@ -3,6 +3,7 @@ #include "map/place_page_info.hpp" #include "indexer/feature.hpp" +#include "indexer/feature_decl.hpp" #include "indexer/feature_algo.hpp" namespace utils @@ -49,4 +50,21 @@ eye::MapObject MakeEyeMapObject(FeatureType & ft) return mapObject; } + +search::ReverseGeocoder::Address GetAddressAtPoint(DataSource const & dataSource, + m2::PointD const & pt) +{ + double const kDistanceThresholdMeters = 0.5; + + search::ReverseGeocoder const coder(dataSource); + search::ReverseGeocoder::Address address; + coder.GetNearbyAddress(pt, address); + + // We do not init nearby address info for points that are located + // outside of the nearby building. + if (address.GetDistance() < kDistanceThresholdMeters) + return address; + + return {}; +} } // namespace utils diff --git a/map/utils.hpp b/map/utils.hpp index b3d4c8589b..bb0f08ba4e 100644 --- a/map/utils.hpp +++ b/map/utils.hpp @@ -1,16 +1,24 @@ #pragma once +#include "search/reverse_geocoder.hpp" + #include "metrics/eye_info.hpp" +#include "geometry/point2d.hpp" + namespace place_page { class Info; } +class DataSource; class FeatureType; namespace utils { eye::MapObject MakeEyeMapObject(place_page::Info const & info); eye::MapObject MakeEyeMapObject(FeatureType & ft); + +search::ReverseGeocoder::Address GetAddressAtPoint(DataSource const & dataSource, + m2::PointD const & pt); } // namespace utils