blob: 7c6e191bb6dae9bcf34e38059a153844e7399e37 [file] [log] [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "cc/animation/scroll_offset_animation_curve.h"
#include <utility>
#include <vector>
#include "base/test/scoped_feature_list.h"
#include "base/time/time.h"
#include "cc/animation/scroll_offset_animation_curve_factory.h"
#include "cc/base/features.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/test/geometry_util.h"
using DurationBehavior = cc::ScrollOffsetAnimationCurve::DurationBehavior;
const double kConstantDuration = 9.0;
const double kDurationDivisor = 60.0;
const double kInverseDeltaMaxDuration = 12.0;
namespace cc {
class ScrollOffsetAnimationCurveTest : public ::testing::Test {
public:
ScrollOffsetAnimationCurveTest() {
std::vector<base::test::FeatureRefAndParams> enabled_features = {
{
features::kProgrammaticScrollAnimationOverride,
{
{"cubic_bezier_x1", "0.4"}, //
{"cubic_bezier_y1", "0.0"}, //
{"cubic_bezier_x2", "0.0"}, //
{"cubic_bezier_y2", "1.0"}, //
{"max_animation_duration", "1.5s"}, //
} //
} //
};
scoped_feature_list_.InitWithFeaturesAndParameters(
/*enabled_features=*/
std::move(enabled_features),
/*disabled_features=*/{});
}
~ScrollOffsetAnimationCurveTest() override = default;
private:
base::test::ScopedFeatureList scoped_feature_list_;
};
TEST_F(ScrollOffsetAnimationCurveTest, DeltaBasedDuration) {
gfx::PointF target_value(100.f, 200.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
target_value));
curve->SetInitialValue(target_value);
EXPECT_DOUBLE_EQ(0.0, curve->Duration().InSecondsF());
// x decreases, y stays the same.
curve->SetInitialValue(gfx::PointF(136.f, 200.f));
EXPECT_DOUBLE_EQ(0.1, curve->Duration().InSecondsF());
// x increases, y stays the same.
curve->SetInitialValue(gfx::PointF(19.f, 200.f));
EXPECT_DOUBLE_EQ(0.15, curve->Duration().InSecondsF());
// x stays the same, y decreases.
curve->SetInitialValue(gfx::PointF(100.f, 344.f));
EXPECT_DOUBLE_EQ(0.2, curve->Duration().InSecondsF());
// x stays the same, y increases.
curve->SetInitialValue(gfx::PointF(100.f, 191.f));
EXPECT_DOUBLE_EQ(0.05, curve->Duration().InSecondsF());
// x decreases, y decreases.
curve->SetInitialValue(gfx::PointF(32500.f, 500.f));
EXPECT_DOUBLE_EQ(1.5, curve->Duration().InSecondsF());
// x decreases, y increases.
curve->SetInitialValue(gfx::PointF(150.f, 119.f));
EXPECT_DOUBLE_EQ(0.15, curve->Duration().InSecondsF());
// x increases, y decreases.
curve->SetInitialValue(gfx::PointF(0.f, 14600.f));
EXPECT_DOUBLE_EQ(1.5, curve->Duration().InSecondsF());
// x increases, y increases.
curve->SetInitialValue(gfx::PointF(95.f, 191.f));
EXPECT_DOUBLE_EQ(0.05, curve->Duration().InSecondsF());
}
TEST_F(ScrollOffsetAnimationCurveTest, GetValue) {
gfx::PointF initial_value(2.f, 40.f);
gfx::PointF target_value(10.f, 20.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
target_value));
curve->SetInitialValue(initial_value);
base::TimeDelta duration = curve->Duration();
EXPECT_GT(curve->Duration().InSecondsF(), 0);
EXPECT_LT(curve->Duration().InSecondsF(), 0.1);
EXPECT_EQ(duration, curve->Duration());
EXPECT_POINTF_EQ(initial_value, curve->GetValue(base::Seconds(-1.0)));
EXPECT_POINTF_EQ(initial_value, curve->GetValue(base::TimeDelta()));
EXPECT_POINTF_NEAR(gfx::PointF(8.8921f, 22.7697f),
curve->GetValue(duration * 0.5f), 0.00025);
EXPECT_POINTF_EQ(target_value, curve->GetValue(duration));
EXPECT_POINTF_EQ(target_value,
curve->GetValue(duration + base::Seconds(1.0)));
// Verify that GetValue takes the timing function into account.
gfx::PointF value = curve->GetValue(duration * 0.25f);
EXPECT_NEAR(5.2583f, value.x(), 0.0002f);
EXPECT_NEAR(31.8541f, value.y(), 0.0002f);
}
TEST_F(ScrollOffsetAnimationCurveTest, MaxAnimationDuration) {
gfx::PointF initial_value(2.f, 40.f);
gfx::PointF target_value(200000.f, 400000.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
target_value));
curve->SetInitialValue(initial_value);
EXPECT_EQ(curve->Duration().InSecondsF(), 1.5);
}
// Verify that a clone behaves exactly like the original.
TEST_F(ScrollOffsetAnimationCurveTest, Clone) {
gfx::PointF initial_value(2.f, 40.f);
gfx::PointF target_value(10.f, 20.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
target_value));
curve->SetInitialValue(initial_value);
base::TimeDelta duration = curve->Duration();
std::unique_ptr<gfx::AnimationCurve> clone(curve->Clone());
EXPECT_EQ(duration, clone->Duration());
ScrollOffsetAnimationCurve* cloned_curve =
ScrollOffsetAnimationCurve::ToScrollOffsetAnimationCurve(clone.get());
EXPECT_POINTF_EQ(initial_value, cloned_curve->GetValue(base::Seconds(-1.0)));
EXPECT_POINTF_EQ(initial_value, cloned_curve->GetValue(base::TimeDelta()));
EXPECT_POINTF_NEAR(gfx::PointF(8.89213f, 22.76970f),
cloned_curve->GetValue(duration * 0.5f), 0.00025);
EXPECT_POINTF_EQ(target_value, cloned_curve->GetValue(duration));
EXPECT_POINTF_EQ(target_value,
cloned_curve->GetValue(duration + base::Seconds(1.f)));
// Verify that the timing function was cloned correctly.
gfx::PointF value = cloned_curve->GetValue(duration * 0.25f);
EXPECT_NEAR(5.2583f, value.x(), 0.0002f);
EXPECT_NEAR(31.8541f, value.y(), 0.0002f);
}
TEST_F(ScrollOffsetAnimationCurveTest, EaseInOutUpdateTarget) {
gfx::PointF initial_value(0.f, 0.f);
gfx::PointF target_value(0.f, 3600.f);
double duration = kConstantDuration / kDurationDivisor;
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
target_value, DurationBehavior::kConstant));
curve->SetInitialValue(initial_value);
EXPECT_NEAR(duration, curve->Duration().InSecondsF(), 0.0002f);
EXPECT_NEAR(3101.4792f, curve->GetValue(base::Seconds(duration / 2.0)).y(),
0.0002f);
EXPECT_NEAR(3600.0, curve->GetValue(base::Seconds(duration)).y(), 0.0002f);
curve->UpdateTarget(base::Seconds(duration / 2), gfx::PointF(0.0, 9900.0));
EXPECT_NEAR(duration * 1.5, curve->Duration().InSecondsF(), 0.0002f);
EXPECT_NEAR(3101.4792f, curve->GetValue(base::Seconds(duration / 2.0)).y(),
0.0002f);
EXPECT_NEAR(9097.8, curve->GetValue(base::Seconds(duration)).y(), 0.1f);
EXPECT_NEAR(9900.0, curve->GetValue(base::Seconds(duration * 1.5)).y(),
0.0002f);
curve->UpdateTarget(base::Seconds(duration), gfx::PointF(0.0, 7200.0));
// A closer target at high velocity reduces the duration.
EXPECT_NEAR(duration * 2, curve->Duration().InSecondsF(), 0.0002f);
EXPECT_NEAR(9097.8, curve->GetValue(base::Seconds(duration)).y(), 0.1f);
EXPECT_NEAR(7200.0, curve->GetValue(base::Seconds(duration * 2)).y(),
0.0002f);
}
TEST_F(ScrollOffsetAnimationCurveTest, InverseDeltaDuration) {
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
gfx::PointF(0.f, 100.f), DurationBehavior::kInverseDelta));
curve->SetInitialValue(gfx::PointF());
double smallDeltaDuration = curve->Duration().InSecondsF();
curve->UpdateTarget(base::Seconds(0.01f), gfx::PointF(0.f, 300.f));
double mediumDeltaDuration = curve->Duration().InSecondsF();
curve->UpdateTarget(base::Seconds(0.01f), gfx::PointF(0.f, 500.f));
double largeDeltaDuration = curve->Duration().InSecondsF();
EXPECT_GT(smallDeltaDuration, mediumDeltaDuration);
EXPECT_GT(mediumDeltaDuration, largeDeltaDuration);
curve->UpdateTarget(base::Seconds(0.01f), gfx::PointF(0.f, 5000.f));
EXPECT_EQ(largeDeltaDuration, curve->Duration().InSecondsF());
}
TEST_F(ScrollOffsetAnimationCurveTest, LinearAnimation) {
// Testing autoscroll downwards for a scroller of length 1000px.
gfx::PointF current_offset(0.f, 0.f);
gfx::PointF target_offset(0.f, 1000.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateLinearAnimationForTesting(
target_offset));
const float autoscroll_velocity = 800.f; // pixels per second.
curve->SetInitialValue(current_offset, base::TimeDelta(),
autoscroll_velocity);
EXPECT_FLOAT_EQ(1.25f, curve->Duration().InSecondsF());
// Test scrolling down from half way.
current_offset = gfx::PointF(0.f, 500.f);
curve->SetInitialValue(current_offset, base::TimeDelta(),
autoscroll_velocity);
EXPECT_FLOAT_EQ(0.625f, curve->Duration().InSecondsF());
// Test scrolling down when max_offset is reached.
current_offset = gfx::PointF(0.f, 1000.f);
curve->SetInitialValue(current_offset, base::TimeDelta(),
autoscroll_velocity);
EXPECT_FLOAT_EQ(0.f, curve->Duration().InSecondsF());
}
TEST_F(ScrollOffsetAnimationCurveTest, CurveWithDelay) {
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
gfx::PointF(0.f, 100.f), DurationBehavior::kInverseDelta));
double duration_in_seconds = kInverseDeltaMaxDuration / kDurationDivisor;
double delay_in_seconds = 0.02;
double curve_duration = duration_in_seconds - delay_in_seconds;
curve->SetInitialValue(gfx::PointF(), base::Seconds(delay_in_seconds));
EXPECT_NEAR(curve_duration, curve->Duration().InSecondsF(), 0.0002f);
curve->UpdateTarget(base::Seconds(0.01f), gfx::PointF(0.f, 500.f));
EXPECT_GT(curve_duration, curve->Duration().InSecondsF());
EXPECT_EQ(gfx::PointF(0.f, 500.f), curve->target_value());
}
TEST_F(ScrollOffsetAnimationCurveTest, CurveWithLargeDelay) {
DurationBehavior duration_hint = DurationBehavior::kInverseDelta;
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
gfx::PointF(0.f, 100.f), duration_hint));
curve->SetInitialValue(gfx::PointF(), base::Seconds(0.2));
EXPECT_EQ(0.f, curve->Duration().InSecondsF());
// Re-targeting when animation duration is 0.
curve->UpdateTarget(base::Seconds(-0.01), gfx::PointF(0.f, 300.f));
double duration =
curve
->EaseInOutSegmentDuration(gfx::Vector2dF(0.f, 200.f), duration_hint,
base::Seconds(0.01))
.InSecondsF();
EXPECT_EQ(duration, curve->Duration().InSecondsF());
// Re-targeting before last_retarget_, the difference should be accounted for
// in duration.
curve->UpdateTarget(base::Seconds(-0.01), gfx::PointF(0.f, 500.f));
duration = curve
->EaseInOutSegmentDuration(gfx::Vector2dF(0.f, 500.f),
duration_hint, base::Seconds(0.01))
.InSecondsF();
EXPECT_EQ(duration, curve->Duration().InSecondsF());
EXPECT_POINTF_EQ(gfx::PointF(0.f, 500.f),
curve->GetValue(base::Seconds(1.0)));
}
// This test verifies that if the last segment duration is zero, ::UpdateTarget
// simply updates the total animation duration see crbug.com/645317.
TEST_F(ScrollOffsetAnimationCurveTest, UpdateTargetZeroLastSegmentDuration) {
DurationBehavior duration_hint = DurationBehavior::kInverseDelta;
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateEaseInOutAnimationForTesting(
gfx::PointF(0.f, 100.f), duration_hint));
double duration_in_seconds = kInverseDeltaMaxDuration / kDurationDivisor;
double delay_in_seconds = 0.02;
double curve_duration = duration_in_seconds - delay_in_seconds;
curve->SetInitialValue(gfx::PointF(), base::Seconds(delay_in_seconds));
EXPECT_NEAR(curve_duration, curve->Duration().InSecondsF(), 0.0002f);
// Re-target 1, this should set last_retarget_ to 0.05.
gfx::Vector2dF new_delta =
gfx::PointF(0.f, 200.f) - curve->GetValue(base::Seconds(0.05));
double expected_duration =
curve
->EaseInOutSegmentDuration(new_delta, duration_hint,
base::TimeDelta())
.InSecondsF() +
0.05;
curve->UpdateTarget(base::Seconds(0.05), gfx::PointF(0.f, 200.f));
EXPECT_NEAR(expected_duration, curve->Duration().InSecondsF(), 0.0002f);
// Re-target 2, this should set total_animation_duration to t, which is
// last_retarget_. This is what would cause the DCHECK failure in
// crbug.com/645317.
curve->UpdateTarget(base::Seconds(-0.145), gfx::PointF(0.f, 300.f));
EXPECT_NEAR(0.05, curve->Duration().InSecondsF(), 0.0002f);
// Re-target 3, this should set total_animation_duration based on new_delta.
new_delta = gfx::PointF(0.f, 500.f) - curve->GetValue(base::Seconds(0.05));
expected_duration = curve
->EaseInOutSegmentDuration(new_delta, duration_hint,
base::Seconds(0.15))
.InSecondsF();
curve->UpdateTarget(base::Seconds(-0.1), gfx::PointF(0.f, 500.f));
EXPECT_NEAR(expected_duration, curve->Duration().InSecondsF(), 0.0002f);
}
class ScrollOffsetAnimationCurveTestWithProgrammaticOverride
: public ::testing::Test {
public:
ScrollOffsetAnimationCurveTestWithProgrammaticOverride() {
std::vector<base::test::FeatureRefAndParams> enabled_features = {
{
features::kProgrammaticScrollAnimationOverride,
{
{"cubic_bezier_x1", "0.42"}, //
{"cubic_bezier_y1", "0.0"}, //
{"cubic_bezier_x2", "0.58"}, //
{"cubic_bezier_y2", "1.0"}, //
{"max_animation_duration", "0.7s"}, //
} //
} //
};
scoped_feature_list_.InitWithFeaturesAndParameters(
/*enabled_features=*/
std::move(enabled_features),
/*disabled_features=*/{});
}
~ScrollOffsetAnimationCurveTestWithProgrammaticOverride() override = default;
private:
base::test::ScopedFeatureList scoped_feature_list_;
};
TEST_F(ScrollOffsetAnimationCurveTestWithProgrammaticOverride, GetValue) {
gfx::PointF initial_value(2.f, 40.f);
gfx::PointF target_value(10.f, 20.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateAnimation(
target_value, ScrollOffsetAnimationCurve::ScrollType::kProgrammatic));
curve->SetInitialValue(initial_value);
base::TimeDelta duration = curve->Duration();
EXPECT_GT(curve->Duration().InSecondsF(), 0);
EXPECT_LT(curve->Duration().InSecondsF(), 0.1);
EXPECT_EQ(duration, curve->Duration());
EXPECT_POINTF_EQ(initial_value, curve->GetValue(base::Seconds(-1.0)));
EXPECT_POINTF_EQ(initial_value, curve->GetValue(base::TimeDelta()));
EXPECT_POINTF_NEAR(gfx::PointF(6.f, 30.f), curve->GetValue(duration * 0.5f),
0.001f);
EXPECT_POINTF_EQ(target_value, curve->GetValue(duration));
EXPECT_POINTF_EQ(target_value,
curve->GetValue(duration + base::Seconds(1.0)));
// Verify that GetValue takes the timing function into account.
gfx::PointF value = curve->GetValue(duration * 0.25f);
EXPECT_NEAR(3.033f, value.x(), 0.001f);
EXPECT_NEAR(37.416f, value.y(), 0.001f);
}
TEST_F(ScrollOffsetAnimationCurveTestWithProgrammaticOverride,
MaxAnimationDuration) {
gfx::PointF initial_value(2.f, 40.f);
gfx::PointF target_value(200000.f, 400000.f);
std::unique_ptr<ScrollOffsetAnimationCurve> curve(
ScrollOffsetAnimationCurveFactory::CreateAnimation(
target_value, ScrollOffsetAnimationCurve::ScrollType::kProgrammatic));
curve->SetInitialValue(initial_value);
EXPECT_EQ(curve->Duration().InSecondsF(), 0.7);
}
} // namespace cc