forked from organicmaps/organicmaps
[drape] kinetic scroll
This commit is contained in:
parent
ced224e1f7
commit
d6acc20e7e
7 changed files with 155 additions and 6 deletions
|
@ -5,7 +5,7 @@ namespace df
|
|||
|
||||
ModelViewAnimation::ModelViewAnimation(m2::AnyRectD const & startRect, m2::AnyRectD const & endRect,
|
||||
double aDuration, double mDuration, double sDuration)
|
||||
: BaseInterpolator(max(max(aDuration, mDuration), sDuration))
|
||||
: BaseModelViewAnimation(max(max(aDuration, mDuration), sDuration))
|
||||
, m_angleInterpolator(startRect.Angle().val(), endRect.Angle().val())
|
||||
, m_startZero(startRect.GlobalZero())
|
||||
, m_endZero(endRect.GlobalZero())
|
||||
|
|
|
@ -9,7 +9,16 @@
|
|||
namespace df
|
||||
{
|
||||
|
||||
class ModelViewAnimation : public BaseInterpolator
|
||||
class BaseModelViewAnimation : public BaseInterpolator
|
||||
{
|
||||
public:
|
||||
BaseModelViewAnimation(double duration, double delay = 0) : BaseInterpolator(duration, delay) {}
|
||||
|
||||
virtual m2::AnyRectD GetCurrentRect() const = 0;
|
||||
virtual m2::AnyRectD GetTargetRect() const = 0;
|
||||
};
|
||||
|
||||
class ModelViewAnimation : public BaseModelViewAnimation
|
||||
{
|
||||
public:
|
||||
static double GetRotateDuration(double startAngle, double endAngle);
|
||||
|
@ -21,8 +30,8 @@ public:
|
|||
/// sDuration - scaleDuration
|
||||
ModelViewAnimation(m2::AnyRectD const & startRect, m2::AnyRectD const & endRect,
|
||||
double aDuration, double mDuration, double sDuration);
|
||||
m2::AnyRectD GetCurrentRect() const;
|
||||
m2::AnyRectD GetTargetRect() const;
|
||||
m2::AnyRectD GetCurrentRect() const override;
|
||||
m2::AnyRectD GetTargetRect() const override;
|
||||
|
||||
private:
|
||||
m2::AnyRectD GetRect(double elapsedTime) const;
|
||||
|
|
|
@ -73,6 +73,7 @@ SOURCES += \
|
|||
user_marks_provider.cpp \
|
||||
viewport.cpp \
|
||||
visual_params.cpp \
|
||||
kinetic_scroller.cpp \
|
||||
|
||||
HEADERS += \
|
||||
animation/base_interpolator.hpp \
|
||||
|
@ -144,3 +145,4 @@ HEADERS += \
|
|||
user_marks_provider.hpp \
|
||||
viewport.hpp \
|
||||
visual_params.hpp \
|
||||
kinetic_scroller.hpp \
|
||||
|
|
92
drape_frontend/kinetic_scroller.cpp
Normal file
92
drape_frontend/kinetic_scroller.cpp
Normal file
|
@ -0,0 +1,92 @@
|
|||
#include "kinetic_scroller.hpp"
|
||||
#include "visual_params.hpp"
|
||||
#include "base/logging.hpp"
|
||||
|
||||
namespace df
|
||||
{
|
||||
|
||||
class KineticScrollAnimation : public BaseModelViewAnimation
|
||||
{
|
||||
public:
|
||||
KineticScrollAnimation(m2::AnyRectD const & startRect, m2::PointD const & direction, double duration)
|
||||
: BaseModelViewAnimation(duration)
|
||||
, m_targetCenter(startRect.GlobalCenter() + direction)
|
||||
, m_angle(startRect.Angle())
|
||||
, m_localRect(startRect.GetLocalRect())
|
||||
, m_direction(direction)
|
||||
{
|
||||
}
|
||||
|
||||
m2::AnyRectD GetCurrentRect() const override
|
||||
{
|
||||
m2::PointD center = -m_direction * exp(-GetT());
|
||||
m2::AnyRectD rect(m_targetCenter + center, m_angle, m_localRect);
|
||||
return rect;
|
||||
}
|
||||
|
||||
m2::AnyRectD GetTargetRect() const override
|
||||
{
|
||||
return GetCurrentRect();
|
||||
}
|
||||
|
||||
private:
|
||||
m2::PointD m_targetCenter;
|
||||
ang::AngleD m_angle;
|
||||
m2::RectD m_localRect;
|
||||
m2::PointD m_direction;
|
||||
};
|
||||
|
||||
KineticScroller::KineticScroller()
|
||||
: m_lastTimestamp(-1.0)
|
||||
, m_direction(m2::PointD::Zero())
|
||||
{
|
||||
}
|
||||
|
||||
void KineticScroller::InitGrab(ScreenBase const & modelView, double timeStamp)
|
||||
{
|
||||
ASSERT_LESS(m_lastTimestamp, 0.0, ());
|
||||
m_lastTimestamp = timeStamp;
|
||||
m_lastRect = modelView.GlobalRect();
|
||||
}
|
||||
|
||||
void KineticScroller::GrabViewRect(ScreenBase const & modelView, double timeStamp)
|
||||
{
|
||||
ASSERT_GREATER(m_lastTimestamp, 0.0, ());
|
||||
ASSERT_GREATER(timeStamp, m_lastTimestamp, ());
|
||||
double elapsed = timeStamp - m_lastTimestamp;
|
||||
|
||||
m2::PointD lastCenter = m_lastRect.GlobalCenter();
|
||||
m2::PointD currentCenter = modelView.GlobalRect().GlobalCenter();
|
||||
double pxDeltaLength = (modelView.GtoP(currentCenter) - modelView.GtoP(lastCenter)).Length();
|
||||
m2::PointD delta = (currentCenter - lastCenter);
|
||||
if (!delta.IsAlmostZero())
|
||||
delta = delta.Normalize();
|
||||
|
||||
double v = pxDeltaLength / elapsed;
|
||||
m_direction = delta * 0.8 * v + m_direction * 0.2;
|
||||
|
||||
m_lastTimestamp = timeStamp;
|
||||
m_lastRect = modelView.GlobalRect();
|
||||
}
|
||||
|
||||
void KineticScroller::CancelGrab()
|
||||
{
|
||||
m_lastTimestamp = -1;
|
||||
m_direction = m2::PointD::Zero();
|
||||
}
|
||||
|
||||
unique_ptr<BaseModelViewAnimation> KineticScroller::CreateKineticAnimation(ScreenBase const & modelView)
|
||||
{
|
||||
static double VELOCITY_THRESHOLD = 10.0 * VisualParams::Instance().GetVisualScale();
|
||||
if (m_direction.Length() < VELOCITY_THRESHOLD)
|
||||
return unique_ptr<BaseModelViewAnimation>();
|
||||
|
||||
double const KINETIC_DURATION = 0.375;
|
||||
m2::PointD center = m_lastRect.GlobalCenter();
|
||||
double glbLength = 0.5 * (modelView.PtoG(modelView.GtoP(center) + m_direction) - center).Length();
|
||||
return unique_ptr<BaseModelViewAnimation>(new KineticScrollAnimation(m_lastRect,
|
||||
m_direction.Normalize() * glbLength,
|
||||
KINETIC_DURATION));
|
||||
}
|
||||
|
||||
} // namespace df
|
26
drape_frontend/kinetic_scroller.hpp
Normal file
26
drape_frontend/kinetic_scroller.hpp
Normal file
|
@ -0,0 +1,26 @@
|
|||
#pragma once
|
||||
|
||||
#include "animation/model_view_animation.hpp"
|
||||
|
||||
#include "geometry/any_rect2d.hpp"
|
||||
|
||||
namespace df
|
||||
{
|
||||
|
||||
class KineticScroller
|
||||
{
|
||||
public:
|
||||
KineticScroller();
|
||||
|
||||
void InitGrab(ScreenBase const & modelView, double timeStamp);
|
||||
void GrabViewRect(ScreenBase const & modelView, double timeStamp);
|
||||
void CancelGrab();
|
||||
unique_ptr<BaseModelViewAnimation> CreateKineticAnimation(ScreenBase const & modelView);
|
||||
|
||||
private:
|
||||
double m_lastTimestamp;
|
||||
m2::AnyRectD m_lastRect;
|
||||
m2::PointD m_direction;
|
||||
};
|
||||
|
||||
} // namespace df
|
|
@ -233,15 +233,30 @@ bool UserEventStream::ProcessTouch(TouchEvent const & touch)
|
|||
{
|
||||
case TouchEvent::TOUCH_DOWN:
|
||||
isMapTouch |= TouchDown(touches);
|
||||
if (isMapTouch)
|
||||
m_scroller.InitGrab(m_navigator.Screen(), touch.m_timeStamp);
|
||||
break;
|
||||
case TouchEvent::TOUCH_MOVE:
|
||||
isMapTouch |= TouchMove(touches);
|
||||
if (isMapTouch)
|
||||
m_scroller.GrabViewRect(m_navigator.Screen(), touch.m_timeStamp);
|
||||
break;
|
||||
case TouchEvent::TOUCH_CANCEL:
|
||||
isMapTouch |= TouchCancel(touches);
|
||||
if (isMapTouch)
|
||||
m_scroller.CancelGrab();
|
||||
break;
|
||||
case TouchEvent::TOUCH_UP:
|
||||
isMapTouch |= TouchUp(touches);
|
||||
{
|
||||
isMapTouch |= TouchUp(touches);
|
||||
if (isMapTouch)
|
||||
{
|
||||
m_scroller.GrabViewRect(m_navigator.Screen(), touch.m_timeStamp);
|
||||
m_animation = m_scroller.CreateKineticAnimation(m_navigator.Screen());
|
||||
isMapTouch = false;
|
||||
m_scroller.CancelGrab();
|
||||
}
|
||||
}
|
||||
break;
|
||||
default:
|
||||
ASSERT(false, ());
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
#pragma once
|
||||
|
||||
#include "drape_frontend/kinetic_scroller.hpp"
|
||||
#include "drape_frontend/navigator.hpp"
|
||||
#include "drape_frontend/animation/model_view_animation.hpp"
|
||||
|
||||
|
@ -30,6 +31,7 @@ struct TouchEvent
|
|||
{
|
||||
TouchEvent()
|
||||
: m_type(TOUCH_CANCEL)
|
||||
, m_timeStamp(my::Timer::LocalTime())
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -43,6 +45,7 @@ struct TouchEvent
|
|||
|
||||
ETouchType m_type;
|
||||
array<Touch, 2> m_touches;
|
||||
double m_timeStamp; // seconds
|
||||
};
|
||||
|
||||
struct ScaleEvent
|
||||
|
@ -257,13 +260,15 @@ private:
|
|||
array<Touch, 2> m_touches;
|
||||
size_t m_validTouchesCount;
|
||||
|
||||
unique_ptr<ModelViewAnimation> m_animation;
|
||||
unique_ptr<BaseModelViewAnimation> m_animation;
|
||||
ref_ptr<Listener> m_listener;
|
||||
|
||||
#ifdef DEBUG
|
||||
TTestBridge m_testFn;
|
||||
#endif
|
||||
m2::PointD m_startDragOrg;
|
||||
|
||||
KineticScroller m_scroller;
|
||||
};
|
||||
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue