#include "cc/animation/scroll_timeline.h"
#include <limits>
#include <vector>
#include "cc/trees/property_tree.h"
#include "cc/trees/scroll_node.h"
#include "cc/trees/transform_node.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/point_f.h"
namespace cc {
namespace {
static constexpr double time_error_ms = 0.001;
#define EXPECT_SCROLL_TIMELINE_TIME_NEAR(expected, value) \
EXPECT_NEAR(expected, ToDouble(value), time_error_ms)
#define EXPECT_SCROLL_TIMELINE_BEFORE_START(value) \
EXPECT_LT(ToDouble(value), 0);
#define EXPECT_SCROLL_TIMELINE_AFTER_END(value) \
EXPECT_GT(ToDouble(value), ScrollTimeline::kScrollTimelineDurationMs);
void SetScrollOffset(PropertyTrees* property_trees,
ElementId scroller_id,
gfx::PointF offset) {
property_trees->scroll_tree_mutable().SetScrollOffset(scroller_id, offset);
TransformNode* transform_node =
property_trees->transform_tree_mutable().FindNodeFromElementId(
scroller_id);
transform_node->scroll_offset = offset;
transform_node->needs_local_transform_update = true;
}
void CreateScrollingElement(PropertyTrees* property_trees,
ElementId scroller_id,
gfx::Size content_size,
gfx::Size container_size) {
TransformNode transform_node;
transform_node.scrolls = true;
int transform_node_id =
property_trees->transform_tree_mutable().Insert(transform_node, 0);
property_trees->transform_tree_mutable().SetElementIdForNodeId(
transform_node_id, scroller_id);
ScrollNode scroll_node;
scroll_node.scrollable = true;
scroll_node.bounds = content_size;
scroll_node.container_bounds = container_size;
scroll_node.element_id = scroller_id;
scroll_node.transform_id = transform_node_id;
int scroll_node_id =
property_trees->scroll_tree_mutable().Insert(scroll_node, 0);
property_trees->scroll_tree_mutable().SetElementIdForNodeId(scroll_node_id,
scroller_id);
}
double CalculateCurrentTime(double current_scroll_offset,
double start_scroll_offset,
double end_scroll_offset) {
return ((current_scroll_offset - start_scroll_offset) /
(end_scroll_offset - start_scroll_offset)) *
ScrollTimeline::kScrollTimelineDurationMs;
}
double ToDouble(absl::optional<base::TimeTicks> time_ticks) {
if (time_ticks)
return (time_ticks.value() - base::TimeTicks()).InMillisecondsF();
return std::numeric_limits<double>::quiet_NaN();
}
}
class ScrollTimelineTest : public ::testing::Test,
public ProtectedSequenceSynchronizer {
public:
ScrollTimelineTest()
: property_trees_(*this),
scroller_id_(1),
container_size_(100, 100),
content_size_(500, 500) {
property_trees_.set_is_main_thread(true);
property_trees_.set_is_active(false);
CreateScrollingElement(&property_trees_, scroller_id_, content_size_,
container_size_);
}
PropertyTrees& property_trees() { return property_trees_; }
ScrollTree& scroll_tree() { return property_trees_.scroll_tree_mutable(); }
ElementId scroller_id() const { return scroller_id_; }
gfx::Size container_size() const { return container_size_; }
gfx::Size content_size() const { return content_size_; }
bool IsOwnerThread() const override { return true; }
bool InProtectedSequence() const override { return false; }
void WaitForProtectedSequenceCompletion() const override {}
private:
PropertyTrees property_trees_;
ElementId scroller_id_;
gfx::Size container_size_;
gfx::Size content_size_;
};
TEST_F(ScrollTimelineTest, BasicCurrentTimeCalculations) {
ScrollTimeline::ScrollOffsets scroll_offsets(0, 100);
scoped_refptr<ScrollTimeline> vertical_timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
scoped_refptr<ScrollTimeline> horizontal_timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollRight, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF());
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
0, vertical_timeline->CurrentTime(scroll_tree(), false));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
0, horizontal_timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(75, 50));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
0.5 * ScrollTimeline::kScrollTimelineDurationMs,
vertical_timeline->CurrentTime(scroll_tree(), false));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
0.75 * ScrollTimeline::kScrollTimelineDurationMs,
horizontal_timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, ActiveTimeIsSetOnlyAfterPromotion) {
PropertyTrees pending_tree(*this);
PropertyTrees active_tree(*this);
pending_tree.set_is_active(false);
active_tree.set_is_active(true);
pending_tree.set_is_main_thread(true);
active_tree.set_is_main_thread(true);
ElementId scroller_id(1);
CreateScrollingElement(&pending_tree, scroller_id, content_size(),
container_size());
double scroll_size = content_size().height() - container_size().height();
ScrollTimeline::ScrollOffsets scroll_offsets(0, scroll_size);
double halfwayY = scroll_size / 2.;
double expectedTime = 0.5 * ScrollTimeline::kScrollTimelineDurationMs;
SetScrollOffset(&pending_tree, scroller_id, gfx::PointF(0, halfwayY));
scoped_refptr<ScrollTimeline> main_timeline = ScrollTimeline::Create(
scroller_id, ScrollTimeline::ScrollDown, scroll_offsets);
scoped_refptr<ScrollTimeline> impl_timeline = base::WrapRefCounted(
ToScrollTimeline(main_timeline->CreateImplInstance().get()));
EXPECT_TRUE(std::isnan(
ToDouble(impl_timeline->CurrentTime(active_tree.scroll_tree(), true))));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
expectedTime,
impl_timeline->CurrentTime(pending_tree.scroll_tree(), false));
impl_timeline->ActivateTimeline();
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
expectedTime,
impl_timeline->CurrentTime(pending_tree.scroll_tree(), true));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
expectedTime,
impl_timeline->CurrentTime(pending_tree.scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeIsAdjustedForPixelSnapping) {
double scroll_size = content_size().height() - container_size().height();
ScrollTimeline::ScrollOffsets scroll_offsets(0, scroll_size);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 50));
TransformNode* transform_node =
property_trees().transform_tree_mutable().FindNodeFromElementId(
scroller_id());
transform_node->snap_amount = gfx::Vector2dF(0, 0.5);
double scale = ScrollTimeline::kScrollTimelineDurationMs / scroll_size;
EXPECT_SCROLL_TIMELINE_TIME_NEAR(49.5 * scale,
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeHandlesStartScrollOffset) {
double scroll_size = content_size().height() - container_size().height();
const double start_scroll_offset = 20;
ScrollTimeline::ScrollOffsets scroll_offsets(start_scroll_offset,
scroll_size);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF());
EXPECT_SCROLL_TIMELINE_BEFORE_START(
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 19));
EXPECT_SCROLL_TIMELINE_BEFORE_START(
timeline->CurrentTime(scroll_tree(), false).value());
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 20));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(0,
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 50));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
CalculateCurrentTime(50, start_scroll_offset, scroll_size),
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 200));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
CalculateCurrentTime(200, start_scroll_offset, scroll_size),
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeHandlesEndScrollOffset) {
double scroll_size = content_size().height() - container_size().height();
const double end_scroll_offset = scroll_size - 20;
ScrollTimeline::ScrollOffsets scroll_offsets(0, end_scroll_offset);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, scroll_size));
EXPECT_SCROLL_TIMELINE_AFTER_END(timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, scroll_size - 20));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(ScrollTimeline::kScrollTimelineDurationMs,
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, scroll_size - 50));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
CalculateCurrentTime(scroll_size - 50, 0, end_scroll_offset),
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, scroll_size - 200));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
CalculateCurrentTime(scroll_size - 200, 0, end_scroll_offset),
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeHandlesCombinedStartAndEndScrollOffset) {
double scroll_size = content_size().height() - container_size().height();
double start_scroll_offset = 20;
double end_scroll_offset = scroll_size - 50;
ScrollTimeline::ScrollOffsets scroll_offsets(start_scroll_offset,
end_scroll_offset);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, scroll_size - 150));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
CalculateCurrentTime(scroll_size - 150, start_scroll_offset,
end_scroll_offset),
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeHandlesEqualStartAndEndScrollOffset) {
ScrollTimeline::ScrollOffsets scroll_offsets(20, 20);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 20));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(ScrollTimeline::kScrollTimelineDurationMs,
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 150));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(ScrollTimeline::kScrollTimelineDurationMs,
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest,
CurrentTimeHandlesStartOffsetLargerThanEndScrollOffset) {
ScrollTimeline::ScrollOffsets scroll_offsets(50, 10);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 0));
EXPECT_SCROLL_TIMELINE_AFTER_END(timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 30));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(
ScrollTimeline::kScrollTimelineDurationMs / 2,
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(), gfx::PointF(0, 150));
EXPECT_SCROLL_TIMELINE_BEFORE_START(
timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, CurrentTimeHandlesScrollOffsets) {
const double start_scroll_offset = 20;
const double scroller_height =
content_size().height() - container_size().height();
const double end_scroll_offset = scroller_height - 20;
ScrollTimeline::ScrollOffsets scroll_offsets(start_scroll_offset,
end_scroll_offset);
scoped_refptr<ScrollTimeline> timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, start_scroll_offset - 10));
EXPECT_SCROLL_TIMELINE_BEFORE_START(
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, end_scroll_offset));
EXPECT_SCROLL_TIMELINE_TIME_NEAR(ScrollTimeline::kScrollTimelineDurationMs,
timeline->CurrentTime(scroll_tree(), false));
SetScrollOffset(&property_trees(), scroller_id(),
gfx::PointF(0, end_scroll_offset + 10));
EXPECT_SCROLL_TIMELINE_AFTER_END(timeline->CurrentTime(scroll_tree(), false));
}
TEST_F(ScrollTimelineTest, Activeness) {
double scroll_size = content_size().height() - container_size().height();
ScrollTimeline::ScrollOffsets scroll_offsets(0, scroll_size);
scoped_refptr<ScrollTimeline> inactive_timeline1 = ScrollTimeline::Create(
absl::nullopt, ScrollTimeline::ScrollDown, scroll_offsets);
EXPECT_FALSE(
inactive_timeline1->IsActive(scroll_tree(), false ));
EXPECT_FALSE(
inactive_timeline1->IsActive(scroll_tree(), true ));
scoped_refptr<ScrollTimeline> inactive_timeline2 = ScrollTimeline::Create(
ElementId(2), ScrollTimeline::ScrollDown, scroll_offsets);
EXPECT_FALSE(
inactive_timeline2->IsActive(scroll_tree(), false ));
inactive_timeline2->ActivateTimeline();
EXPECT_FALSE(
inactive_timeline2->IsActive(scroll_tree(), true ));
scoped_refptr<ScrollTimeline> inactive_timeline3 =
ScrollTimeline::Create(scroller_id(), ScrollTimeline::ScrollDown,
absl::nullopt);
EXPECT_FALSE(
inactive_timeline3->IsActive(scroll_tree(), false ));
EXPECT_FALSE(
inactive_timeline3->IsActive(scroll_tree(), true ));
inactive_timeline3->ActivateTimeline();
EXPECT_FALSE(
inactive_timeline3->IsActive(scroll_tree(), false ));
EXPECT_FALSE(
inactive_timeline3->IsActive(scroll_tree(), true ));
scoped_refptr<ScrollTimeline> active_timeline = ScrollTimeline::Create(
scroller_id(), ScrollTimeline::ScrollDown, scroll_offsets);
EXPECT_TRUE(
active_timeline->IsActive(scroll_tree(), false ));
EXPECT_FALSE(
active_timeline->IsActive(scroll_tree(), true ));
active_timeline->ActivateTimeline();
EXPECT_TRUE(
active_timeline->IsActive(scroll_tree(), false ));
EXPECT_TRUE(
active_timeline->IsActive(scroll_tree(), true ));
}
}