summaryrefslogtreecommitdiff
path: root/chromium/cc/layers/layer_impl_unittest.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/cc/layers/layer_impl_unittest.cc')
-rw-r--r--chromium/cc/layers/layer_impl_unittest.cc562
1 files changed, 562 insertions, 0 deletions
diff --git a/chromium/cc/layers/layer_impl_unittest.cc b/chromium/cc/layers/layer_impl_unittest.cc
new file mode 100644
index 00000000000..9c6dbfe3790
--- /dev/null
+++ b/chromium/cc/layers/layer_impl_unittest.cc
@@ -0,0 +1,562 @@
+// Copyright 2011 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "cc/layers/layer_impl.h"
+
+#include "cc/output/filter_operation.h"
+#include "cc/output/filter_operations.h"
+#include "cc/test/fake_impl_proxy.h"
+#include "cc/test/fake_layer_tree_host_impl.h"
+#include "cc/test/fake_output_surface.h"
+#include "cc/test/geometry_test_utils.h"
+#include "cc/trees/layer_tree_impl.h"
+#include "cc/trees/single_thread_proxy.h"
+#include "testing/gmock/include/gmock/gmock.h"
+#include "testing/gtest/include/gtest/gtest.h"
+#include "third_party/skia/include/effects/SkBlurImageFilter.h"
+
+namespace cc {
+namespace {
+
+#define EXECUTE_AND_VERIFY_SUBTREE_CHANGED(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ code_to_test; \
+ EXPECT_TRUE(root->LayerPropertyChanged()); \
+ EXPECT_TRUE(child->LayerPropertyChanged()); \
+ EXPECT_TRUE(grand_child->LayerPropertyChanged()); \
+ EXPECT_FALSE(root->LayerSurfacePropertyChanged())
+
+#define EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ code_to_test; \
+ EXPECT_FALSE(root->LayerPropertyChanged()); \
+ EXPECT_FALSE(child->LayerPropertyChanged()); \
+ EXPECT_FALSE(grand_child->LayerPropertyChanged()); \
+ EXPECT_FALSE(root->LayerSurfacePropertyChanged())
+
+#define EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ code_to_test; \
+ EXPECT_TRUE(root->LayerPropertyChanged()); \
+ EXPECT_FALSE(child->LayerPropertyChanged()); \
+ EXPECT_FALSE(grand_child->LayerPropertyChanged()); \
+ EXPECT_FALSE(root->LayerSurfacePropertyChanged())
+
+#define EXECUTE_AND_VERIFY_ONLY_SURFACE_CHANGED(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ code_to_test; \
+ EXPECT_FALSE(root->LayerPropertyChanged()); \
+ EXPECT_FALSE(child->LayerPropertyChanged()); \
+ EXPECT_FALSE(grand_child->LayerPropertyChanged()); \
+ EXPECT_TRUE(root->LayerSurfacePropertyChanged())
+
+#define VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ host_impl.ForcePrepareToDraw(); \
+ EXPECT_FALSE(host_impl.active_tree()->needs_update_draw_properties()); \
+ code_to_test; \
+ EXPECT_TRUE(host_impl.active_tree()->needs_update_draw_properties());
+
+#define VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(code_to_test) \
+ root->ResetAllChangeTrackingForSubtree(); \
+ host_impl.ForcePrepareToDraw(); \
+ EXPECT_FALSE(host_impl.active_tree()->needs_update_draw_properties()); \
+ code_to_test; \
+ EXPECT_FALSE(host_impl.active_tree()->needs_update_draw_properties());
+
+TEST(LayerImplTest, VerifyLayerChangesAreTrackedProperly) {
+ //
+ // This test checks that layerPropertyChanged() has the correct behavior.
+ //
+
+ // The constructor on this will fake that we are on the correct thread.
+ // Create a simple LayerImpl tree:
+ FakeImplProxy proxy;
+ FakeLayerTreeHostImpl host_impl(&proxy);
+ EXPECT_TRUE(host_impl.InitializeRenderer(CreateFakeOutputSurface()));
+ scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1);
+ root->AddChild(LayerImpl::Create(host_impl.active_tree(), 2));
+ LayerImpl* child = root->children()[0];
+ child->AddChild(LayerImpl::Create(host_impl.active_tree(), 3));
+ LayerImpl* grand_child = child->children()[0];
+
+ root->SetScrollable(true);
+
+ // Adding children is an internal operation and should not mark layers as
+ // changed.
+ EXPECT_FALSE(root->LayerPropertyChanged());
+ EXPECT_FALSE(child->LayerPropertyChanged());
+ EXPECT_FALSE(grand_child->LayerPropertyChanged());
+
+ gfx::PointF arbitrary_point_f = gfx::PointF(0.125f, 0.25f);
+ float arbitrary_number = 0.352f;
+ gfx::Size arbitrary_size = gfx::Size(111, 222);
+ gfx::Point arbitrary_point = gfx::Point(333, 444);
+ gfx::Vector2d arbitrary_vector2d = gfx::Vector2d(111, 222);
+ gfx::Rect arbitrary_rect = gfx::Rect(arbitrary_point, arbitrary_size);
+ gfx::RectF arbitrary_rect_f =
+ gfx::RectF(arbitrary_point_f, gfx::SizeF(1.234f, 5.678f));
+ SkColor arbitrary_color = SkColorSetRGB(10, 20, 30);
+ gfx::Transform arbitrary_transform;
+ arbitrary_transform.Scale3d(0.1, 0.2, 0.3);
+ FilterOperations arbitrary_filters;
+ arbitrary_filters.Append(FilterOperation::CreateOpacityFilter(0.5f));
+ skia::RefPtr<SkImageFilter> arbitrary_filter =
+ skia::AdoptRef(new SkBlurImageFilter(SK_Scalar1, SK_Scalar1));
+
+ // These properties are internal, and should not be considered "change" when
+ // they are used.
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->set_update_rect(arbitrary_rect_f));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetMaxScrollOffset(arbitrary_vector2d));
+
+ // Changing these properties affects the entire subtree of layers.
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetAnchorPoint(arbitrary_point_f));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetAnchorPointZ(arbitrary_number));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetFilters(arbitrary_filters));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetFilters(FilterOperations()));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetFilter(arbitrary_filter));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(
+ root->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 4)));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetMasksToBounds(true));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetContentsOpaque(true));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(
+ root->SetReplicaLayer(LayerImpl::Create(host_impl.active_tree(), 5)));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetPosition(arbitrary_point_f));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetPreserves3d(true));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(
+ root->SetDoubleSided(false)); // constructor initializes it to "true".
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->ScrollBy(arbitrary_vector2d));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetScrollDelta(gfx::Vector2d()));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetScrollOffset(arbitrary_vector2d));
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetHideLayerAndSubtree(true));
+
+ // Changing these properties only affects the layer itself.
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(root->SetContentBounds(arbitrary_size));
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(
+ root->SetContentsScale(arbitrary_number, arbitrary_number));
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(root->SetDrawsContent(true));
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(
+ root->SetBackgroundColor(arbitrary_color));
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(
+ root->SetBackgroundFilters(arbitrary_filters));
+
+ // Changing these properties only affects how render surface is drawn
+ EXECUTE_AND_VERIFY_ONLY_SURFACE_CHANGED(root->SetOpacity(arbitrary_number));
+ EXECUTE_AND_VERIFY_ONLY_SURFACE_CHANGED(
+ root->SetTransform(arbitrary_transform));
+
+ // Special case: check that sublayer transform changes all layer's
+ // descendants, but not the layer itself.
+ root->ResetAllChangeTrackingForSubtree();
+ root->SetSublayerTransform(arbitrary_transform);
+ EXPECT_FALSE(root->LayerPropertyChanged());
+ EXPECT_TRUE(child->LayerPropertyChanged());
+ EXPECT_TRUE(grand_child->LayerPropertyChanged());
+
+ // Special case: check that SetBounds changes behavior depending on
+ // masksToBounds.
+ root->SetMasksToBounds(false);
+ EXECUTE_AND_VERIFY_ONLY_LAYER_CHANGED(root->SetBounds(gfx::Size(135, 246)));
+ root->SetMasksToBounds(true);
+ // Should be a different size than previous call, to ensure it marks tree
+ // changed.
+ EXECUTE_AND_VERIFY_SUBTREE_CHANGED(root->SetBounds(arbitrary_size));
+
+ // After setting all these properties already, setting to the exact same
+ // values again should not cause any change.
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetAnchorPoint(arbitrary_point_f));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetAnchorPointZ(arbitrary_number));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetMasksToBounds(true));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetPosition(arbitrary_point_f));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetPreserves3d(true));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetTransform(arbitrary_transform));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetDoubleSided(false)); // constructor initializes it to "true".
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetScrollDelta(gfx::Vector2d()));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetScrollOffset(arbitrary_vector2d));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetContentBounds(arbitrary_size));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetContentsScale(arbitrary_number, arbitrary_number));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetContentsOpaque(true));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetOpacity(arbitrary_number));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetDrawsContent(true));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(
+ root->SetSublayerTransform(arbitrary_transform));
+ EXECUTE_AND_VERIFY_SUBTREE_DID_NOT_CHANGE(root->SetBounds(arbitrary_size));
+}
+
+TEST(LayerImplTest, VerifyNeedsUpdateDrawProperties) {
+ FakeImplProxy proxy;
+ FakeLayerTreeHostImpl host_impl(&proxy);
+ EXPECT_TRUE(host_impl.InitializeRenderer(CreateFakeOutputSurface()));
+ scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1);
+ root->SetScrollable(true);
+
+ gfx::PointF arbitrary_point_f = gfx::PointF(0.125f, 0.25f);
+ float arbitrary_number = 0.352f;
+ gfx::Size arbitrary_size = gfx::Size(111, 222);
+ gfx::Point arbitrary_point = gfx::Point(333, 444);
+ gfx::Vector2d arbitrary_vector2d = gfx::Vector2d(111, 222);
+ gfx::Vector2d large_vector2d = gfx::Vector2d(1000, 1000);
+ gfx::Rect arbitrary_rect = gfx::Rect(arbitrary_point, arbitrary_size);
+ gfx::RectF arbitrary_rect_f =
+ gfx::RectF(arbitrary_point_f, gfx::SizeF(1.234f, 5.678f));
+ SkColor arbitrary_color = SkColorSetRGB(10, 20, 30);
+ gfx::Transform arbitrary_transform;
+ arbitrary_transform.Scale3d(0.1, 0.2, 0.3);
+ FilterOperations arbitrary_filters;
+ arbitrary_filters.Append(FilterOperation::CreateOpacityFilter(0.5f));
+ skia::RefPtr<SkImageFilter> arbitrary_filter =
+ skia::AdoptRef(new SkBlurImageFilter(SK_Scalar1, SK_Scalar1));
+
+ // Related filter functions.
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilters(arbitrary_filters));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilters(arbitrary_filters));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilters(FilterOperations()));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilter(arbitrary_filter));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilter(arbitrary_filter));
+
+ // Related scrolling functions.
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetMaxScrollOffset(large_vector2d));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetMaxScrollOffset(large_vector2d));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->ScrollBy(arbitrary_vector2d));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->ScrollBy(gfx::Vector2d()));
+ root->SetScrollDelta(gfx::Vector2d(0, 0));
+ host_impl.ForcePrepareToDraw();
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetScrollDelta(arbitrary_vector2d));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetScrollDelta(arbitrary_vector2d));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetScrollOffset(arbitrary_vector2d));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetScrollOffset(arbitrary_vector2d));
+
+ // Unrelated functions, always set to new values, always set needs update.
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetAnchorPointZ(arbitrary_number));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 4)));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetMasksToBounds(true));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetContentsOpaque(true));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetReplicaLayer(LayerImpl::Create(host_impl.active_tree(), 5)));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetPosition(arbitrary_point_f));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetPreserves3d(true));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetDoubleSided(false)); // constructor initializes it to "true".
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetContentBounds(arbitrary_size));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetContentsScale(arbitrary_number, arbitrary_number));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetDrawsContent(true));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetBackgroundColor(arbitrary_color));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetBackgroundFilters(arbitrary_filters));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetOpacity(arbitrary_number));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetTransform(arbitrary_transform));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetSublayerTransform(arbitrary_transform));
+ VERIFY_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetBounds(arbitrary_size));
+
+ // Unrelated functions, set to the same values, no needs update.
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetAnchorPointZ(arbitrary_number));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetFilter(arbitrary_filter));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetMasksToBounds(true));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetContentsOpaque(true));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetPosition(arbitrary_point_f));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetPreserves3d(true));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetDoubleSided(false)); // constructor initializes it to "true".
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetContentBounds(arbitrary_size));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetContentsScale(arbitrary_number, arbitrary_number));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetDrawsContent(true));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetBackgroundColor(arbitrary_color));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetBackgroundFilters(arbitrary_filters));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetOpacity(arbitrary_number));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetTransform(arbitrary_transform));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(
+ root->SetSublayerTransform(arbitrary_transform));
+ VERIFY_NO_NEEDS_UPDATE_DRAW_PROPERTIES(root->SetBounds(arbitrary_size));
+}
+
+TEST(LayerImplTest, SafeOpaqueBackgroundColor) {
+ FakeImplProxy proxy;
+ FakeLayerTreeHostImpl host_impl(&proxy);
+ EXPECT_TRUE(host_impl.InitializeRenderer(CreateFakeOutputSurface()));
+ scoped_ptr<LayerImpl> layer = LayerImpl::Create(host_impl.active_tree(), 1);
+
+ for (int contents_opaque = 0; contents_opaque < 2; ++contents_opaque) {
+ for (int layer_opaque = 0; layer_opaque < 2; ++layer_opaque) {
+ for (int host_opaque = 0; host_opaque < 2; ++host_opaque) {
+ layer->SetContentsOpaque(!!contents_opaque);
+ layer->SetBackgroundColor(layer_opaque ? SK_ColorRED
+ : SK_ColorTRANSPARENT);
+ host_impl.active_tree()->set_background_color(
+ host_opaque ? SK_ColorRED : SK_ColorTRANSPARENT);
+
+ SkColor safe_color = layer->SafeOpaqueBackgroundColor();
+ if (contents_opaque) {
+ EXPECT_EQ(SkColorGetA(safe_color), 255u)
+ << "Flags: " << contents_opaque << ", " << layer_opaque << ", "
+ << host_opaque << "\n";
+ } else {
+ EXPECT_NE(SkColorGetA(safe_color), 255u)
+ << "Flags: " << contents_opaque << ", " << layer_opaque << ", "
+ << host_opaque << "\n";
+ }
+ }
+ }
+ }
+}
+
+class LayerImplScrollTest : public testing::Test {
+ public:
+ LayerImplScrollTest() : host_impl_(&proxy_), root_id_(7) {
+ host_impl_.active_tree()
+ ->SetRootLayer(LayerImpl::Create(host_impl_.active_tree(), root_id_));
+ host_impl_.active_tree()->root_layer()->SetScrollable(true);
+ }
+
+ LayerImpl* layer() { return host_impl_.active_tree()->root_layer(); }
+
+ private:
+ FakeImplProxy proxy_;
+ FakeLayerTreeHostImpl host_impl_;
+ int root_id_;
+};
+
+TEST_F(LayerImplScrollTest, ScrollByWithZeroOffset) {
+ // Test that LayerImpl::ScrollBy only affects ScrollDelta and total scroll
+ // offset is bounded by the range [0, max scroll offset].
+ gfx::Vector2d max_scroll_offset(50, 80);
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->ScrollDelta());
+
+ layer()->ScrollBy(gfx::Vector2dF(-100, 100));
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(0, 80), layer()->TotalScrollOffset());
+
+ EXPECT_VECTOR_EQ(layer()->ScrollDelta(), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->scroll_offset());
+
+ layer()->ScrollBy(gfx::Vector2dF(100, -100));
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(50, 0), layer()->TotalScrollOffset());
+
+ EXPECT_VECTOR_EQ(layer()->ScrollDelta(), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->scroll_offset());
+}
+
+TEST_F(LayerImplScrollTest, ScrollByWithNonZeroOffset) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->ScrollDelta());
+
+ layer()->ScrollBy(gfx::Vector2dF(-100, 100));
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(0, 80), layer()->TotalScrollOffset());
+
+ EXPECT_VECTOR_EQ(layer()->ScrollDelta() + scroll_offset,
+ layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ layer()->ScrollBy(gfx::Vector2dF(100, -100));
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(50, 0), layer()->TotalScrollOffset());
+
+ EXPECT_VECTOR_EQ(layer()->ScrollDelta() + scroll_offset,
+ layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+}
+
+class ScrollDelegateIgnore : public LayerScrollOffsetDelegate {
+ public:
+ virtual void SetTotalScrollOffset(gfx::Vector2dF new_value) OVERRIDE {}
+ virtual gfx::Vector2dF GetTotalScrollOffset() OVERRIDE {
+ return fixed_offset_;
+ }
+
+ void set_fixed_offset(gfx::Vector2dF fixed_offset) {
+ fixed_offset_ = fixed_offset;
+ }
+
+ private:
+ gfx::Vector2dF fixed_offset_;
+};
+
+TEST_F(LayerImplScrollTest, ScrollByWithIgnoringDelegate) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->ScrollDelta());
+
+ ScrollDelegateIgnore delegate;
+ gfx::Vector2dF fixed_offset(32, 12);
+ delegate.set_fixed_offset(fixed_offset);
+ layer()->SetScrollOffsetDelegate(&delegate);
+
+ EXPECT_VECTOR_EQ(fixed_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ layer()->ScrollBy(gfx::Vector2dF(-100, 100));
+
+ EXPECT_VECTOR_EQ(fixed_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ layer()->SetScrollOffsetDelegate(NULL);
+
+ EXPECT_VECTOR_EQ(fixed_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ gfx::Vector2dF scroll_delta(1, 1);
+ layer()->ScrollBy(scroll_delta);
+
+ EXPECT_VECTOR_EQ(fixed_offset + scroll_delta, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+}
+
+class ScrollDelegateAccept : public LayerScrollOffsetDelegate {
+ public:
+ virtual void SetTotalScrollOffset(gfx::Vector2dF new_value) OVERRIDE {
+ current_offset_ = new_value;
+ }
+ virtual gfx::Vector2dF GetTotalScrollOffset() OVERRIDE {
+ return current_offset_;
+ }
+
+ private:
+ gfx::Vector2dF current_offset_;
+};
+
+TEST_F(LayerImplScrollTest, ScrollByWithAcceptingDelegate) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->ScrollDelta());
+
+ ScrollDelegateAccept delegate;
+ layer()->SetScrollOffsetDelegate(&delegate);
+
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(), layer()->ScrollDelta());
+
+ layer()->ScrollBy(gfx::Vector2dF(-100, 100));
+
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(0, 80), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ layer()->SetScrollOffsetDelegate(NULL);
+
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(0, 80), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+
+ gfx::Vector2dF scroll_delta(1, 1);
+ layer()->ScrollBy(scroll_delta);
+
+ EXPECT_VECTOR_EQ(gfx::Vector2dF(1, 80), layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+}
+
+TEST_F(LayerImplScrollTest, ApplySentScrollsNoDelegate) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ gfx::Vector2dF scroll_delta(20.5f, 8.5f);
+ gfx::Vector2d sent_scroll_delta(12, -3);
+
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+ layer()->ScrollBy(scroll_delta);
+ layer()->SetSentScrollDelta(sent_scroll_delta);
+
+ EXPECT_VECTOR_EQ(scroll_offset + scroll_delta, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_delta, layer()->ScrollDelta());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(sent_scroll_delta, layer()->sent_scroll_delta());
+
+ layer()->ApplySentScrollDeltas();
+
+ EXPECT_VECTOR_EQ(scroll_offset + scroll_delta, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_delta - sent_scroll_delta, layer()->ScrollDelta());
+ EXPECT_VECTOR_EQ(scroll_offset + sent_scroll_delta, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2d(), layer()->sent_scroll_delta());
+}
+
+TEST_F(LayerImplScrollTest, ApplySentScrollsWithIgnoringDelegate) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ gfx::Vector2d sent_scroll_delta(12, -3);
+ gfx::Vector2dF fixed_offset(32, 12);
+
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+ ScrollDelegateIgnore delegate;
+ delegate.set_fixed_offset(fixed_offset);
+ layer()->SetScrollOffsetDelegate(&delegate);
+ layer()->SetSentScrollDelta(sent_scroll_delta);
+
+ EXPECT_VECTOR_EQ(fixed_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(sent_scroll_delta, layer()->sent_scroll_delta());
+
+ layer()->ApplySentScrollDeltas();
+
+ EXPECT_VECTOR_EQ(fixed_offset, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset + sent_scroll_delta, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2d(), layer()->sent_scroll_delta());
+}
+
+TEST_F(LayerImplScrollTest, ApplySentScrollsWithAcceptingDelegate) {
+ gfx::Vector2d max_scroll_offset(50, 80);
+ gfx::Vector2d scroll_offset(10, 5);
+ gfx::Vector2d sent_scroll_delta(12, -3);
+ gfx::Vector2dF scroll_delta(20.5f, 8.5f);
+
+ layer()->SetMaxScrollOffset(max_scroll_offset);
+ layer()->SetScrollOffset(scroll_offset);
+ ScrollDelegateAccept delegate;
+ layer()->SetScrollOffsetDelegate(&delegate);
+ layer()->ScrollBy(scroll_delta);
+ layer()->SetSentScrollDelta(sent_scroll_delta);
+
+ EXPECT_VECTOR_EQ(scroll_offset + scroll_delta, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(sent_scroll_delta, layer()->sent_scroll_delta());
+
+ layer()->ApplySentScrollDeltas();
+
+ EXPECT_VECTOR_EQ(scroll_offset + scroll_delta, layer()->TotalScrollOffset());
+ EXPECT_VECTOR_EQ(scroll_offset + sent_scroll_delta, layer()->scroll_offset());
+ EXPECT_VECTOR_EQ(gfx::Vector2d(), layer()->sent_scroll_delta());
+}
+
+} // namespace
+} // namespace cc