[drape] my position controller and user event stream communications

This commit is contained in:
ExMix 2015-05-26 12:28:48 +03:00 committed by r.kuznetsov
parent 148265a517
commit 46ef7942f6
11 changed files with 260 additions and 70 deletions

View file

@ -8,8 +8,7 @@ namespace
double GetMoveDuration(m2::PointD const & startPx, m2::PointD const & endPt, int minDisplaySize)
{
double const length = (endPt - startPx).Length();
return (length / minDisplaySize);
return max(0.5, min(0.5, 0.5 * startPx.Length(endPt) / 50.0));
}
double GetRotateDuration(double const & startAngle, double const & endAngle)

View file

@ -6,9 +6,9 @@ namespace df
OpacityAnimation::OpacityAnimation(double duration, double startOpacity, double endOpacity)
: BaseInterpolator(duration)
, m_opacity(startOpacity)
, m_startOpacity(startOpacity)
, m_endOpacity(endOpacity)
, m_opacity(startOpacity)
{
}

View file

@ -12,22 +12,27 @@
namespace
{
class UserEventStreamTest
class UserEventStreamTest : df::UserEventStream::Listener
{
public:
UserEventStreamTest(bool filtrateTouches)
: m_stream([](m2::PointD const &) { return true; })
, m_filtrate(filtrateTouches)
{
auto const tapDetectedFn = [](m2::PointD const &, bool isLong) {};
auto const filtrateFn = [&filtrateTouches](m2::PointD const &, df::TouchEvent::ETouchType)
{
return filtrateTouches;
};
m_stream.SetTapListener(tapDetectedFn, filtrateFn);
m_stream.SetTestBridge(bind(&UserEventStreamTest::TestBridge, this, _1));
}
void OnTap(const m2::PointD & pt, bool isLong) override {}
bool OnSingleTouchFiltrate(m2::PointD const & pt, df::TouchEvent::ETouchType type) override { return m_filtrate; }
void OnDragStarted() override {}
void OnDragEnded(m2::PointD const & /*distance*/) override {}
void OnRotated() override {}
void OnScaleStarted() override {}
void CorrectScalePoint(m2::PointD & pt) const override {}
void CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const override {}
void OnScaleEnded() override {}
void AddUserEvent(df::TouchEvent const & event)
{
m_stream.AddEvent(event);
@ -62,6 +67,7 @@ private:
private:
df::UserEventStream m_stream;
list<char const *> m_expectation;
bool m_filtrate;
};
df::TouchEvent MakeTouchEvent(m2::PointD const & pt1, m2::PointD const & pt2, df::TouchEvent::ETouchType type)

View file

@ -445,12 +445,12 @@ void FrontendRenderer::ResolveZoomLevel(const ScreenBase & screen)
m_currentZoomLevel = (zoomLevel <= upperScale ? zoomLevel : upperScale);
}
void FrontendRenderer::TapDetected(m2::PointD const & pt, bool isLongTap)
void FrontendRenderer::OnTap(m2::PointD const & pt, bool isLongTap)
{
LOG(LINFO, ("Tap detected. Is long = ", isLongTap));
}
bool FrontendRenderer::SingleTouchFiltration(m2::PointD const & pt, TouchEvent::ETouchType type)
bool FrontendRenderer::OnSingleTouchFiltrate(m2::PointD const & pt, TouchEvent::ETouchType type)
{
float const rectHalfSize = 5 * df::VisualParams::Instance().GetVisualScale();
m2::RectD r(-rectHalfSize, -rectHalfSize, rectHalfSize, rectHalfSize);
@ -460,15 +460,12 @@ bool FrontendRenderer::SingleTouchFiltration(m2::PointD const & pt, TouchEvent::
{
case TouchEvent::ETouchType::TOUCH_DOWN:
return m_guiRenderer->OnTouchDown(r);
case TouchEvent::ETouchType::TOUCH_UP:
m_guiRenderer->OnTouchUp(r);
return false;
case TouchEvent::ETouchType::TOUCH_CANCEL:
m_guiRenderer->OnTouchCancel(r);
return false;
case TouchEvent::ETouchType::TOUCH_MOVE:
return false;
}
@ -476,6 +473,41 @@ bool FrontendRenderer::SingleTouchFiltration(m2::PointD const & pt, TouchEvent::
return false;
}
void FrontendRenderer::OnDragStarted()
{
m_myPositionController->DragStarted();
}
void FrontendRenderer::OnDragEnded(m2::PointD const & distance)
{
m_myPositionController->DragEnded(distance);
}
void FrontendRenderer::OnScaleStarted()
{
m_myPositionController->ScaleStarted();
}
void FrontendRenderer::OnRotated()
{
m_myPositionController->Rotated();
}
void FrontendRenderer::CorrectScalePoint(m2::PointD & pt) const
{
m_myPositionController->CorrectScalePoint(pt);
}
void FrontendRenderer::CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const
{
m_myPositionController->CorrectScalePoint(pt1, pt2);
}
void FrontendRenderer::OnScaleEnded()
{
m_myPositionController->ScaleEnded();
}
void FrontendRenderer::ResolveTileKeys(ScreenBase const & screen, TTilesCollection & tiles)
{
// equal for x and y
@ -513,15 +545,13 @@ void FrontendRenderer::Routine::Do()
{
gui::DrapeGui::Instance().ConnectOnCompassTappedHandler(bind(&FrontendRenderer::OnCompassTapped, &m_renderer));
m_renderer.m_myPositionController->SetListener(ref_ptr<MyPositionController::Listener>(&m_renderer));
m_renderer.m_userEventStream.SetListener(ref_ptr<UserEventStream::Listener>(&m_renderer));
m_renderer.m_tileTree->SetHandlers(bind(&FrontendRenderer::OnAddRenderGroup, &m_renderer, _1, _2, _3),
bind(&FrontendRenderer::OnDeferRenderGroup, &m_renderer, _1, _2, _3),
bind(&FrontendRenderer::OnActivateTile, &m_renderer, _1),
bind(&FrontendRenderer::OnRemoveTile, &m_renderer, _1));
m_renderer.m_userEventStream.SetTapListener(bind(&FrontendRenderer::TapDetected, &m_renderer, _1, _2),
bind(&FrontendRenderer::SingleTouchFiltration, &m_renderer, _1, _2));
dp::OGLContext * context = m_renderer.m_contextFactory->getDrawContext();
context->makeCurrent();
GLFunctions::Init();
@ -634,7 +664,7 @@ void FrontendRenderer::ChangeModelView(m2::PointD const & userPos, double azimut
viewVector.Normalize();
AddUserEvent(SetAnyRectEvent(m2::AnyRectD(userPos + (viewVector * offset), -azimuth,
screen.GlobalRect().GetLocalRect()), true /* animate */));
m_userEventStream.GetTargetRect().GetLocalRect()), true /* animate */));
}
ScreenBase const & FrontendRenderer::UpdateScene(bool & modelViewChanged)

View file

@ -41,6 +41,7 @@ namespace df
class FrontendRenderer : public BaseRenderer
, public MyPositionController::Listener
, public UserEventStream::Listener
{
public:
using TModelViewChanged = function<void (ScreenBase const & screen)>;
@ -112,8 +113,16 @@ private:
int GetCurrentZoomLevel() const;
void ResolveZoomLevel(ScreenBase const & screen);
void TapDetected(const m2::PointD & pt, bool isLongTap);
bool SingleTouchFiltration(m2::PointD const & pt, TouchEvent::ETouchType type);
void OnTap(m2::PointD const & pt, bool isLong) override;
bool OnSingleTouchFiltrate(m2::PointD const & pt, TouchEvent::ETouchType type) override;
void OnDragStarted() override;
void OnDragEnded(m2::PointD const & distance) override;
void OnScaleStarted() override;
void OnRotated() override;
void CorrectScalePoint(m2::PointD & pt) const override;
void CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const override;
void OnScaleEnded() override;
private:
class Routine : public threads::IRoutine

View file

@ -92,6 +92,7 @@ void MyPosition::Render(ScreenBase const & screen,
dp::UniformValuesStorage accuracyUniforms = uniforms;
accuracyUniforms.SetFloatValue("u_position", m_position.x, m_position.y, dp::depth::POSITION_ACCURACY);
accuracyUniforms.SetFloatValue("u_accuracy", pixelAccuracy);
accuracyUniforms.SetFloatValue("u_opacity", 1.0);
RenderPart(mng, accuracyUniforms, MY_POSITION_ACCURACY);
}
@ -99,6 +100,7 @@ void MyPosition::Render(ScreenBase const & screen,
dp::UniformValuesStorage arrowUniforms = uniforms;
arrowUniforms.SetFloatValue("u_position", m_position.x, m_position.y, dp::depth::MY_POSITION_MARK);
arrowUniforms.SetFloatValue("u_azimut", -(m_azimuth + screen.GetAngle()));
arrowUniforms.SetFloatValue("u_opacity", 1.0);
RenderPart(mng, arrowUniforms, (m_showAzimuth == true) ? MY_POSITION_ARROW : MY_POSITION_POINT);
}
}

View file

@ -15,27 +15,27 @@ int const POSITION_Y_OFFSET = 120;
double const GPS_BEARING_LIFETIME_S = 5.0;
double const MIN_SPEED_THRESHOLD_MPS = 1.0;
uint16_t SetModeBit(uint16_t mode, uint16_t bit)
uint16_t SetModeBit(uint32_t mode, uint32_t bit)
{
return mode | bit;
}
//uint16_t ResetModeBit(uint16_t mode, uint16_t bit)
//{
// return mode & (~bit);
//}
uint16_t ResetModeBit(uint32_t mode, uint32_t bit)
{
return mode & (~bit);
}
location::EMyPositionMode ResetAllModeBits(uint16_t mode)
location::EMyPositionMode ResetAllModeBits(uint32_t mode)
{
return (location::EMyPositionMode)(mode & 0xF);
}
uint16_t ChangeMode(uint16_t mode, location::EMyPositionMode newMode)
uint16_t ChangeMode(uint32_t mode, location::EMyPositionMode newMode)
{
return (mode & 0xF0) | newMode;
}
bool TestModeBit(uint16_t mode, uint16_t bit)
bool TestModeBit(uint32_t mode, uint32_t bit)
{
return (mode & bit) != 0;
}
@ -89,6 +89,59 @@ bool MyPositionController::IsModeHasPosition() const
return GetMode() >= location::MODE_NOT_FOLLOW;
}
void MyPositionController::DragStarted()
{
SetModeInfo(SetModeBit(m_modeInfo, BlockAnimation));
}
void MyPositionController::DragEnded(m2::PointD const & distance)
{
SetModeInfo(ResetModeBit(m_modeInfo, BlockAnimation));
if (distance.Length() > 0.2 * min(m_pixelRect.SizeX(), m_pixelRect.SizeY()))
StopLocationFollow();
Follow();
}
void MyPositionController::ScaleStarted()
{
SetModeInfo(SetModeBit(m_modeInfo, BlockAnimation));
}
void MyPositionController::Rotated()
{
location::EMyPositionMode mode = GetMode();
if (mode == location::MODE_ROTATE_AND_FOLLOW)
SetModeInfo(SetModeBit(m_modeInfo, StopFollowOnActionEnd));
}
void MyPositionController::CorrectScalePoint(m2::PointD & pt) const
{
if (IsModeChangeViewport())
pt = GetCurrentPixelBinding();
}
void MyPositionController::CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const
{
if (IsModeChangeViewport())
{
m2::PointD const ptDiff = GetCurrentPixelBinding() - (pt1 + (pt2 - pt1) * 0.5);
pt1 += ptDiff;
pt2 += ptDiff;
}
}
void MyPositionController::ScaleEnded()
{
SetModeInfo(ResetModeBit(m_modeInfo, BlockAnimation));
if (TestModeBit(m_modeInfo, StopFollowOnActionEnd))
{
SetModeInfo(ResetModeBit(m_modeInfo, StopFollowOnActionEnd));
StopLocationFollow();
}
Follow();
}
void MyPositionController::SetRenderShape(drape_ptr<MyPosition> && shape)
{
m_shape = move(shape);
@ -145,6 +198,7 @@ void MyPositionController::NextMode()
}
SetModeInfo(ChangeMode(m_modeInfo, newMode));
Follow();
}
void MyPositionController::TurnOff()
@ -201,7 +255,7 @@ void MyPositionController::Render(ScreenBase const & screen, ref_ptr<dp::GpuProg
location::EMyPositionMode currentMode = GetMode();
if (m_shape != nullptr && IsVisible() && currentMode > location::MODE_PENDING_POSITION)
{
if (m_isDirtyViewport)
if (m_isDirtyViewport && !TestModeBit(m_modeInfo, BlockAnimation))
{
Follow();
m_isDirtyViewport = false;
@ -273,7 +327,7 @@ void MyPositionController::SetDirection(double bearing)
SetModeInfo(SetModeBit(m_modeInfo, KnownDirectionBit));
}
void MyPositionController::SetModeInfo(uint16_t modeInfo, bool force)
void MyPositionController::SetModeInfo(uint32_t modeInfo, bool force)
{
location::EMyPositionMode const newMode = ResetAllModeBits(modeInfo);
location::EMyPositionMode const oldMode = GetMode();
@ -290,7 +344,7 @@ location::EMyPositionMode MyPositionController::GetMode() const
return ResetAllModeBits(m_modeInfo);
}
void MyPositionController::CallModeListener(uint16_t mode)
void MyPositionController::CallModeListener(uint32_t mode)
{
if (m_modeChangeCallback != nullptr)
m_modeChangeCallback(ResetAllModeBits(mode));
@ -321,6 +375,7 @@ void MyPositionController::StopCompassFollow()
return;
SetModeInfo(ChangeMode(m_modeInfo, location::MODE_FOLLOW));
Follow();
}
void MyPositionController::ChangeModelView(m2::PointD const & center)
@ -352,15 +407,28 @@ void MyPositionController::Follow()
{
location::EMyPositionMode currentMode = GetMode();
if (currentMode == location::MODE_FOLLOW)
{
ChangeModelView(m_position);
}
else if (currentMode == location::MODE_ROTATE_AND_FOLLOW)
{
m2::PointD pxZero(m_pixelRect.Center().x,
m_pixelRect.maxY() - POSITION_Y_OFFSET * VisualParams::Instance().GetVisualScale());
ChangeModelView(m_position, m_drawDirection, pxZero);
}
ChangeModelView(m_position, m_drawDirection, GetRaFPixelBinding());
}
m2::PointD MyPositionController::GetRaFPixelBinding() const
{
return m2::PointD (m_pixelRect.Center().x,
m_pixelRect.maxY() - POSITION_Y_OFFSET * VisualParams::Instance().GetVisualScale());
}
m2::PointD MyPositionController::GetCurrentPixelBinding() const
{
location::EMyPositionMode mode = GetMode();
if (mode == location::MODE_FOLLOW)
return m_pixelRect.Center();
else if (mode == location::MODE_ROTATE_AND_FOLLOW)
return GetRaFPixelBinding();
else
ASSERT(false, ());
return m2::PointD::Zero();
}
}

View file

@ -42,6 +42,15 @@ public:
bool IsModeChangeViewport() const;
bool IsModeHasPosition() const;
void DragStarted();
void DragEnded(m2::PointD const & distance);
void ScaleStarted();
void Rotated();
void CorrectScalePoint(m2::PointD & pt) const;
void CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const;
void ScaleEnded();
void SetRenderShape(drape_ptr<MyPosition> && shape);
void SetFixedZoom();
@ -67,9 +76,9 @@ private:
void Assign(location::CompassInfo const & info);
void SetDirection(double bearing);
void SetModeInfo(uint16_t modeInfo, bool force = false);
void SetModeInfo(uint32_t modeInfo, bool force = false);
location::EMyPositionMode GetMode() const;
void CallModeListener(uint16_t mode);
void CallModeListener(uint32_t mode);
bool IsInRouting() const;
bool IsRotationActive() const;
@ -83,16 +92,20 @@ private:
void ChangeModelView(m2::PointD const & userPos, double azimuth, m2::PointD const & pxZero);
void Follow();
m2::PointD GetRaFPixelBinding() const;
m2::PointD GetCurrentPixelBinding() const;
private:
// Mode bits
// {
static uint16_t const FixedZoomBit = 0x20;
static uint16_t const RoutingSessionBit = 0x40;
static uint16_t const KnownDirectionBit = 0x80;
static uint32_t const FixedZoomBit = 0x20;
static uint32_t const RoutingSessionBit = 0x40;
static uint32_t const KnownDirectionBit = 0x80;
static uint32_t const BlockAnimation = 0x100;
static uint32_t const StopFollowOnActionEnd = 0x200;
// }
uint16_t m_modeInfo; // combination of Mode enum and "Mode bits"
uint32_t m_modeInfo; // combination of Mode enum and "Mode bits"
location::EMyPositionMode m_afterPendingMode;
location::TMyPositionModeChanged m_modeChangeCallback;

View file

@ -100,6 +100,11 @@ bool RenderGroupComparator::operator()(drape_ptr<RenderGroup> const & l, drape_p
if (rPendingOnDelete == lPendingOnDelete)
{
dp::GLState::DepthLayer lDepth = lState.GetDepthLayer();
dp::GLState::DepthLayer rDepth = rState.GetDepthLayer();
if (lDepth != rDepth)
return lDepth < rDepth;
if (my::AlmostEqual(l->GetOpacity(), r->GetOpacity()))
return lState < rState;
else

View file

@ -39,6 +39,8 @@ char const * UserEventStream::CANCEL_FILTER = "CancelFilter";
UserEventStream::UserEventStream(TIsCountryLoaded const & fn)
: m_isCountryLoaded(fn)
, m_state(STATE_EMPTY)
, m_validTouchesCount(0)
, m_startDragOrg(m2::PointD::Zero())
{
}
@ -69,7 +71,7 @@ ScreenBase const & UserEventStream::ProcessEvents(bool & modelViewChange, bool &
switch (e.m_type)
{
case UserEvent::EVENT_SCALE:
breakAnim = Scale(e.m_scaleEvent.m_pxPoint, e.m_scaleEvent.m_factor, e.m_scaleEvent.m_isAnim);
breakAnim = SetScale(e.m_scaleEvent.m_pxPoint, e.m_scaleEvent.m_factor, e.m_scaleEvent.m_isAnim);
TouchCancel(m_touches);
break;
case UserEvent::EVENT_RESIZE:
@ -129,24 +131,22 @@ ScreenBase const & UserEventStream::GetCurrentScreen() const
return m_navigator.Screen();
}
void UserEventStream::SetTapListener(TTapDetectedFn const & tapCallback, TSingleTouchFilterFn const & filterFn)
bool UserEventStream::SetScale(m2::PointD const & pxScaleCenter, double factor, bool isAnim)
{
m_tapDetectedFn = tapCallback;
m_filterFn = filterFn;
}
m2::PointD scaleCenter = pxScaleCenter;
if (m_listener)
m_listener->CorrectScalePoint(scaleCenter);
bool UserEventStream::Scale(m2::PointD const & pxScaleCenter, double factor, bool isAnim)
{
if (isAnim)
{
m2::PointD glbCenter = m_navigator.PtoG(pxScaleCenter);
m2::PointD glbCenter = m_navigator.PtoG(scaleCenter);
m2::AnyRectD rect = GetTargetRect();
m2::RectD sizeRect = rect.GetLocalRect();
sizeRect.Scale(1.0 / factor);
return SetRect(m2::AnyRectD(glbCenter, rect.Angle(), sizeRect), true);
}
m_navigator.Scale(pxScaleCenter, factor);
m_navigator.Scale(scaleCenter, factor);
return true;
}
@ -396,6 +396,9 @@ void UserEventStream::BeginDrag(Touch const & t)
TEST_CALL(BEGIN_DRAG);
ASSERT(m_state == STATE_EMPTY, ());
m_state = STATE_DRAG;
m_startDragOrg = m_navigator.Screen().GetOrg();
if (m_listener)
m_listener->OnDragStarted();
m_navigator.StartDrag(t.m_location);
}
@ -411,6 +414,10 @@ void UserEventStream::EndDrag(Touch const & t)
TEST_CALL(END_DRAG);
ASSERT(m_state == STATE_DRAG, ());
m_state = STATE_EMPTY;
if (m_listener)
m_listener->OnDragEnded(m_navigator.GtoP(m_navigator.Screen().GetOrg()) - m_navigator.GtoP(m_startDragOrg));
m_startDragOrg = m2::PointD::Zero();
m_navigator.StopDrag(t.m_location);
}
@ -419,14 +426,35 @@ void UserEventStream::BeginScale(Touch const & t1, Touch const & t2)
TEST_CALL(BEGIN_SCALE);
ASSERT(m_state == STATE_EMPTY, ());
m_state = STATE_SCALE;
m_navigator.StartScale(t1.m_location, t2.m_location);
m2::PointD touch1 = t1.m_location;
m2::PointD touch2 = t2.m_location;
if (m_listener)
{
m_listener->OnScaleStarted();
m_listener->CorrectScalePoint(touch1, touch2);
}
m_navigator.StartScale(touch1, touch2);
}
void UserEventStream::Scale(Touch const & t1, Touch const & t2)
{
TEST_CALL(SCALE);
ASSERT(m_state == STATE_SCALE, ());
m_navigator.DoScale(t1.m_location, t2.m_location);
m2::PointD touch1 = t1.m_location;
m2::PointD touch2 = t2.m_location;
if (m_listener)
{
if (m_navigator.IsRotatingDuringScale())
m_listener->OnRotated();
m_listener->CorrectScalePoint(touch1, touch2);
}
m_navigator.DoScale(touch1, touch2);
}
void UserEventStream::EndScale(const Touch & t1, const Touch & t2)
@ -434,7 +462,17 @@ void UserEventStream::EndScale(const Touch & t1, const Touch & t2)
TEST_CALL(END_SCALE);
ASSERT(m_state == STATE_SCALE, ());
m_state = STATE_EMPTY;
m_navigator.StopScale(t1.m_location, t2.m_location);
m2::PointD touch1 = t1.m_location;
m2::PointD touch2 = t2.m_location;
if (m_listener)
{
m_listener->CorrectScalePoint(touch1, touch2);
m_listener->OnScaleEnded();
}
m_navigator.StopScale(touch1, touch2);
}
void UserEventStream::BeginTapDetector()
@ -452,7 +490,8 @@ void UserEventStream::DetectLongTap(Touch const & touch)
{
TEST_CALL(LONG_TAP_DETECTED);
m_state = STATE_EMPTY;
m_tapDetectedFn(touch.m_location, true);
if (m_listener)
m_listener->OnTap(touch.m_location, true);
}
}
@ -461,7 +500,8 @@ void UserEventStream::EndTapDetector(Touch const & touch)
TEST_CALL(SHORT_TAP_DETECTED);
ASSERT(m_state == STATE_TAP_DETECTION, ());
m_state = STATE_EMPTY;
m_tapDetectedFn(touch.m_location, false);
if (m_listener)
m_listener->OnTap(touch.m_location, false);
}
void UserEventStream::CancelTapDetector()
@ -475,7 +515,7 @@ bool UserEventStream::TryBeginFilter(Touch const & t)
{
TEST_CALL(TRY_FILTER);
ASSERT(m_state == STATE_EMPTY, ());
if (m_filterFn(t.m_location, TouchEvent::TOUCH_DOWN))
if (m_listener && m_listener->OnSingleTouchFiltrate(t.m_location, TouchEvent::TOUCH_DOWN))
{
m_state = STATE_FILTER;
return true;
@ -489,7 +529,8 @@ void UserEventStream::EndFilter(const Touch & t)
TEST_CALL(END_FILTER);
ASSERT(m_state == STATE_FILTER, ());
m_state = STATE_EMPTY;
m_filterFn(t.m_location, TouchEvent::TOUCH_UP);
if (m_listener)
m_listener->OnSingleTouchFiltrate(t.m_location, TouchEvent::TOUCH_UP);
}
void UserEventStream::CancelFilter(Touch const & t)
@ -497,7 +538,8 @@ void UserEventStream::CancelFilter(Touch const & t)
TEST_CALL(CANCEL_FILTER);
ASSERT(m_state == STATE_FILTER, ());
m_state = STATE_EMPTY;
m_filterFn(t.m_location, TouchEvent::TOUCH_CANCEL);
if (m_listener)
m_listener->OnSingleTouchFiltrate(t.m_location, TouchEvent::TOUCH_CANCEL);
}
}

View file

@ -3,6 +3,8 @@
#include "drape_frontend/navigator.hpp"
#include "drape_frontend/animation/model_view_animation.hpp"
#include "drape/pointers.hpp"
#include "geometry/point2d.hpp"
#include "geometry/rect2d.hpp"
#include "geometry/any_rect2d.hpp"
@ -150,14 +152,29 @@ struct UserEvent
class UserEventStream
{
public:
class Listener
{
public:
virtual void OnTap(m2::PointD const & pt, bool isLong) = 0;
virtual bool OnSingleTouchFiltrate(m2::PointD const & pt, TouchEvent::ETouchType type) = 0;
virtual void OnDragStarted() = 0;
virtual void OnDragEnded(m2::PointD const & distance) = 0;
virtual void OnScaleStarted() = 0;
virtual void OnRotated() = 0;
virtual void CorrectScalePoint(m2::PointD & pt) const = 0;
virtual void CorrectScalePoint(m2::PointD & pt1, m2::PointD & pt2) const = 0;
virtual void OnScaleEnded() = 0;
};
UserEventStream(TIsCountryLoaded const & fn);
void AddEvent(UserEvent const & event);
ScreenBase const & ProcessEvents(bool & modelViewChange, bool & viewportChanged);
ScreenBase const & GetCurrentScreen() const;
using TTapDetectedFn = function<void (m2::PointD const & pt, bool isLong)>;
using TSingleTouchFilterFn = function<bool (m2::PointD const &, TouchEvent::ETouchType type)>;
void SetTapListener(TTapDetectedFn const & tapCallback, TSingleTouchFilterFn const & filterFn);
m2::AnyRectD GetTargetRect() const;
void SetListener(ref_ptr<Listener> listener) { m_listener = listener; }
#ifdef DEBUG
static char const * BEGIN_DRAG;
@ -179,13 +196,12 @@ public:
#endif
private:
bool Scale(m2::PointD const & pxScaleCenter, double factor, bool isAnim);
bool SetScale(m2::PointD const & pxScaleCenter, double factor, bool isAnim);
bool SetCenter(m2::PointD const & center, int zoom, bool isAnim);
bool SetRect(m2::RectD rect, int zoom, bool applyRotation, bool isAnim);
bool SetRect(m2::AnyRectD const & rect, bool isAnim);
m2::AnyRectD GetCurrentRect() const;
m2::AnyRectD GetTargetRect() const;
bool ProcessTouch(TouchEvent const & touch);
@ -215,8 +231,6 @@ private:
private:
TIsCountryLoaded m_isCountryLoaded;
TTapDetectedFn m_tapDetectedFn;
TSingleTouchFilterFn m_filterFn;
list<UserEvent> m_events;
mutable mutex m_lock;
@ -236,10 +250,12 @@ private:
size_t m_validTouchesCount;
unique_ptr<ModelViewAnimation> m_animation;
ref_ptr<Listener> m_listener;
#ifdef DEBUG
TTestBridge m_testFn;
#endif
m2::PointD m_startDragOrg;
};
}