#include <stddef.h>
#include <memory>
#include "base/notreached.h"
#include "base/time/time.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/events/test/motion_event_test_utils.h"
#include "ui/events/velocity_tracker/motion_event.h"
#include "ui/events/velocity_tracker/velocity_tracker_state.h"
#include "ui/gfx/geometry/point_f.h"
#include "ui/gfx/geometry/vector2d_f.h"
using base::TimeTicks;
using ui::test::MockMotionEvent;
namespace ui {
namespace {
const base::TimeDelta kTenMillis = base::Milliseconds(10);
const base::TimeDelta kOneSecond = base::Seconds(1);
const float kEpsilson = .01f;
const char* GetStrategyName(VelocityTracker::Strategy strategy) {
switch (strategy) {
case VelocityTracker::LSQ1: return "LSQ1";
case VelocityTracker::LSQ2: return "LSQ2";
case VelocityTracker::LSQ2_RESTRICTED: return "LSQ2_RESTRICTED";
case VelocityTracker::LSQ3: return "LSQ3";
case VelocityTracker::WLSQ2_DELTA: return "WLSQ2_DELTA";
case VelocityTracker::WLSQ2_CENTRAL: return "WLSQ2_CENTRAL";
case VelocityTracker::WLSQ2_RECENT: return "WLSQ2_RECENT";
case VelocityTracker::INT1: return "INT1";
case VelocityTracker::INT2: return "INT2";
}
NOTREACHED() << "Invalid strategy";
}
}
class VelocityTrackerTest : public testing::Test {
public:
VelocityTrackerTest() {}
~VelocityTrackerTest() override {}
protected:
static MockMotionEvent Sample(MotionEvent::Action action,
const gfx::PointF& p0,
TimeTicks t0,
const gfx::Vector2dF& v,
base::TimeDelta dt) {
const gfx::PointF p = p0 + ScaleVector2d(v, dt.InSecondsF());
return MockMotionEvent(action, t0 + dt, p.x(), p.y());
}
static void ApplyMovementSequence(VelocityTrackerState* state,
const gfx::PointF& p0,
const gfx::Vector2dF& v,
TimeTicks t0,
base::TimeDelta t,
size_t samples) {
EXPECT_TRUE(samples);
if (!samples)
return;
const base::TimeDelta dt = t / samples;
state->AddMovement(Sample(MotionEvent::Action::DOWN, p0, t0, v, dt * 0));
ApplyMovement(state, p0, v, t0, t, samples);
state->AddMovement(Sample(MotionEvent::Action::UP, p0, t0, v, t));
}
static void ApplyMovement(VelocityTrackerState* state,
const gfx::PointF& p0,
const gfx::Vector2dF& v,
TimeTicks t0,
base::TimeDelta t,
size_t samples) {
EXPECT_TRUE(samples);
if (!samples)
return;
const base::TimeDelta dt = t / samples;
for (size_t i = 0; i < samples; ++i)
state->AddMovement(Sample(MotionEvent::Action::MOVE, p0, t0, v, dt * i));
}
};
TEST_F(VelocityTrackerTest, Basic) {
const gfx::PointF p0(0, 0);
const gfx::Vector2dF v(0, 500);
const size_t samples = 60;
for (int i = 0; i <= VelocityTracker::STRATEGY_MAX; ++i) {
VelocityTracker::Strategy strategy =
static_cast<VelocityTracker::Strategy>(i);
SCOPED_TRACE(GetStrategyName(strategy));
VelocityTrackerState state(strategy);
EXPECT_EQ(0, state.GetXVelocity(0));
EXPECT_EQ(0, state.GetYVelocity(0));
ApplyMovementSequence(&state, p0, v, TimeTicks::Now(), kOneSecond, samples);
state.ComputeCurrentVelocity(1000, 20000);
EXPECT_NEAR(v.x(), state.GetXVelocity(0), kEpsilson * v.x());
EXPECT_NEAR(v.y(), state.GetYVelocity(0), kEpsilson * v.y());
EXPECT_NEAR(v.x(), state.GetXVelocity(-1), kEpsilson * v.x());
EXPECT_NEAR(v.y(), state.GetYVelocity(-1), kEpsilson * v.y());
EXPECT_EQ(0, state.GetXVelocity(1));
EXPECT_EQ(0, state.GetYVelocity(1));
EXPECT_EQ(0, state.GetXVelocity(7));
EXPECT_EQ(0, state.GetYVelocity(7));
}
}
TEST_F(VelocityTrackerTest, MaxVelocity) {
const gfx::PointF p0(0, 0);
const gfx::Vector2dF v(-50000, 50000);
const size_t samples = 3;
const base::TimeDelta dt = kTenMillis * 2;
VelocityTrackerState state(VelocityTracker::Strategy::LSQ2);
ApplyMovementSequence(&state, p0, v, TimeTicks::Now(), dt, samples);
state.ComputeCurrentVelocity(1000, 100);
EXPECT_NEAR(-100, state.GetXVelocity(0), kEpsilson);
EXPECT_NEAR(100, state.GetYVelocity(0), kEpsilson);
state.ComputeCurrentVelocity(1000, 1000);
EXPECT_NEAR(-1000, state.GetXVelocity(0), kEpsilson);
EXPECT_NEAR(1000, state.GetYVelocity(0), kEpsilson);
}
TEST_F(VelocityTrackerTest, VaryingVelocity) {
const gfx::PointF p0(0, 0);
const gfx::Vector2dF vFast(0, 500);
const gfx::Vector2dF vSlow = ScaleVector2d(vFast, 0.5f);
const size_t samples = 12;
for (int i = 0; i <= VelocityTracker::STRATEGY_MAX; ++i) {
VelocityTracker::Strategy strategy =
static_cast<VelocityTracker::Strategy>(i);
SCOPED_TRACE(GetStrategyName(strategy));
VelocityTrackerState state(strategy);
base::TimeTicks t0 = base::TimeTicks::Now();
base::TimeDelta dt = kTenMillis * 10;
state.AddMovement(
Sample(MotionEvent::Action::DOWN, p0, t0, vFast, base::TimeDelta()));
gfx::PointF pCurr = p0;
base::TimeTicks tCurr = t0;
ApplyMovement(&state, pCurr, vFast, tCurr, dt, samples);
state.ComputeCurrentVelocity(1000, 20000);
float vOldY = state.GetYVelocity(0);
pCurr += ScaleVector2d(vFast, dt.InSecondsF());
tCurr += dt;
ApplyMovement(&state, pCurr, vSlow, tCurr, dt, samples);
state.ComputeCurrentVelocity(1000, 20000);
float vCurrentY = state.GetYVelocity(0);
EXPECT_GT(vFast.y(), vCurrentY);
EXPECT_GT(vOldY, vCurrentY);
vOldY = vCurrentY;
pCurr += ScaleVector2d(vSlow, dt.InSecondsF());
tCurr += dt;
ApplyMovement(&state, pCurr, vFast, tCurr, dt, samples);
state.ComputeCurrentVelocity(1000, 20000);
vCurrentY = state.GetYVelocity(0);
EXPECT_LT(vSlow.y(), vCurrentY);
EXPECT_LT(vOldY, vCurrentY);
}
}
TEST_F(VelocityTrackerTest, DelayedActionUp) {
const gfx::PointF p0(0, 0);
const gfx::Vector2dF v(-50000, 50000);
const size_t samples = 10;
const base::TimeTicks t0 = base::TimeTicks::Now();
const base::TimeDelta dt = kTenMillis * 2;
VelocityTrackerState state(VelocityTracker::Strategy::LSQ2);
state.AddMovement(
Sample(MotionEvent::Action::DOWN, p0, t0, v, base::TimeDelta()));
ApplyMovement(&state, p0, v, t0, dt, samples);
state.ComputeCurrentVelocity(1000, 1000);
EXPECT_NEAR(-1000, state.GetXVelocity(0), kEpsilson);
EXPECT_NEAR(1000, state.GetYVelocity(0), kEpsilson);
const gfx::PointF p1 = p0 + ScaleVector2d(v, dt.InSecondsF());
const base::TimeTicks t1 = t0 + dt + kTenMillis * 10;
state.AddMovement(
Sample(MotionEvent::Action::UP, p1, t1, v, base::TimeDelta()));
state.ComputeCurrentVelocity(1000, 1000);
EXPECT_EQ(0.f, state.GetXVelocity(0));
EXPECT_EQ(0.f, state.GetYVelocity(0));
}
TEST_F(VelocityTrackerTest, NoDirectionReversal) {
VelocityTrackerState state_unrestricted(VelocityTracker::LSQ2);
VelocityTrackerState state_restricted(VelocityTracker::LSQ2_RESTRICTED);
const base::TimeTicks t0 = base::TimeTicks::Now();
const base::TimeDelta dt = base::Milliseconds(1);
const size_t samples = 60;
gfx::PointF p(0, 0);
MockMotionEvent m1(MotionEvent::Action::DOWN, t0, p.x(), p.y());
state_unrestricted.AddMovement(m1);
state_restricted.AddMovement(m1);
for (size_t i = 0; i < samples; ++i) {
if (i < 50)
p.set_y(p.y() + 10);
MockMotionEvent mi(MotionEvent::Action::MOVE, t0 + dt * i, p.x(), p.y());
state_unrestricted.AddMovement(mi);
state_restricted.AddMovement(mi);
}
state_restricted.ComputeCurrentVelocity(1000, 20000);
EXPECT_EQ(0, state_restricted.GetXVelocity(0));
EXPECT_EQ(0, state_restricted.GetYVelocity(0));
state_unrestricted.ComputeCurrentVelocity(1000, 20000);
EXPECT_EQ(0, state_unrestricted.GetXVelocity(0));
EXPECT_GT(0, state_unrestricted.GetYVelocity(0));
}
class MockEvent : public MockMotionEvent {
public:
MockEvent(Action action, base::TimeTicks time, float x, float y)
: MockMotionEvent(action, time, x, y) {}
MOCK_METHOD(base::TimeTicks,
GetHistoricalEventTime,
(size_t),
(const, override));
MOCK_METHOD(base::TimeTicks, GetLatestEventTime, (), (const, override));
};
TEST_F(VelocityTrackerTest, UsesLatestEventTimeWhenNotResampled) {
VelocityTrackerState state(VelocityTracker::LSQ2);
base::TimeTicks now = base::TimeTicks::Now();
MockEvent m1(MotionEvent::Action::MOVE, now, 0, 0);
m1.PushHistoricalEvent(std::make_unique<MockEvent>(
MotionEvent::Action::MOVE, base::TimeTicks::Now() - base::Milliseconds(4),
-1, -1));
m1.SetIsLatestEventTimeResampled(false);
EXPECT_CALL(m1, GetHistoricalEventTime(0)).Times(1);
EXPECT_CALL(m1, GetLatestEventTime()).Times(1);
state.AddMovement(m1);
}
TEST_F(VelocityTrackerTest, DoesNotUseLatestEventTimeWhenResampled) {
VelocityTrackerState state(VelocityTracker::LSQ2);
base::TimeTicks now = base::TimeTicks::Now();
MockEvent m1(MotionEvent::Action::MOVE, now, 0, 0);
m1.PushHistoricalEvent(std::make_unique<MockEvent>(
MotionEvent::Action::MOVE, base::TimeTicks::Now() - base::Milliseconds(4),
-1, -1));
m1.SetIsLatestEventTimeResampled(true);
EXPECT_CALL(m1, GetHistoricalEventTime(0)).Times(1);
EXPECT_CALL(m1, GetLatestEventTime()).Times(0);
state.AddMovement(m1);
}
}