From b6859f9c6e98179d40f353f5d4e6dd2b6ff73b80 Mon Sep 17 00:00:00 2001 From: ExMix Date: Tue, 2 Jun 2015 15:48:01 +0300 Subject: [PATCH] [drape] select object on map --- drape/overlay_handle.cpp | 19 ++ drape/overlay_handle.hpp | 2 + drape/overlay_tree.cpp | 46 ++- drape/overlay_tree.hpp | 11 +- drape_frontend/drape_engine.cpp | 42 +++ drape_frontend/drape_engine.hpp | 11 + drape_frontend/frontend_renderer.cpp | 72 ++++- drape_frontend/frontend_renderer.hpp | 24 +- drape_frontend/message.hpp | 3 +- drape_frontend/message_subclasses.hpp | 65 ++++ drape_frontend/my_position_controller.cpp | 3 + drape_frontend/my_position_controller.hpp | 1 + drape_frontend/visual_params.cpp | 5 + drape_frontend/visual_params.hpp | 3 + iphone/Maps/Classes/MapViewController.mm | 12 - .../Maps/Settings/SettingsViewController.mm | 2 - iphone/Maps/resources-default/default.ui | 50 --- map/address_finder.cpp | 2 +- map/framework.cpp | 290 +++++++----------- map/framework.hpp | 25 +- map/pin_click_manager.cpp | 11 +- map/user_mark.cpp | 40 ++- map/user_mark.hpp | 6 +- qt/draw_widget.cpp | 16 +- qt/draw_widget.hpp | 3 - 25 files changed, 445 insertions(+), 319 deletions(-) delete mode 100644 iphone/Maps/resources-default/default.ui diff --git a/drape/overlay_handle.cpp b/drape/overlay_handle.cpp index 2933817995..955407f644 100644 --- a/drape/overlay_handle.cpp +++ b/drape/overlay_handle.cpp @@ -40,6 +40,25 @@ void OverlayHandle::SetIsVisible(bool isVisible) m_isVisible = isVisible; } +m2::PointD OverlayHandle::GetPivot(ScreenBase const & screen) const +{ + m2::RectD r = GetPixelRect(screen); + m2::PointD size(0.5 * r.SizeX(), 0.5 * r.SizeY()); + m2::PointD result = r.Center(); + + if (m_anchor & dp::Left) + result.x += size.x; + else if (m_anchor & dp::Right) + result.x -= size.x; + + if (m_anchor & dp::Top) + result.y += size.y; + else if (m_anchor & dp::Bottom) + result.y -= size.y; + + return result; +} + bool OverlayHandle::IsIntersect(ScreenBase const & screen, ref_ptr const h) const { Rects ar1; diff --git a/drape/overlay_handle.hpp b/drape/overlay_handle.hpp index 2ee3bec2f7..855f393792 100644 --- a/drape/overlay_handle.hpp +++ b/drape/overlay_handle.hpp @@ -32,6 +32,8 @@ public: bool IsValid() const { return m_isValid; } + m2::PointD GetPivot(ScreenBase const & screen) const; + virtual void Update(ScreenBase const & /*screen*/) {} virtual m2::RectD GetPixelRect(ScreenBase const & screen) const = 0; diff --git a/drape/overlay_tree.cpp b/drape/overlay_tree.cpp index 24e6065da7..2b75ade6a8 100644 --- a/drape/overlay_tree.cpp +++ b/drape/overlay_tree.cpp @@ -5,11 +5,32 @@ namespace dp { +int const FRAME_UPDATE_PERIOD = 10; + +void OverlayTree::Frame() +{ + m_frameCounter++; + if (m_frameCounter >= FRAME_UPDATE_PERIOD) + m_frameCounter = -1; +} + +bool OverlayTree::IsNeedUpdate() const +{ + return m_frameCounter == -1; +} + +void OverlayTree::ForceUpdate() +{ + Clear(); + m_frameCounter = -1; +} + void OverlayTree::StartOverlayPlacing(ScreenBase const & screen, bool canOverlap) { + ASSERT(IsNeedUpdate(), ()); + Clear(); m_traits.m_modelView = screen; m_canOverlap = canOverlap; - ASSERT(IsEmpty(), ()); } void OverlayTree::Add(ref_ptr handle, bool isTransparent) @@ -64,8 +85,29 @@ void OverlayTree::EndOverlayPlacing() { info.m_handle->SetIsVisible(true); }); +} - Clear(); +void OverlayTree::Select(m2::RectD const & rect, TSelectResult & result) const +{ + ScreenBase screen = m_traits.m_modelView; + ForEachInRect(rect, [&](detail::OverlayInfo const & info) + { + if (info.m_handle->IsValid() && + info.m_handle->IsVisible() && + info.m_handle->GetFeatureID().IsValid()) + { + OverlayHandle::Rects shape; + info.m_handle->GetPixelShape(screen, shape); + for (m2::RectF const & rShape : shape) + { + if (rShape.IsIntersect(m2::RectF(rect))) + { + result.push_back(info.m_handle); + break; + } + } + } + }); } } // namespace dp diff --git a/drape/overlay_tree.hpp b/drape/overlay_tree.hpp index 6c0282e3e6..78769a17fd 100644 --- a/drape/overlay_tree.hpp +++ b/drape/overlay_tree.hpp @@ -5,6 +5,7 @@ #include "geometry/screenbase.hpp" #include "geometry/tree4d.hpp" +#include "base/buffer_vector.hpp" namespace dp { @@ -46,15 +47,23 @@ class OverlayTree : public m4::Tree typedef m4::Tree BaseT; public: + void Frame(); + bool IsNeedUpdate() const; + void ForceUpdate(); + void StartOverlayPlacing(ScreenBase const & screen, bool canOverlap = false); void Add(ref_ptr handle, bool isTransparent); void EndOverlayPlacing(); + using TSelectResult = buffer_vector, 8>; + void Select(m2::RectD const & rect, TSelectResult & result) const; + private: ScreenBase const & GetModelView() const { return m_traits.m_modelView; } private: - bool m_canOverlap; + bool m_canOverlap = false; + int m_frameCounter = -1; }; } // namespace dp diff --git a/drape_frontend/drape_engine.cpp b/drape_frontend/drape_engine.cpp index f17dde014a..6a7b3e61d3 100644 --- a/drape_frontend/drape_engine.cpp +++ b/drape_frontend/drape_engine.cpp @@ -67,6 +67,8 @@ DrapeEngine::DrapeEngine(Params const & params) make_ref(m_textureManager), m_viewport, bind(&DrapeEngine::ModelViewChanged, this, _1), params.m_model.GetIsCountryLoadedFn(), + bind(&DrapeEngine::TapEvent, this, _1, _2, _3, _4), + bind(&DrapeEngine::UserPositionChanged, this, _1), bind(&DrapeEngine::MyPositionModeChanged, this, _1), static_cast(modeValue)); @@ -187,6 +189,24 @@ void DrapeEngine::MyPositionModeChanged(location::EMyPositionMode mode) }); } +void DrapeEngine::TapEvent(m2::PointD const & pxPoint, bool isLong, bool isMyPosition, FeatureID const & feature) +{ + GetPlatform().RunOnGuiThread([=]() + { + if (m_tapListener) + m_tapListener(pxPoint, isLong, isMyPosition, feature); + }); +} + +void DrapeEngine::UserPositionChanged(m2::PointD const & position) +{ + GetPlatform().RunOnGuiThread([this, position]() + { + if (m_userPositionChangedFn) + m_userPositionChangedFn(position); + }); +} + void DrapeEngine::SetCountryInfo(gui::CountryInfo const & info, bool isCurrentCountry, bool isCountryLoaded) { m_threadCommutator->PostMessage(ThreadsCommutator::ResourceUploadThread, @@ -241,4 +261,26 @@ void DrapeEngine::SetMyPositionModeListener(location::TMyPositionModeChanged con m_myPositionModeChanged = fn; } +void DrapeEngine::SetTapEventInfoListener(TTapEventInfoFn const & fn) +{ + m_tapListener = fn; +} + +void DrapeEngine::SetUserPositionListener(DrapeEngine::TUserPositionChangedFn const & fn) +{ + m_userPositionChangedFn = fn; +} + +FeatureID DrapeEngine::GetVisiblePOI(m2::PointD const & glbPoint) +{ + FeatureID result; + BaseBlockingMessage::Blocker blocker; + m_threadCommutator->PostMessage(ThreadsCommutator::RenderThread, + make_unique_dp(blocker, glbPoint, result), + MessagePriority::High); + + blocker.Wait(); + return result; +} + } // namespace df diff --git a/drape_frontend/drape_engine.hpp b/drape_frontend/drape_engine.hpp index 6776cd1c39..c3ea78ab19 100644 --- a/drape_frontend/drape_engine.hpp +++ b/drape_frontend/drape_engine.hpp @@ -83,12 +83,21 @@ public: void InvalidateMyPosition(); void SetMyPositionModeListener(location::TMyPositionModeChanged const & fn); + using TTapEventInfoFn = FrontendRenderer::TTapEventInfoFn; + void SetTapEventInfoListener(TTapEventInfoFn const & fn); + using TUserPositionChangedFn = FrontendRenderer::TUserPositionChangedFn; + void SetUserPositionListener(TUserPositionChangedFn const & fn); + + FeatureID GetVisiblePOI(m2::PointD const & glbPoint); + private: void AddUserEvent(UserEvent const & e); void ModelViewChanged(ScreenBase const & screen); void ModelViewChangedGuiThread(ScreenBase const & screen); void MyPositionModeChanged(location::EMyPositionMode mode); + void TapEvent(m2::PointD const & pxPoint, bool isLong, bool isMyPosition, FeatureID const & feature); + void UserPositionChanged(m2::PointD const & position); private: drape_ptr m_frontend; @@ -102,6 +111,8 @@ private: TListenerMap m_listeners; location::TMyPositionModeChanged m_myPositionModeChanged; + TTapEventInfoFn m_tapListener; + TUserPositionChangedFn m_userPositionChangedFn; }; } // namespace df diff --git a/drape_frontend/frontend_renderer.cpp b/drape_frontend/frontend_renderer.cpp index 9238d0dc13..03443c37ed 100755 --- a/drape_frontend/frontend_renderer.cpp +++ b/drape_frontend/frontend_renderer.cpp @@ -43,10 +43,11 @@ FrontendRenderer::FrontendRenderer(Params const & params) : BaseRenderer(ThreadsCommutator::RenderThread, params) , m_gpuProgramManager(new dp::GpuProgramManager()) , m_overlayTree(new dp::OverlayTree()) - , m_overlayTreeIsUpdating(true) , m_viewport(params.m_viewport) , m_userEventStream(params.m_isCountryLoadedFn) , m_modelViewChangedFn(params.m_modelViewChangedFn) + , m_tapEventInfoFn(params.m_tapEventFn) + , m_userPositionChangedFn(params.m_positionChangedFn) , m_tileTree(new TileTree()) { #ifdef DRAW_INFO @@ -230,6 +231,13 @@ void FrontendRenderer::AcceptMessage(ref_ptr message) break; } + case Message::FindVisiblePOI: + { + ref_ptr msg = message; + msg->SetFeatureID(GetVisiblePOI(m_userEventStream.GetCurrentScreen().GtoP(msg->GetPoint()))); + break; + } + default: ASSERT(false, ()); } @@ -292,6 +300,7 @@ void FrontendRenderer::OnActivateTile(TileKey const & tileKey) void FrontendRenderer::OnRemoveTile(TileKey const & tileKey) { + m_overlayTree->ForceUpdate(); for(auto const & group : m_renderGroups) { if (group->GetTileKey() == tileKey) @@ -317,25 +326,48 @@ void FrontendRenderer::OnCompassTapped() m_userEventStream.AddEvent(RotateEvent(0.0)); } -void FrontendRenderer::BeginUpdateOverlayTree(ScreenBase const & modelView) +FeatureID FrontendRenderer::GetVisiblePOI(m2::PointD const & pixelPoint) const { - static int framesCounter = 0; - int const updatePeriod = 10; + double halfSize = VisualParams::Instance().GetTouchRectRadius(); + m2::PointD sizePoint(halfSize, halfSize); + m2::RectD selectRect(pixelPoint - sizePoint, pixelPoint + sizePoint); + return GetVisiblePOI(selectRect); +} - framesCounter++; - if (framesCounter % updatePeriod == 0) +FeatureID FrontendRenderer::GetVisiblePOI(m2::RectD const & pixelRect) const +{ + m2::PointD pt = pixelRect.Center(); + dp::OverlayTree::TSelectResult selectResult; + m_overlayTree->Select(pixelRect, selectResult); + + double dist = numeric_limits::max(); + FeatureID featureID; + + ScreenBase const & screen = m_userEventStream.GetCurrentScreen(); + for (ref_ptr handle : selectResult) { - framesCounter = 0; - m_overlayTreeIsUpdating = true; + double const curDist = pt.SquareLength(handle->GetPivot(screen)); + if (curDist < dist) + { + dist = curDist; + featureID = handle->GetFeatureID(); + } } - if (m_overlayTreeIsUpdating) + return featureID; +} + +void FrontendRenderer::BeginUpdateOverlayTree(ScreenBase const & modelView) +{ + m_overlayTree->Frame(); + + if (m_overlayTree->IsNeedUpdate()) m_overlayTree->StartOverlayPlacing(modelView); } void FrontendRenderer::UpdateOverlayTree(ScreenBase const & modelView, drape_ptr & renderGroup) { - if (m_overlayTreeIsUpdating) + if (m_overlayTree->IsNeedUpdate()) renderGroup->CollectOverlay(make_ref(m_overlayTree)); else renderGroup->Update(modelView); @@ -343,10 +375,8 @@ void FrontendRenderer::UpdateOverlayTree(ScreenBase const & modelView, drape_ptr void FrontendRenderer::EndUpdateOverlayTree() { - if (m_overlayTreeIsUpdating) + if (m_overlayTree->IsNeedUpdate()) m_overlayTree->EndOverlayPlacing(); - - m_overlayTreeIsUpdating = false; } void FrontendRenderer::RenderScene(ScreenBase const & modelView) @@ -477,7 +507,16 @@ void FrontendRenderer::ResolveZoomLevel(const ScreenBase & screen) void FrontendRenderer::OnTap(m2::PointD const & pt, bool isLongTap) { - LOG(LINFO, ("Tap detected. Is long = ", isLongTap)); + double halfSize = VisualParams::Instance().GetTouchRectRadius(); + m2::PointD sizePoint(halfSize, halfSize); + m2::RectD selectRect(pt - sizePoint, pt + sizePoint); + + ScreenBase const & screen = m_userEventStream.GetCurrentScreen(); + bool isMyPosition = false; + if (m_myPositionController->IsModeHasPosition()) + isMyPosition = selectRect.IsPointInside(screen.GtoP(m_myPositionController->Position())); + + m_tapEventInfoFn(pt, isLongTap, isMyPosition, GetVisiblePOI(selectRect)); } bool FrontendRenderer::OnSingleTouchFiltrate(m2::PointD const & pt, TouchEvent::ETouchType type) @@ -671,6 +710,11 @@ void FrontendRenderer::AddUserEvent(UserEvent const & event) CancelMessageWaiting(); } +void FrontendRenderer::PositionChanged(m2::PointD const & position) +{ + m_userPositionChangedFn(position); +} + void FrontendRenderer::ChangeModelView(m2::PointD const & center) { AddUserEvent(SetCenterEvent(center, -1, true)); diff --git a/drape_frontend/frontend_renderer.hpp b/drape_frontend/frontend_renderer.hpp index 03a1afd923..c39ac335ee 100755 --- a/drape_frontend/frontend_renderer.hpp +++ b/drape_frontend/frontend_renderer.hpp @@ -43,6 +43,15 @@ namespace dp namespace df { +struct TapInfo +{ + m2::PointD const m_pixelPoint; + bool m_isLong; + + bool m_isMyPositionTapped; + FeatureID m_featureTapped; +}; + class FrontendRenderer : public BaseRenderer , public MyPositionController::Listener , public UserEventStream::Listener @@ -50,6 +59,8 @@ class FrontendRenderer : public BaseRenderer public: using TModelViewChanged = function; using TIsCountryLoaded = TIsCountryLoaded; + using TTapEventInfoFn = function; + using TUserPositionChangedFn = function; struct Params : BaseRenderer::Params { @@ -59,12 +70,16 @@ public: Viewport viewport, TModelViewChanged const & modelViewChangedFn, TIsCountryLoaded const & isCountryLoaded, + TTapEventInfoFn const & tapEventFn, + TUserPositionChangedFn const & positionChangedFn, location::TMyPositionModeChanged myPositionModeCallback, location::EMyPositionMode initMode) : BaseRenderer::Params(commutator, factory, texMng) , m_viewport(viewport) , m_modelViewChangedFn(modelViewChangedFn) , m_isCountryLoadedFn(isCountryLoaded) + , m_tapEventFn(tapEventFn) + , m_positionChangedFn(positionChangedFn) , m_myPositionModeCallback(myPositionModeCallback) , m_initMyPositionMode(initMode) {} @@ -72,6 +87,8 @@ public: Viewport m_viewport; TModelViewChanged m_modelViewChangedFn; TIsCountryLoaded m_isCountryLoadedFn; + TTapEventInfoFn m_tapEventFn; + TUserPositionChangedFn m_positionChangedFn; location::TMyPositionModeChanged m_myPositionModeCallback; location::EMyPositionMode m_initMyPositionMode; }; @@ -95,6 +112,7 @@ public: void AddUserEvent(UserEvent const & event); /// MyPositionController::Listener + void PositionChanged(m2::PointD const & position) override; void ChangeModelView(m2::PointD const & center) override; void ChangeModelView(double azimuth) override; void ChangeModelView(m2::RectD const & rect) override; @@ -160,6 +178,9 @@ private: void OnCompassTapped(); + FeatureID GetVisiblePOI(m2::PointD const & pixelPoint) const; + FeatureID GetVisiblePOI(m2::RectD const & pixelRect) const; + private: drape_ptr m_gpuProgramManager; @@ -172,13 +193,14 @@ private: drape_ptr m_myPositionController; drape_ptr m_overlayTree; - bool m_overlayTreeIsUpdating; dp::UniformValuesStorage m_generalUniforms; Viewport m_viewport; UserEventStream m_userEventStream; TModelViewChanged m_modelViewChangedFn; + TTapEventInfoFn m_tapEventInfoFn; + TUserPositionChangedFn m_userPositionChangedFn; unique_ptr m_tileTree; int m_currentZoomLevel = -1; diff --git a/drape_frontend/message.hpp b/drape_frontend/message.hpp index b02fcda1eb..b11ff99c53 100644 --- a/drape_frontend/message.hpp +++ b/drape_frontend/message.hpp @@ -28,7 +28,8 @@ public: StopRendering, ChangeMyPostitionMode, CompassInfo, - GpsInfo + GpsInfo, + FindVisiblePOI }; virtual ~Message() {} diff --git a/drape_frontend/message_subclasses.hpp b/drape_frontend/message_subclasses.hpp index c7d2bf79ed..989ab68dba 100644 --- a/drape_frontend/message_subclasses.hpp +++ b/drape_frontend/message_subclasses.hpp @@ -19,6 +19,7 @@ #include "platform/location.hpp" +#include "std/condition_variable.hpp" #include "std/shared_ptr.hpp" #include "std/set.hpp" #include "std/function.hpp" @@ -349,4 +350,68 @@ private: location::RouteMatchingInfo const m_routeInfo; }; +class BaseBlockingMessage : public Message +{ +public: + struct Blocker + { + void Wait() + { + unique_lock lock(m_lock); + m_signal.wait(lock, [this]{return !m_blocked;} ); + } + + private: + friend class BaseBlockingMessage; + + void Signal() + { + lock_guard lock(m_lock); + m_blocked = false; + m_signal.notify_one(); + } + + private: + mutex m_lock; + condition_variable m_signal; + bool m_blocked = true; + }; + + BaseBlockingMessage(Blocker & blocker) + : m_blocker(blocker) + { + } + + ~BaseBlockingMessage() + { + m_blocker.Signal(); + } + +private: + Blocker & m_blocker; +}; + +class FindVisiblePOIMessage : public BaseBlockingMessage +{ +public: + FindVisiblePOIMessage(Blocker & blocker, m2::PointD const & glbPt, FeatureID & featureID) + : BaseBlockingMessage(blocker) + , m_pt(glbPt) + , m_featureID(featureID) + { + } + + Type GetType() const override { return FindVisiblePOI; } + + m2::PointD const & GetPoint() const { return m_pt; } + void SetFeatureID(FeatureID const & id) + { + m_featureID = id; + } + +private: + m2::PointD m_pt; + FeatureID & m_featureID; +}; + } // namespace df diff --git a/drape_frontend/my_position_controller.cpp b/drape_frontend/my_position_controller.cpp index 55cd0ad3e4..e417d93376 100644 --- a/drape_frontend/my_position_controller.cpp +++ b/drape_frontend/my_position_controller.cpp @@ -306,6 +306,9 @@ void MyPositionController::Assign(location::GpsInfo const & info, bool isNavigab m_lastGPSBearing.Reset(); } + if (m_listener) + m_listener->PositionChanged(Position()); + m_isDirtyViewport = true; } diff --git a/drape_frontend/my_position_controller.hpp b/drape_frontend/my_position_controller.hpp index 3d74f78a4e..17427f3827 100644 --- a/drape_frontend/my_position_controller.hpp +++ b/drape_frontend/my_position_controller.hpp @@ -21,6 +21,7 @@ public: { public: virtual ~Listener() {} + virtual void PositionChanged(m2::PointD const & position) = 0; /// Show map with center in "center" point and current zoom virtual void ChangeModelView(m2::PointD const & center) = 0; /// Change azimuth of current ModelView diff --git a/drape_frontend/visual_params.cpp b/drape_frontend/visual_params.cpp index 9e4b19cb1d..5ca17a4c57 100644 --- a/drape_frontend/visual_params.cpp +++ b/drape_frontend/visual_params.cpp @@ -98,6 +98,11 @@ uint32_t VisualParams::GetTileSize() const return m_tileSize; } +uint32_t VisualParams::GetTouchRectRadius() const +{ + return 20 * GetVisualScale(); +} + VisualParams::VisualParams() : m_tileSize(0) , m_visualScale(0.0) diff --git a/drape_frontend/visual_params.hpp b/drape_frontend/visual_params.hpp index eb8343c59a..5cb38bf5ae 100644 --- a/drape_frontend/visual_params.hpp +++ b/drape_frontend/visual_params.hpp @@ -24,6 +24,9 @@ public: double GetVisualScale() const; uint32_t GetTileSize() const; + /// How many pixels around touch point are used to get bookmark or POI in consideration of visual scale + uint32_t GetTouchRectRadius() const; + private: int m_tileSize; double m_visualScale; diff --git a/iphone/Maps/Classes/MapViewController.mm b/iphone/Maps/Classes/MapViewController.mm index e78adbf096..bf84b18190 100644 --- a/iphone/Maps/Classes/MapViewController.mm +++ b/iphone/Maps/Classes/MapViewController.mm @@ -257,7 +257,6 @@ typedef NS_ENUM(NSUInteger, UserTouchesAction) - (void)popoverControllerDidDismissPopover:(UIPopoverController *)popoverController { [self destroyPopover]; - [self invalidate]; } - (void)sendTouchType:(df::TouchEvent::ETouchType)type withEvent:(UIEvent *)event @@ -345,7 +344,6 @@ typedef NS_ENUM(NSUInteger, UserTouchesAction) - (void)didRotateFromInterfaceOrientation:(UIInterfaceOrientation)fromInterfaceOrientation { [self showPopover]; - [self invalidate]; } - (void)didReceiveMemoryWarning @@ -382,7 +380,6 @@ typedef NS_ENUM(NSUInteger, UserTouchesAction) if (self.isViewLoaded && self.view.window) { - [self invalidate]; // only invalidate when map is displayed on the screen [self.controlsManager onEnterForeground]; } } @@ -840,14 +837,6 @@ NSInteger compareAddress(id l, id r, void * context) return NSOrderedSame; } -- (void)invalidate -{ - ///@TODO UVR -// Framework & f = GetFramework(); -// if (!f.SetUpdatesEnabled(true)) -// f.Invalidate(); -} - - (void)destroyPopover { self.popoverVC = nil; @@ -870,7 +859,6 @@ NSInteger compareAddress(id l, id r, void * context) { [self.popoverVC dismissPopoverAnimated:YES]; [self destroyPopover]; - [self invalidate]; } - (void)setRestoreRouteDestination:(m2::PointD)restoreRouteDestination diff --git a/iphone/Maps/Settings/SettingsViewController.mm b/iphone/Maps/Settings/SettingsViewController.mm index 5ab3f65882..88c28e0635 100644 --- a/iphone/Maps/Settings/SettingsViewController.mm +++ b/iphone/Maps/Settings/SettingsViewController.mm @@ -62,8 +62,6 @@ typedef NS_ENUM(NSUInteger, Section) - (void)viewWillDisappear:(BOOL)animated { [super viewWillDisappear:animated]; - ///@TODO UVR - //GetFramework().Invalidate(true); } #pragma mark - Table view data source diff --git a/iphone/Maps/resources-default/default.ui b/iphone/Maps/resources-default/default.ui deleted file mode 100644 index 50138f245a..0000000000 --- a/iphone/Maps/resources-default/default.ui +++ /dev/null @@ -1,50 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/map/address_finder.cpp b/map/address_finder.cpp index 955caabded..ce4cf1c41b 100644 --- a/map/address_finder.cpp +++ b/map/address_finder.cpp @@ -165,7 +165,7 @@ namespace void Framework::GetFeatureTypes(m2::PointD const & pxPoint, vector & types) const { m2::AnyRectD rect; - m_currentMovelView.GetTouchRect(pxPoint, TOUCH_PIXEL_RADIUS * df::VisualParams::Instance().GetVisualScale(), rect); + m_currentMovelView.GetTouchRect(pxPoint, df::VisualParams::Instance().GetTouchRectRadius(), rect); // This scale should fit in geometry scales range. int const scale = min(GetDrawScale(), scales::GetUpperScale()); diff --git a/map/framework.cpp b/map/framework.cpp index 668512a2c5..7207af61f0 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -163,6 +163,12 @@ void Framework::SetMyPositionModeListener(location::TMyPositionModeChanged const CallDrapeFunction(bind(&df::DrapeEngine::SetMyPositionModeListener, _1, fn)); } +void Framework::OnUserPositionChanged(m2::PointD const & position) +{ + MyPositionMarkPoint * myPostition = UserMarkContainer::UserMarkForMyPostion(); + myPostition->SetPtOrg(position); +} + void Framework::CallDrapeFunction(TDrapeFunction const & fn) { if (m_drapeEngine) @@ -538,7 +544,7 @@ void Framework::ShowBookmark(BookmarkAndCategory const & bnc) scale = scales::GetUpperComfortScale(); CallDrapeFunction(bind(&df::DrapeEngine::SetModelViewCenter, _1, mark->GetPivot(), scale, true)); - ActivateUserMark(mark); + ActivateUserMark(mark, true); m_balloonManager.OnShowMark(mark); } @@ -1351,6 +1357,8 @@ void Framework::CreateDrapeEngine(ref_ptr contextFactory, { m_currentMovelView = screen; }); + m_drapeEngine->SetTapEventInfoListener(bind(&Framework::OnTapEvent, this, _1, _2, _3, _4)); + m_drapeEngine->SetUserPositionListener(bind(&Framework::OnUserPositionChanged, this, _1)); OnSize(w, h); } @@ -1513,82 +1521,34 @@ void Framework::DisconnectMyPositionUpdate() } } -namespace +bool Framework::GetVisiblePOI(m2::PointD const & glbPoint, search::AddressInfo & info, feature::Metadata & metadata) const { + ASSERT(m_drapeEngine != nullptr, ()); + FeatureID id = m_drapeEngine->GetVisiblePOI(glbPoint); + if (!id.IsValid()) + return false; -///@TODO UVR -//OEPointerT GetClosestToPivot(list const & l, m2::PointD const & pxPoint) -//{ -// double dist = numeric_limits::max(); -// OEPointerT res; - -// for (list::const_iterator it = l.begin(); it != l.end(); ++it) -// { -// double const curDist = pxPoint.SquareLength((*it)->pivot()); -// if (curDist < dist) -// { -// dist = curDist; -// res = *it; -// } -// } - -// return res; -//} - + GetVisiblePOI(id, info, metadata); + return true; } -bool Framework::GetVisiblePOI(m2::PointD const & pxPoint, m2::PointD & pxPivot, - search::AddressInfo & info, feature::Metadata & metadata) const +m2::PointD Framework::GetVisiblePOI(FeatureID id, search::AddressInfo & info, feature::Metadata & metadata) const { -///@TODO UVR -// graphics::OverlayElement::UserInfo ui; + ASSERT(id.IsValid(), ()); + Index::FeaturesLoaderGuard guard(m_model.GetIndex(), id.m_mwmId); -// { -// // It seems like we don't need to lock frame here. -// // Overlay locking and storing items as shared_ptr is enough here. -// //m_renderPolicy->FrameLock(); + FeatureType ft; + guard.GetFeature(id.m_offset, ft); -// m2::PointD const pt = m_navigator.ShiftPoint(pxPoint); -// double const halfSize = TOUCH_PIXEL_RADIUS * GetVisualScale(); + ft.ParseMetadata(); + metadata = ft.GetMetadata(); -// list candidates; -// m2::RectD const rect(pt.x - halfSize, pt.y - halfSize, -// pt.x + halfSize, pt.y + halfSize); + ASSERT_NOT_EQUAL(ft.GetFeatureType(), feature::GEOM_LINE, ()); + m2::PointD const center = feature::GetCenter(ft); -// graphics::Overlay * frameOverlay = m_renderPolicy->FrameOverlay(); -// frameOverlay->lock(); -// frameOverlay->selectOverlayElements(rect, candidates); -// frameOverlay->unlock(); + GetAddressInfo(ft, center, info); -// OEPointerT elem = GetClosestToPivot(candidates, pt); - -// if (elem) -// ui = elem->userInfo(); - -// //m_renderPolicy->FrameUnlock(); -// } - - //if (ui.IsValid()) - //{ - //Index::FeaturesLoaderGuard guard(m_model.GetIndex(), ui.m_featureID.m_mwmId); - - //FeatureType ft; - //guard.GetFeatureByIndex(ui.m_featureID.m_index, ft); - - //ft.ParseMetadata(); - //metadata = ft.GetMetadata(); - - // @TODO experiment with other pivots - //ASSERT_NOT_EQUAL(ft.GetFeatureType(), feature::GEOM_LINE, ()); - //m2::PointD const center = feature::GetCenter(ft); - -// GetAddressInfo(ft, center, info); - -// pxPivot = GtoP(center); -// return true; -// } - - return false; + return GtoP(center); } namespace @@ -1649,111 +1609,6 @@ void Framework::FindClosestPOIMetadata(m2::PointD const & pt, feature::Metadata doFind.LoadMetadata(m_model, metadata); } -//shared_ptr const & Framework::GetLocationState() const -//{ -// return m_informationDisplay.locationState(); -//} - -void Framework::ActivateUserMark(UserMark const * mark, bool needAnim) -{ - ///@TODO UVR -// if (mark != UserMarkContainer::UserMarkForMyPostion()) -// DisconnectMyPositionUpdate(); -// m_bmManager.ActivateMark(mark, needAnim); -} - -bool Framework::HasActiveUserMark() const -{ - return false; - ///@TODO UVR - //return m_bmManager.UserMarkHasActive(); -} - -UserMark const * Framework::GetUserMark(m2::PointD const & pxPoint, bool isLongPress) -{ - // The main idea is to calculate POI rank based on the frequency users are clicking them. - UserMark const * mark = GetUserMarkWithoutLogging(pxPoint, isLongPress); - alohalytics::TStringMap details {{"isLongPress", isLongPress ? "1" : "0"}}; - if (mark) - mark->FillLogEvent(details); - alohalytics::Stats::Instance().LogEvent("$GetUserMark", details); - return mark; -} - -UserMark const * Framework::GetUserMarkWithoutLogging(m2::PointD const & pxPoint, bool isLongPress) -{ - DisconnectMyPositionUpdate(); - m2::AnyRectD rect; - double vs = df::VisualParams::Instance().GetVisualScale(); - m_currentMovelView.GetTouchRect(pxPoint, TOUCH_PIXEL_RADIUS * vs, rect); - - ///@TODO UVR - //shared_ptr const & locationState = GetLocationState(); -// if (locationState->IsModeHasPosition()) -// { -// m2::PointD const & glPivot = locationState->Position(); -// if (rect.IsPointInside(glPivot)) -// { -// search::AddressInfo info; -// info.m_name = m_stringsBundle.GetString("my_position"); -// MyPositionMarkPoint * myPostition = UserMarkContainer::UserMarkForMyPostion(); -// m_locationChangedSlotID = locationState->AddPositionChangedListener(bind(&Framework::UpdateSelectedMyPosition, this, _1)); -// myPostition->SetPtOrg(glPivot); -// myPostition->SetInfo(info); -// return myPostition; -// } -// } - - m2::AnyRectD bmSearchRect; - double const pxWidth = TOUCH_PIXEL_RADIUS * vs; - double const pxHeight = (TOUCH_PIXEL_RADIUS + BM_TOUCH_PIXEL_INCREASE) * vs; - m_currentMovelView.GetTouchRect(pxPoint + m2::PointD(0, BM_TOUCH_PIXEL_INCREASE), - pxWidth, pxHeight, bmSearchRect); - - UserMark const * mark = m_bmManager.FindNearestUserMark( - [&rect, &bmSearchRect](UserMarkType type) -> m2::AnyRectD const & - { - return (type == UserMarkContainer::BOOKMARK_MARK ? bmSearchRect : rect); - }); - - if (mark == NULL) - { - bool needMark = false; - m2::PointD pxPivot; - search::AddressInfo info; - feature::Metadata metadata; - if (GetVisiblePOI(pxPoint, pxPivot, info, metadata)) - needMark = true; - else if (isLongPress) - { - GetAddressInfoForPixelPoint(pxPoint, info); - pxPivot = pxPoint; - needMark = true; - } - - if (needMark) - { - PoiMarkPoint * poiMark = UserMarkContainer::UserMarkForPoi(); - poiMark->SetPtOrg(m_currentMovelView.PtoG(pxPivot)); - poiMark->SetInfo(info); - poiMark->SetMetadata(move(metadata)); - mark = poiMark; - } - } - - return mark; -} - -PoiMarkPoint * Framework::GetAddressMark(m2::PointD const & globalPoint) const -{ - search::AddressInfo info; - GetAddressInfoForGlobalPoint(globalPoint, info); - PoiMarkPoint * mark = UserMarkContainer::UserMarkForPoi(); - mark->SetPtOrg(globalPoint); - mark->SetInfo(info); - return mark; -} - BookmarkAndCategory Framework::FindBookmark(UserMark const * mark) const { BookmarkAndCategory empty = MakeEmptyBookmarkAndCategory(); @@ -1782,6 +1637,97 @@ BookmarkAndCategory Framework::FindBookmark(UserMark const * mark) const return result; } +PoiMarkPoint * Framework::GetAddressMark(m2::PointD const & globalPoint) const +{ + search::AddressInfo info; + GetAddressInfoForGlobalPoint(globalPoint, info); + PoiMarkPoint * mark = UserMarkContainer::UserMarkForPoi(); + mark->SetPtOrg(globalPoint); + mark->SetInfo(info); + return mark; +} + +void Framework::ActivateUserMark(UserMark const * mark, bool needAnim) +{ + m_balloonManager.OnShowMark(mark); +} + +void Framework::OnTapEvent(m2::PointD pxPoint, bool isLong, bool isMyPosition, FeatureID feature) +{ + UserMark const * mark = OnTapEventImpl(pxPoint, isLong, isMyPosition, feature); + + { + alohalytics::TStringMap details {{"isLongPress", isLong ? "1" : "0"}}; + if (mark) + mark->FillLogEvent(details); + alohalytics::Stats::Instance().LogEvent("$GetUserMark", details); + } + + ActivateUserMark(mark, true); +} + +UserMark const * Framework::OnTapEventImpl(m2::PointD pxPoint, bool isLong, bool isMyPosition, FeatureID feature) +{ + if (isMyPosition) + { + search::AddressInfo info; + info.m_name = m_stringsBundle.GetString("my_position"); + MyPositionMarkPoint * myPostition = UserMarkContainer::UserMarkForMyPostion(); + myPostition->SetInfo(info); + + return myPostition; + } + + df::VisualParams const & vp = df::VisualParams::Instance(); + + m2::AnyRectD rect; + uint32_t const touchRadius = vp.GetTouchRectRadius(); + m_currentMovelView.GetTouchRect(pxPoint, touchRadius, rect); + + m2::AnyRectD bmSearchRect; + double const bmAddition = BM_TOUCH_PIXEL_INCREASE * vp.GetVisualScale(); + double const pxWidth = touchRadius; + double const pxHeight = touchRadius + bmAddition; + m_currentMovelView.GetTouchRect(pxPoint + m2::PointD(0, bmAddition), + pxWidth, pxHeight, bmSearchRect); + UserMark const * mark = m_bmManager.FindNearestUserMark( + [&rect, &bmSearchRect](UserMarkType type) -> m2::AnyRectD const & + { + return (type == UserMarkContainer::BOOKMARK_MARK ? bmSearchRect : rect); + }); + + if (mark != nullptr) + return mark; + + bool needMark = false; + m2::PointD pxPivot; + search::AddressInfo info; + feature::Metadata metadata; + + if (feature.IsValid()) + { + pxPivot = GetVisiblePOI(feature, info, metadata); + needMark = true; + } + else if (isLong) + { + GetAddressInfoForPixelPoint(pxPoint, info); + pxPivot = pxPoint; + needMark = true; + } + + if (needMark) + { + PoiMarkPoint * poiMark = UserMarkContainer::UserMarkForPoi(); + poiMark->SetPtOrg(m_currentMovelView.PtoG(pxPivot)); + poiMark->SetInfo(info); + poiMark->SetMetadata(metadata); + return poiMark; + } + + return nullptr; +} + void Framework::PredictLocation(double & lat, double & lon, double accuracy, double bearing, double speed, double elapsedSeconds) { diff --git a/map/framework.hpp b/map/framework.hpp index 9ec50ccfb3..b987be8474 100644 --- a/map/framework.hpp +++ b/map/framework.hpp @@ -111,9 +111,6 @@ protected: shared_ptr m_activeMaps; storage::CountryTree m_globalCntTree; - /// How many pixels around touch point are used to get bookmark or POI - static const int TOUCH_PIXEL_RADIUS = 20; - /// This function is called by m_storage when latest local files /// were changed. void UpdateLatestCountryFile(platform::LocalCountryFile const & localFile); @@ -236,16 +233,19 @@ public: bool AddBookmarksFile(string const & filePath); - void ActivateUserMark(UserMark const * mark, bool needAnim = true); - bool HasActiveUserMark() const; - UserMark const * GetUserMarkWithoutLogging(m2::PointD const & pxPoint, bool isLongPress); - UserMark const * GetUserMark(m2::PointD const & pxPoint, bool isLongPress); - PoiMarkPoint * GetAddressMark(m2::PointD const & globalPoint) const; BookmarkAndCategory FindBookmark(UserMark const * mark) const; - BookmarkManager & GetBookmarkManager() { return m_bmManager; } + + void ActivateUserMark(UserMark const * mark, bool needAnim); + PoiMarkPoint * GetAddressMark(m2::PointD const & globalPoint) const; + +private: + void OnTapEvent(m2::PointD pxPoint, bool isLong, bool isMyPosition, FeatureID feature); + UserMark const * OnTapEventImpl(m2::PointD pxPoint, bool isLong, bool isMyPosition, FeatureID feature); //@} +public: + /// @name GPS location updates routine. //@{ void OnLocationError(location::TLocationError error); @@ -254,8 +254,12 @@ public: void SwitchMyPositionNextMode(); void InvalidateMyPosition(); void SetMyPositionModeListener(location::TMyPositionModeChanged const & fn); + +private: + void OnUserPositionChanged(m2::PointD const & position); //@} +public: void CreateDrapeEngine(ref_ptr contextFactory, float vs, int w, int h); ref_ptr GetDrapeEngine(); void DestroyDrapeEngine(); @@ -388,7 +392,8 @@ private: void GetLocality(m2::PointD const & pt, search::AddressInfo & info) const; public: - bool GetVisiblePOI(m2::PointD const & pxPoint, m2::PointD & pxPivot, search::AddressInfo & info, feature::Metadata & metadata) const; + bool GetVisiblePOI(m2::PointD const & glbPoint, search::AddressInfo & info, feature::Metadata & metadata) const; + m2::PointD GetVisiblePOI(FeatureID id, search::AddressInfo & info, feature::Metadata & metadata) const; void FindClosestPOIMetadata(m2::PointD const & pt, feature::Metadata & metadata) const; void MemoryWarning(); diff --git a/map/pin_click_manager.cpp b/map/pin_click_manager.cpp index 733743d4b3..5764fd1ecf 100644 --- a/map/pin_click_manager.cpp +++ b/map/pin_click_manager.cpp @@ -14,8 +14,6 @@ PinClickManager::PinClickManager(Framework & f) void PinClickManager::Hide() { - ///@TODO UVR - //m_f.Invalidate(); } void PinClickManager::OnShowMark(UserMark const * mark) @@ -27,18 +25,13 @@ void PinClickManager::OnShowMark(UserMark const * mark) void PinClickManager::SetBalloonVisible(bool isVisible) { - if (!isVisible && m_f.HasActiveUserMark()) + if (!isVisible) OnDismiss(); - - ///@TODO UVR - //m_f.Invalidate(); } void PinClickManager::RemovePin() { - m_f.ActivateUserMark(NULL); - ///@TODO UVR - //m_f.Invalidate(); + ///@TODO } void PinClickManager::Dismiss() diff --git a/map/user_mark.cpp b/map/user_mark.cpp index a88ebee6c1..cc2e69c60b 100644 --- a/map/user_mark.cpp +++ b/map/user_mark.cpp @@ -1,6 +1,23 @@ #include "user_mark.hpp" #include "user_mark_container.hpp" +namespace +{ + +string ToString(UserMark::Type t) +{ + switch (t) + { + case UserMark::Type::BOOKMARK: return "BOOKMARK"; + case UserMark::Type::API: return "API"; + case UserMark::Type::MY_POSITION: return "MY_POSITION"; + case UserMark::Type::POI: return "POI"; + case UserMark::Type::SEARCH: return "SEARCH"; + case UserMark::Type::DEBUG_MARK: return "DEBUG"; + } +} + +} UserMark::UserMark(m2::PointD const & ptOrg, UserMarkContainer * container) : m_ptOrg(ptOrg), m_container(container) @@ -40,6 +57,7 @@ void UserMark::FillLogEvent(UserMark::TEventContainer & details) const GetLatLon(lat, lon); details.emplace("lat", strings::to_string(lat)); details.emplace("lon", strings::to_string(lon)); + details.emplace("markType", ToString(GetMarkType())); } //////////////////////////////////////////////////////////////// @@ -115,7 +133,6 @@ unique_ptr ApiMarkPoint::Copy() const void ApiMarkPoint::FillLogEvent(UserMark::TEventContainer & details) const { UserMark::FillLogEvent(details); - details.emplace("markType", "API"); details.emplace("name", GetName()); } @@ -172,7 +189,6 @@ unique_ptr SearchMarkPoint::Copy() const void SearchMarkPoint::FillLogEvent(UserMark::TEventContainer & details) const { UserMark::FillLogEvent(details); - details.emplace("markType", "SEARCH"); details.emplace("name", m_info.GetPinName()); details.emplace("type", m_info.GetPinType()); details.emplace("metaData", m_metadata.Empty() ? "0" : "1"); @@ -192,13 +208,6 @@ unique_ptr PoiMarkPoint::Copy() const { return unique_ptr(new UserMarkCopy(this, false)); } - -void PoiMarkPoint::FillLogEvent(UserMark::TEventContainer & details) const -{ - SearchMarkPoint::FillLogEvent(details); - details.emplace("markType", "POI"); -} - void PoiMarkPoint::SetPtOrg(m2::PointD const & ptOrg) { m_ptOrg = ptOrg; @@ -221,13 +230,6 @@ UserMark::Type MyPositionMarkPoint::GetMarkType() const return UserMark::Type::MY_POSITION; } -void MyPositionMarkPoint::FillLogEvent(UserMark::TEventContainer & details) const -{ - PoiMarkPoint::FillLogEvent(details); - details.emplace("markType", "MY_POSITION"); -} - - DebugMarkPoint::DebugMarkPoint(const m2::PointD & ptOrg, UserMarkContainer * container) : UserMark(ptOrg, container) { @@ -242,9 +244,3 @@ unique_ptr DebugMarkPoint::Copy() const { return unique_ptr(new UserMarkCopy(new DebugMarkPoint(m_ptOrg, m_container))); } - -void DebugMarkPoint::FillLogEvent(UserMark::TEventContainer & details) const -{ - UserMark::FillLogEvent(details); - details.emplace("markType", "DEBUG"); -} diff --git a/map/user_mark.hpp b/map/user_mark.hpp index 83b3b1e066..e508b2cbfd 100644 --- a/map/user_mark.hpp +++ b/map/user_mark.hpp @@ -130,7 +130,6 @@ public: PoiMarkPoint(UserMarkContainer * container); UserMark::Type GetMarkType() const override; unique_ptr Copy() const override; - virtual void FillLogEvent(TEventContainer & details) const override; void SetPtOrg(m2::PointD const & ptOrg); void SetName(string const & name); @@ -142,18 +141,15 @@ public: MyPositionMarkPoint(UserMarkContainer * container); UserMark::Type GetMarkType() const override; - void FillLogEvent(TEventContainer & details) const override; }; class DebugMarkPoint : public UserMark { public: DebugMarkPoint(m2::PointD const & ptOrg, UserMarkContainer * container); -; + string GetSymbolName() const override; Type GetMarkType() const override { return UserMark::Type::DEBUG_MARK; } unique_ptr Copy() const override; - - void FillLogEvent(TEventContainer & details) const override; }; diff --git a/qt/draw_widget.cpp b/qt/draw_widget.cpp index 95624c4555..adb80b0c3f 100644 --- a/qt/draw_widget.cpp +++ b/qt/draw_widget.cpp @@ -80,7 +80,7 @@ bool IsLocationEmulation(QMouseEvent * e) // Initialize with some stubs for test. PinClickManager & manager = GetBalloonManager(); - manager.ConnectUserMarkListener(bind(&DrawWidget::OnActivateMark, this, _1)); + manager.ConnectUserMarkListener([](unique_ptr) {}); manager.ConnectDismissListener(&DummyDismiss); m_framework->SetRouteBuildingListener([] (routing::IRouter::ResultCode, vector const &) @@ -169,12 +169,6 @@ bool IsLocationEmulation(QMouseEvent * e) m_enableScaleUpdate = true; } - void DrawWidget::OnActivateMark(unique_ptr pCopy) - { - UserMark const * pMark = pCopy->GetUserMark(); - m_framework->ActivateUserMark(pMark); - } - void DrawWidget::CreateEngine() { m_framework->CreateDrapeEngine(make_ref(m_contextFactory), m_ratio, @@ -365,14 +359,8 @@ bool IsLocationEmulation(QMouseEvent * e) menu.addAction(QString::fromUtf8(s.c_str())); }; - // Get POI under cursor or nearest address by point. - m2::PointD dummy; search::AddressInfo info; - feature::FeatureMetadata metadata; - if (m_framework->GetVisiblePOI(pt, dummy, info, metadata)) - addStringFn("POI"); - else - m_framework->GetAddressInfoForPixelPoint(pt, info); + m_framework->GetAddressInfoForPixelPoint(pt, info); // Get feature types under cursor. vector types; diff --git a/qt/draw_widget.hpp b/qt/draw_widget.hpp index a4dd8b7e4c..786514a91f 100644 --- a/qt/draw_widget.hpp +++ b/qt/draw_widget.hpp @@ -62,9 +62,6 @@ namespace qt void SetRouter(routing::RouterType routerType); Q_SIGNAL void EngineCreated(); - protected: - void OnActivateMark(unique_ptr pCopy); - void CreateEngine(); protected: