diff --git a/android/jni/com/mapswithme/maps/metrics/UserActionsLogger.cpp b/android/jni/com/mapswithme/maps/metrics/UserActionsLogger.cpp index 37a45359bb..bfce30285e 100644 --- a/android/jni/com/mapswithme/maps/metrics/UserActionsLogger.cpp +++ b/android/jni/com/mapswithme/maps/metrics/UserActionsLogger.cpp @@ -8,12 +8,12 @@ extern "C" { JNIEXPORT void JNICALL -Java_com_mapswithme_maps_metrics_UserActionsLogger_nativeTipShown(JNIEnv * env, jclass, jint type, - jint event) +Java_com_mapswithme_maps_metrics_UserActionsLogger_nativeTipClicked(JNIEnv * env, jclass, jint type, + jint event) { auto const & typeValue = static_cast(type); auto const & eventValue = static_cast(event); - eye::Eye::Event::TipShown(typeValue, eventValue); + eye::Eye::Event::TipClicked(typeValue, eventValue); } JNIEXPORT void JNICALL diff --git a/android/src/com/mapswithme/maps/metrics/UserActionsLogger.java b/android/src/com/mapswithme/maps/metrics/UserActionsLogger.java index 963bb66357..1777a85283 100644 --- a/android/src/com/mapswithme/maps/metrics/UserActionsLogger.java +++ b/android/src/com/mapswithme/maps/metrics/UserActionsLogger.java @@ -9,9 +9,9 @@ import com.mapswithme.maps.tips.TipsApi; public class UserActionsLogger { - public static void logTipShownEvent(@NonNull TipsApi provider, @NonNull TipsAction action) + public static void logTipClickedEvent(@NonNull TipsApi provider, @NonNull TipsAction action) { - nativeTipShown(provider.ordinal(), action.ordinal()); + nativeTipClicked(provider.ordinal(), action.ordinal()); } public static void logBookingFilterUsedEvent() @@ -39,7 +39,7 @@ public class UserActionsLogger nativeDiscoveryItemClicked(event.ordinal()); } - private static native void nativeTipShown(int type, int event); + private static native void nativeTipClicked(int type, int event); private static native void nativeBookingFilterUsed(); private static native void nativeBookmarksCatalogShown(); private static native void nativeDiscoveryShown(); diff --git a/android/src/com/mapswithme/maps/tips/AbstractClickInterceptor.java b/android/src/com/mapswithme/maps/tips/AbstractClickInterceptor.java index e32e60b90d..59f54a0837 100644 --- a/android/src/com/mapswithme/maps/tips/AbstractClickInterceptor.java +++ b/android/src/com/mapswithme/maps/tips/AbstractClickInterceptor.java @@ -24,7 +24,7 @@ public abstract class AbstractClickInterceptor implements ClickInterceptor @Override public final void onInterceptClick(@NonNull MwmActivity activity) { - UserActionsLogger.logTipShownEvent(getType(), TipsAction.ACTION_CLICKED); + UserActionsLogger.logTipClickedEvent(getType(), TipsAction.ACTION_CLICKED); onInterceptClickInternal(activity); } diff --git a/android/src/com/mapswithme/maps/tips/TipsApi.java b/android/src/com/mapswithme/maps/tips/TipsApi.java index 87347f7bfc..017ab964c2 100644 --- a/android/src/com/mapswithme/maps/tips/TipsApi.java +++ b/android/src/com/mapswithme/maps/tips/TipsApi.java @@ -147,7 +147,7 @@ public enum TipsApi private void onPromptStateChanged(int state) { if (state == MaterialTapTargetPrompt.STATE_DISMISSED) - UserActionsLogger.logTipShownEvent(TipsApi.this, TipsAction.GOT_IT_CLICKED); + UserActionsLogger.logTipClickedEvent(TipsApi.this, TipsAction.GOT_IT_CLICKED); } @Nullable diff --git a/iphone/Maps/Classes/CustomViews/MapViewControls/MWMMapViewControlsManager.mm b/iphone/Maps/Classes/CustomViews/MapViewControls/MWMMapViewControlsManager.mm index 74f4f54b01..12c0e5af4f 100644 --- a/iphone/Maps/Classes/CustomViews/MapViewControls/MWMMapViewControlsManager.mm +++ b/iphone/Maps/Classes/CustomViews/MapViewControls/MWMMapViewControlsManager.mm @@ -441,7 +441,7 @@ extern NSString * const kAlohalyticsTapEventKey; - (void)didPressCancel:(MWMTutorialViewController *)viewController { - [MWMEye tipShownWithType:self.tutorialType event:MWMTipEventGotIt]; + [MWMEye tipClickedWithType:self.tutorialType event:MWMTipEventGotIt]; [viewController fadeOutWithCompletion:^{ [viewController willMoveToParentViewController:nil]; [viewController.view removeFromSuperview]; @@ -451,7 +451,7 @@ extern NSString * const kAlohalyticsTapEventKey; - (void)didPressTarget:(MWMTutorialViewController *)viewController { - [MWMEye tipShownWithType:self.tutorialType event:MWMTipEventAction]; + [MWMEye tipClickedWithType:self.tutorialType event:MWMTipEventAction]; [viewController fadeOutWithCompletion:^{ [viewController willMoveToParentViewController:nil]; [viewController.view removeFromSuperview]; diff --git a/iphone/Maps/Core/Metrics/MWMEye.h b/iphone/Maps/Core/Metrics/MWMEye.h index d0b1241bb1..e315fd450f 100644 --- a/iphone/Maps/Core/Metrics/MWMEye.h +++ b/iphone/Maps/Core/Metrics/MWMEye.h @@ -28,7 +28,7 @@ typedef NS_ENUM(NSUInteger, MWMEyeDiscoveryEvent) @interface MWMEye : NSObject + (MWMTip)getTipType; -+ (void)tipShownWithType:(MWMTip)type event:(MWMTipEvent)event; ++ (void)tipClickedWithType:(MWMTip)type event:(MWMTipEvent)event; + (void)bookingFilterUsed; + (void)boomarksCatalogShown; + (void)discoveryShown; diff --git a/iphone/Maps/Core/Metrics/MWMEye.mm b/iphone/Maps/Core/Metrics/MWMEye.mm index fa3776496c..3102dd94d4 100644 --- a/iphone/Maps/Core/Metrics/MWMEye.mm +++ b/iphone/Maps/Core/Metrics/MWMEye.mm @@ -11,9 +11,9 @@ return tutorialType ? (MWMTip)*tutorialType : MWMTipNone; } -+ (void)tipShownWithType:(MWMTip)type event:(MWMTipEvent)event ++ (void)tipClickedWithType:(MWMTip)type event:(MWMTipEvent)event { - eye::Eye::Event::TipShown((eye::Tip::Type)type, (eye::Tip::Event)event); + eye::Eye::Event::TipClicked((eye::Tip::Type)type, (eye::Tip::Event)event); } + (void)bookingFilterUsed diff --git a/map/transit/transit_reader.cpp b/map/transit/transit_reader.cpp index 788b612bdc..fbd04ffd4b 100644 --- a/map/transit/transit_reader.cpp +++ b/map/transit/transit_reader.cpp @@ -300,7 +300,7 @@ void TransitReadManager::UpdateViewport(ScreenBase const & screen) if (hasData && m_trackFirstSchemeData) { - eye::Eye::Event::LayerUsed(eye::Layer::Type::PublicTransport); + eye::Eye::Event::LayerShown(eye::Layer::Type::PublicTransport); m_trackFirstSchemeData = false; } diff --git a/metrics/eye.cpp b/metrics/eye.cpp index b32aecb82a..fe707f38ca 100644 --- a/metrics/eye.cpp +++ b/metrics/eye.cpp @@ -7,6 +7,7 @@ #include "coding/file_writer.hpp" #include "base/logging.hpp" +#include "base/assert.hpp" #include #include @@ -19,8 +20,11 @@ namespace { void Load(Info & info) { - std::vector fileData; - if (!Storage::Load(Storage::GetEyeFilePath(), fileData)) + Storage::Migrate(); + + std::vector infoFileData; + std::vector mapObjectsFileData; + if (!Storage::LoadInfo(infoFileData) || !Storage::LoadMapObjects(mapObjectsFileData)) { info = {}; return; @@ -28,20 +32,39 @@ void Load(Info & info) try { - Serdes::Deserialize(fileData, info); + Serdes::DeserializeInfo(infoFileData, info); + Serdes::DeserializeMapObjects(mapObjectsFileData, info.m_mapObjects); } catch (Serdes::UnknownVersion const & ex) { - LOG(LERROR, ("Unknown eye file version, eye will be disabled. Exception:", ex.Msg())); + LOG(LERROR, ("Cannot load metrics files, eye will be disabled. Exception:", ex.Msg())); + info = {}; } } bool Save(Info const & info) { std::vector fileData; - Serdes::Serialize(info, fileData); - return Storage::Save(Storage::GetEyeFilePath(), fileData); + Serdes::SerializeInfo(info, fileData); + return Storage::SaveInfo(fileData); } + +// TODO: use to trim old map object events. +//bool SaveMapObjects(Info const & info) +//{ +// std::vector fileData; +// Serdes::SerializeMapObjects(info.m_mapObjects, fileData); +// return Storage::SaveMapObjects(fileData); +//} +// +// TODO: use it to save map object events with append only flag. +//bool SaveMapObjectEvent(MapObject const & mapObject, MapObject::Event const & event) +//{ +// std::vector eventData; +// Serdes::SerializeMapObjectEvent(mapObject, event, eventData); +// +// return Storage::AppendMapObjectEvent(eventData); +//} } // namespace namespace eye @@ -65,13 +88,21 @@ Eye::InfoType Eye::GetInfo() const return m_info.Get(); } +void Eye::Subscribe(Subscriber * subscriber) +{ + GetPlatform().RunTask(Platform::Thread::File, [this, subscriber] + { + m_subscribers.push_back(subscriber); + }); +} + void Eye::Save(InfoType const & info) { if (::Save(*info)) m_info.Set(info); } -void Eye::AppendTip(Tip::Type type, Tip::Event event) +void Eye::RegisterTipClick(Tip::Type type, Tip::Event event) { auto const info = m_info.Get(); auto editableInfo = std::make_shared(*info); @@ -82,52 +113,76 @@ void Eye::AppendTip(Tip::Type type, Tip::Event event) return tip.m_type == type; }); + Tip tip; auto const now = Clock::now(); if (it != editableTips.cend()) { it->m_eventCounters.Increment(event); it->m_lastShownTime = now; + tip = *it; } else { - Tip tip; tip.m_type = type; tip.m_eventCounters.Increment(event); tip.m_lastShownTime = now; - editableTips.emplace_back(std::move(tip)); + editableTips.emplace_back(tip); } Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnTipClicked(tip); + } } void Eye::UpdateBookingFilterUsedTime() { auto const info = m_info.Get(); auto editableInfo = std::make_shared(*info); + auto const now = Clock::now(); - editableInfo->m_booking.m_lastFilterUsedTime = Clock::now(); + editableInfo->m_booking.m_lastFilterUsedTime = now; Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnBookingFilterUsed(now); + } } void Eye::UpdateBoomarksCatalogShownTime() { auto const info = m_info.Get(); auto editableInfo = std::make_shared(*info); + auto const now = Clock::now(); - editableInfo->m_bookmarks.m_lastOpenedTime = Clock::now(); + editableInfo->m_bookmarks.m_lastOpenedTime = now; Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnBookmarksCatalogShown(now); + } } void Eye::UpdateDiscoveryShownTime() { auto const info = m_info.Get(); auto editableInfo = std::make_shared(*info); + auto const now = Clock::now(); - editableInfo->m_discovery.m_lastOpenedTime = Clock::now(); + editableInfo->m_discovery.m_lastOpenedTime = now; Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnDiscoveryShown(now); + } } void Eye::IncrementDiscoveryItem(Discovery::Event event) @@ -139,9 +194,14 @@ void Eye::IncrementDiscoveryItem(Discovery::Event event) editableInfo->m_discovery.m_eventCounters.Increment(event); Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnDiscoveryItemClicked(event); + } } -void Eye::AppendLayer(Layer::Type type) +void Eye::RegisterLayerShown(Layer::Type type) { auto const info = m_info.Get(); auto editableInfo = std::make_shared(*info); @@ -152,31 +212,62 @@ void Eye::AppendLayer(Layer::Type type) return layer.m_type == type; }); + Layer layer; if (it != editableLayers.end()) { ++it->m_useCount; it->m_lastTimeUsed = Clock::now(); + layer = *it; } else { - Layer layer; layer.m_type = type; ++layer.m_useCount; layer.m_lastTimeUsed = Clock::now(); - editableLayers.emplace_back(std::move(layer)); + editableLayers.emplace_back(layer); } Save(editableInfo); + + for (auto subscriber : m_subscribers) + { + subscriber->OnLayerUsed(layer); + } +} + +void Eye::RegisterPlacePageOpened() +{ + +} + +void Eye::RegisterUgcEditorOpened() +{ + +} + +void Eye::RegisterUgcSaved() +{ + +} + +void Eye::RegisterAddToBookmarkClicked() +{ + +} + +void Eye::RegisterRouteCreatedToObject() +{ + } // Eye::Event methods ------------------------------------------------------------------------------ // static -void Eye::Event::TipShown(Tip::Type type, Tip::Event event) +void Eye::Event::TipClicked(Tip::Type type, Tip::Event event) { GetPlatform().RunTask(Platform::Thread::File, [type, event] { - Instance().AppendTip(type, event); + Instance().RegisterTipClick(type, event); }); } @@ -217,11 +308,56 @@ void Eye::Event::DiscoveryItemClicked(Discovery::Event event) } // static -void Eye::Event::LayerUsed(Layer::Type type) +void Eye::Event::LayerShown(Layer::Type type) { GetPlatform().RunTask(Platform::Thread::File, [type] { - Instance().AppendLayer(type); + Instance().RegisterLayerShown(type); + }); +} + +// static +void Eye::Event::PlacePageOpened() +{ + GetPlatform().RunTask(Platform::Thread::File, [] + { + Instance().RegisterPlacePageOpened(); + }); +} + +// static +void Eye::Event::UgcEditorOpened() +{ + GetPlatform().RunTask(Platform::Thread::File, [] + { + Instance().RegisterUgcEditorOpened(); + }); +} + +//static +void Eye::Event::UgcSaved() +{ + GetPlatform().RunTask(Platform::Thread::File, [] + { + Instance().RegisterUgcSaved(); + }); +} + +// static +void Eye::Event::AddToBookmarkClicked() +{ + GetPlatform().RunTask(Platform::Thread::File, [] + { + Instance().RegisterAddToBookmarkClicked(); + }); +} + +// static +void Eye::Event::RouteCreatedToObject() +{ + GetPlatform().RunTask(Platform::Thread::File, [] + { + Instance().RegisterRouteCreatedToObject(); }); } } // namespace eye diff --git a/metrics/eye.hpp b/metrics/eye.hpp index 2268cf616d..eb2e86d3a1 100644 --- a/metrics/eye.hpp +++ b/metrics/eye.hpp @@ -5,8 +5,25 @@ #include "base/atomic_shared_ptr.hpp" #include "base/macros.hpp" +#include + namespace eye { +class Subscriber +{ +public: + virtual ~Subscriber() {}; + +public: + virtual void OnTipClicked(Tip const & tip) {}; + virtual void OnBookingFilterUsed(Time const & time) {}; + virtual void OnBookmarksCatalogShown(Time const & time) {}; + virtual void OnDiscoveryShown(Time const & time) {}; + virtual void OnDiscoveryItemClicked(Discovery::Event event) {}; + virtual void OnLayerUsed(Layer const & layer) {}; + virtual void OnPlacePageOpened(MapObject const & poi) {}; +}; + // Note This class IS thread-safe. // All write operations are asynchronous and work on Platform::Thread::File thread. // Read operations are synchronous and return shared pointer with constant copy of internal @@ -20,17 +37,23 @@ public: class Event { public: - static void TipShown(Tip::Type type, Tip::Event event); + static void TipClicked(Tip::Type type, Tip::Event event); static void BookingFilterUsed(); static void BoomarksCatalogShown(); static void DiscoveryShown(); static void DiscoveryItemClicked(Discovery::Event event); - static void LayerUsed(Layer::Type type); + static void LayerShown(Layer::Type type); + static void PlacePageOpened(); + static void UgcEditorOpened(); + static void UgcSaved(); + static void AddToBookmarkClicked(); + static void RouteCreatedToObject(); }; static Eye & Instance(); InfoType GetInfo() const; + void Subscribe(Subscriber * subscriber); private: Eye(); @@ -38,14 +61,20 @@ private: void Save(InfoType const & info); // Event processing: - void AppendTip(Tip::Type type, Tip::Event event); + void RegisterTipClick(Tip::Type type, Tip::Event event); void UpdateBookingFilterUsedTime(); void UpdateBoomarksCatalogShownTime(); void UpdateDiscoveryShownTime(); void IncrementDiscoveryItem(Discovery::Event event); - void AppendLayer(Layer::Type type); + void RegisterLayerShown(Layer::Type type); + void RegisterPlacePageOpened(); + void RegisterUgcEditorOpened(); + void RegisterUgcSaved(); + void RegisterAddToBookmarkClicked(); + void RegisterRouteCreatedToObject(); base::AtomicSharedPtr m_info; + std::vector m_subscribers; DISALLOW_COPY_AND_MOVE(Eye); };