| // 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/trees/layer_tree_host_common.h" | 
 |  | 
 | #include <stddef.h> | 
 |  | 
 | #include <algorithm> | 
 | #include <set> | 
 | #include <vector> | 
 |  | 
 | #include "cc/animation/animation_host.h" | 
 | #include "cc/animation/animation_id_provider.h" | 
 | #include "cc/animation/keyframed_animation_curve.h" | 
 | #include "cc/animation/layer_animation_controller.h" | 
 | #include "cc/animation/transform_operations.h" | 
 | #include "cc/base/math_util.h" | 
 | #include "cc/layers/content_layer_client.h" | 
 | #include "cc/layers/layer.h" | 
 | #include "cc/layers/layer_client.h" | 
 | #include "cc/layers/layer_impl.h" | 
 | #include "cc/layers/layer_iterator.h" | 
 | #include "cc/layers/render_surface_impl.h" | 
 | #include "cc/output/copy_output_request.h" | 
 | #include "cc/output/copy_output_result.h" | 
 | #include "cc/proto/begin_main_frame_and_commit_state.pb.h" | 
 | #include "cc/proto/gfx_conversions.h" | 
 | #include "cc/test/animation_test_common.h" | 
 | #include "cc/test/fake_content_layer_client.h" | 
 | #include "cc/test/fake_impl_task_runner_provider.h" | 
 | #include "cc/test/fake_layer_tree_host.h" | 
 | #include "cc/test/fake_layer_tree_host_impl.h" | 
 | #include "cc/test/fake_output_surface.h" | 
 | #include "cc/test/fake_picture_layer.h" | 
 | #include "cc/test/fake_picture_layer_impl.h" | 
 | #include "cc/test/geometry_test_utils.h" | 
 | #include "cc/test/layer_tree_host_common_test.h" | 
 | #include "cc/test/test_task_graph_runner.h" | 
 | #include "cc/trees/draw_property_utils.h" | 
 | #include "cc/trees/layer_tree_impl.h" | 
 | #include "cc/trees/single_thread_proxy.h" | 
 | #include "cc/trees/task_runner_provider.h" | 
 | #include "testing/gmock/include/gmock/gmock.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 | #include "ui/gfx/geometry/quad_f.h" | 
 | #include "ui/gfx/geometry/vector2d_conversions.h" | 
 | #include "ui/gfx/transform.h" | 
 |  | 
 | namespace cc { | 
 | namespace { | 
 |  | 
 | class LayerWithForcedDrawsContent : public Layer { | 
 |  public: | 
 |   explicit LayerWithForcedDrawsContent(const LayerSettings& settings) | 
 |       : Layer(settings) {} | 
 |  | 
 |   bool DrawsContent() const override; | 
 |  | 
 |  private: | 
 |   ~LayerWithForcedDrawsContent() override {} | 
 | }; | 
 |  | 
 | bool LayerWithForcedDrawsContent::DrawsContent() const { return true; } | 
 |  | 
 | class MockContentLayerClient : public ContentLayerClient { | 
 |  public: | 
 |   MockContentLayerClient() {} | 
 |   ~MockContentLayerClient() override {} | 
 |   gfx::Rect PaintableRegion() override { return gfx::Rect(); } | 
 |   scoped_refptr<DisplayItemList> PaintContentsToDisplayList( | 
 |       PaintingControlSetting picture_control) override { | 
 |     NOTIMPLEMENTED(); | 
 |     return nullptr; | 
 |   } | 
 |   bool FillsBoundsCompletely() const override { return false; } | 
 |   size_t GetApproximateUnsharedMemoryUsage() const override { return 0; } | 
 | }; | 
 |  | 
 | #define EXPECT_CONTENTS_SCALE_EQ(expected, layer)         \ | 
 |   do {                                                    \ | 
 |     EXPECT_FLOAT_EQ(expected, layer->contents_scale_x()); \ | 
 |     EXPECT_FLOAT_EQ(expected, layer->contents_scale_y()); \ | 
 |   } while (false) | 
 |  | 
 | #define EXPECT_IDEAL_SCALE_EQ(expected, layer)                 \ | 
 |   do {                                                         \ | 
 |     EXPECT_FLOAT_EQ(expected, layer->GetIdealContentsScale()); \ | 
 |   } while (false) | 
 |  | 
 | class LayerTreeSettingsScaleContent : public LayerTreeSettings { | 
 |  public: | 
 |   LayerTreeSettingsScaleContent() { | 
 |     layer_transforms_should_scale_layer_contents = true; | 
 |     use_compositor_animation_timelines = true; | 
 |   } | 
 | }; | 
 |  | 
 | class LayerTreeHostCommonScalingTest : public LayerTreeHostCommonTest { | 
 |  public: | 
 |   LayerTreeHostCommonScalingTest() | 
 |       : LayerTreeHostCommonTest(LayerTreeSettingsScaleContent()) {} | 
 | }; | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForNoOpLayer) { | 
 |   // Sanity check: For layers positioned at zero, with zero size, | 
 |   // and with identity transforms, then the draw transform, | 
 |   // screen space transform, and the hierarchy passed on to children | 
 |   // layers should also be identity transforms. | 
 |  | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child = Layer::Create(layer_settings()); | 
 |   parent->AddChild(child); | 
 |   child->AddChild(grand_child); | 
 |  | 
 |   host()->SetRootLayer(parent); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent.get()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, child->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   child->screen_space_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   grand_child->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   grand_child->screen_space_transform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        ScreenSpaceTransformOfSkippedLayersWithHandlers) { | 
 |   // Even for layers that are skipped, we need to compute the correct screen | 
 |   // space transform because it is used during hit testing. | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10, 10), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   // This will cause the subtree to be skipped. | 
 |   child->SetOpacity(0.f); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10, 10), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   grand_child->SetTouchEventHandlerRegion(gfx::Rect(0, 0, 100, 100)); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   EXPECT_TRUE(child->has_render_surface()); | 
 |   EXPECT_FALSE(grand_child->has_render_surface()); | 
 |   // Check that we've computed draw properties for the subtree rooted at | 
 |   // |child|. | 
 |   EXPECT_FALSE(child->render_surface()->screen_space_transform().IsIdentity()); | 
 |   EXPECT_FALSE(grand_child->ScreenSpaceTransform().IsIdentity()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, EffectTreeTransformIdTest) { | 
 |   // Tests that effect tree node gets a valid transform id when a layer | 
 |   // has opacity but doesn't create a render surface. | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10, 10), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   child->SetOpacity(0.f); | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |   EffectTree effect_tree = | 
 |       parent->layer_tree_impl()->property_trees()->effect_tree; | 
 |   EffectNode* node = effect_tree.Node(child->effect_tree_index()); | 
 |   const int transform_tree_size = parent->layer_tree_impl() | 
 |                                       ->property_trees() | 
 |                                       ->transform_tree.next_available_id(); | 
 |   EXPECT_LT(node->data.transform_id, transform_tree_size); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForSingleLayer) { | 
 |   gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> layer = Layer::Create(layer_settings()); | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false); | 
 |   root->AddChild(layer); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   TransformTree& tree = host()->property_trees()->transform_tree; | 
 |  | 
 |   // Case 2: Setting the bounds of the layer should not affect either the draw | 
 |   // transform or the screenspace transform. | 
 |   gfx::Transform translation_to_center; | 
 |   translation_to_center.Translate(5.0, 6.0); | 
 |   SetLayerPropertiesForTesting(layer.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 |  | 
 |   // Case 3: The anchor point by itself (without a layer transform) should have | 
 |   // no effect on the transforms. | 
 |   SetLayerPropertiesForTesting(layer.get(), identity_matrix, | 
 |                                gfx::Point3F(2.5f, 3.0f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 |  | 
 |   // Case 4: A change in actual position affects both the draw transform and | 
 |   // screen space transform. | 
 |   gfx::Transform position_transform; | 
 |   position_transform.Translate(0.f, 1.2f); | 
 |   SetLayerPropertiesForTesting( | 
 |       layer.get(), identity_matrix, gfx::Point3F(2.5f, 3.0f, 0.f), | 
 |       gfx::PointF(0.f, 1.2f), gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       position_transform, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       position_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 |  | 
 |   // Case 5: In the correct sequence of transforms, the layer transform should | 
 |   // pre-multiply the translation_to_center. This is easily tested by using a | 
 |   // scale transform, because scale and translation are not commutative. | 
 |   gfx::Transform layer_transform; | 
 |   layer_transform.Scale3d(2.0, 2.0, 1.0); | 
 |   SetLayerPropertiesForTesting(layer.get(), layer_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       layer_transform, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       layer_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 |  | 
 |   // Case 6: The layer transform should occur with respect to the anchor point. | 
 |   gfx::Transform translation_to_anchor; | 
 |   translation_to_anchor.Translate(5.0, 0.0); | 
 |   gfx::Transform expected_result = | 
 |       translation_to_anchor * layer_transform * Inverse(translation_to_anchor); | 
 |   SetLayerPropertiesForTesting(layer.get(), layer_transform, | 
 |                                gfx::Point3F(5.0f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_result, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_result, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 |  | 
 |   // Case 7: Verify that position pre-multiplies the layer transform.  The | 
 |   // current implementation of CalculateDrawProperties does this implicitly, but | 
 |   // it is still worth testing to detect accidental regressions. | 
 |   expected_result = position_transform * translation_to_anchor * | 
 |                     layer_transform * Inverse(translation_to_anchor); | 
 |   SetLayerPropertiesForTesting( | 
 |       layer.get(), layer_transform, gfx::Point3F(5.0f, 0.f, 0.f), | 
 |       gfx::PointF(0.f, 1.2f), gfx::Size(10, 12), true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_result, DrawTransformFromPropertyTrees(layer.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_result, | 
 |       ScreenSpaceTransformFromPropertyTrees(layer.get(), tree)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsAboutScrollOffset) { | 
 |   const gfx::ScrollOffset kScrollOffset(50, 100); | 
 |   const gfx::Vector2dF kScrollDelta(2.34f, 5.67f); | 
 |   const gfx::Vector2d kMaxScrollOffset(200, 200); | 
 |   const gfx::PointF kScrollLayerPosition(-kScrollOffset.x(), | 
 |                                          -kScrollOffset.y()); | 
 |   const float kPageScale = 0.888f; | 
 |   const float kDeviceScale = 1.666f; | 
 |  | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   scoped_ptr<LayerImpl> sublayer_scoped_ptr( | 
 |       LayerImpl::Create(host_impl.active_tree(), 1)); | 
 |   LayerImpl* sublayer = sublayer_scoped_ptr.get(); | 
 |   sublayer->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(sublayer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                false); | 
 |  | 
 |   scoped_ptr<LayerImpl> scroll_layer_scoped_ptr( | 
 |       LayerImpl::Create(host_impl.active_tree(), 2)); | 
 |   LayerImpl* scroll_layer = scroll_layer_scoped_ptr.get(); | 
 |   SetLayerPropertiesForTesting(scroll_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 20), true, false, | 
 |                                false); | 
 |   scoped_ptr<LayerImpl> clip_layer_scoped_ptr( | 
 |       LayerImpl::Create(host_impl.active_tree(), 4)); | 
 |   LayerImpl* clip_layer = clip_layer_scoped_ptr.get(); | 
 |  | 
 |   scroll_layer->SetScrollClipLayer(clip_layer->id()); | 
 |   clip_layer->SetBounds( | 
 |       gfx::Size(scroll_layer->bounds().width() + kMaxScrollOffset.x(), | 
 |                 scroll_layer->bounds().height() + kMaxScrollOffset.y())); | 
 |   scroll_layer->SetScrollClipLayer(clip_layer->id()); | 
 |   scroll_layer->SetScrollDelta(kScrollDelta); | 
 |   gfx::Transform impl_transform; | 
 |   scroll_layer->AddChild(std::move(sublayer_scoped_ptr)); | 
 |   LayerImpl* scroll_layer_raw_ptr = scroll_layer_scoped_ptr.get(); | 
 |   clip_layer->AddChild(std::move(scroll_layer_scoped_ptr)); | 
 |   scroll_layer_raw_ptr->PushScrollOffsetFromMainThread(kScrollOffset); | 
 |  | 
 |   scoped_ptr<LayerImpl> root(LayerImpl::Create(host_impl.active_tree(), 3)); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                false); | 
 |   root->AddChild(std::move(clip_layer_scoped_ptr)); | 
 |   root->SetHasRenderSurface(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties( | 
 |       root.get(), kDeviceScale, kPageScale, scroll_layer->parent()); | 
 |   gfx::Transform expected_transform = identity_matrix; | 
 |   gfx::PointF sub_layer_screen_position = kScrollLayerPosition - kScrollDelta; | 
 |   expected_transform.Translate(MathUtil::Round(sub_layer_screen_position.x() * | 
 |                                                kPageScale * kDeviceScale), | 
 |                                MathUtil::Round(sub_layer_screen_position.y() * | 
 |                                                kPageScale * kDeviceScale)); | 
 |   expected_transform.Scale(kPageScale * kDeviceScale, | 
 |                            kPageScale * kDeviceScale); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_transform, | 
 |                                   sublayer->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_transform, | 
 |                                   sublayer->ScreenSpaceTransform()); | 
 |  | 
 |   gfx::Transform arbitrary_translate; | 
 |   const float kTranslateX = 10.6f; | 
 |   const float kTranslateY = 20.6f; | 
 |   arbitrary_translate.Translate(kTranslateX, kTranslateY); | 
 |   SetLayerPropertiesForTesting(scroll_layer, arbitrary_translate, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 20), | 
 |                                true, false, false); | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties( | 
 |       root.get(), kDeviceScale, kPageScale, scroll_layer->parent()); | 
 |   expected_transform.MakeIdentity(); | 
 |   expected_transform.Translate( | 
 |       MathUtil::Round(kTranslateX * kPageScale * kDeviceScale + | 
 |                       sub_layer_screen_position.x() * kPageScale * | 
 |                           kDeviceScale), | 
 |       MathUtil::Round(kTranslateY * kPageScale * kDeviceScale + | 
 |                       sub_layer_screen_position.y() * kPageScale * | 
 |                           kDeviceScale)); | 
 |   expected_transform.Scale(kPageScale * kDeviceScale, | 
 |                            kPageScale * kDeviceScale); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_transform, | 
 |                                   sublayer->DrawTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForSimpleHierarchy) { | 
 |   gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child = Layer::Create(layer_settings()); | 
 |   root->AddChild(parent); | 
 |   parent->AddChild(child); | 
 |   child->AddChild(grand_child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   // One-time setup of root layer | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false); | 
 |  | 
 |   TransformTree& tree = host()->property_trees()->transform_tree; | 
 |  | 
 |   // Case 1: parent's anchor point should not affect child or grand_child. | 
 |   SetLayerPropertiesForTesting(parent.get(), identity_matrix, | 
 |                                gfx::Point3F(2.5f, 3.0f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 12), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(16, 18), true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(76, 78), | 
 |                                true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, DrawTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, | 
 |       ScreenSpaceTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, DrawTransformFromPropertyTrees(grand_child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_matrix, | 
 |       ScreenSpaceTransformFromPropertyTrees(grand_child.get(), tree)); | 
 |  | 
 |   // Case 2: parent's position affects child and grand_child. | 
 |   gfx::Transform parent_position_transform; | 
 |   parent_position_transform.Translate(0.f, 1.2f); | 
 |   SetLayerPropertiesForTesting( | 
 |       parent.get(), identity_matrix, gfx::Point3F(2.5f, 3.0f, 0.f), | 
 |       gfx::PointF(0.f, 1.2f), gfx::Size(10, 12), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(16, 18), true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(76, 78), | 
 |                                true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_position_transform, | 
 |       DrawTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_position_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_position_transform, | 
 |       DrawTransformFromPropertyTrees(grand_child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_position_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(grand_child.get(), tree)); | 
 |  | 
 |   // Case 3: parent's local transform affects child and grandchild | 
 |   gfx::Transform parent_layer_transform; | 
 |   parent_layer_transform.Scale3d(2.0, 2.0, 1.0); | 
 |   gfx::Transform parent_translation_to_anchor; | 
 |   parent_translation_to_anchor.Translate(2.5, 3.0); | 
 |   gfx::Transform parent_composite_transform = | 
 |       parent_translation_to_anchor * parent_layer_transform * | 
 |       Inverse(parent_translation_to_anchor); | 
 |   SetLayerPropertiesForTesting(parent.get(), parent_layer_transform, | 
 |                                gfx::Point3F(2.5f, 3.0f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 12), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(16, 18), true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(76, 78), | 
 |                                true, false); | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_composite_transform, | 
 |       DrawTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_composite_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_composite_transform, | 
 |       DrawTransformFromPropertyTrees(grand_child.get(), tree)); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       parent_composite_transform, | 
 |       ScreenSpaceTransformFromPropertyTrees(grand_child.get(), tree)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForSingleRenderSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                true); | 
 |  | 
 |   // Child is set up so that a new render surface should be created. | 
 |   child->SetOpacity(0.5f); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform parent_layer_transform; | 
 |   parent_layer_transform.Scale3d(1.f, 0.9f, 1.f); | 
 |   gfx::Transform parent_translation_to_anchor; | 
 |   parent_translation_to_anchor.Translate(25.0, 30.0); | 
 |  | 
 |   gfx::Transform parent_composite_transform = | 
 |       parent_translation_to_anchor * parent_layer_transform * | 
 |       Inverse(parent_translation_to_anchor); | 
 |   gfx::Vector2dF parent_composite_scale = | 
 |       MathUtil::ComputeTransform2dScaleComponents(parent_composite_transform, | 
 |                                                   1.f); | 
 |   gfx::Transform surface_sublayer_transform; | 
 |   surface_sublayer_transform.Scale(parent_composite_scale.x(), | 
 |                                    parent_composite_scale.y()); | 
 |   gfx::Transform surface_sublayer_composite_transform = | 
 |       parent_composite_transform * Inverse(surface_sublayer_transform); | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, parent_layer_transform, | 
 |                                gfx::Point3F(25.0f, 30.0f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(100, 120), true, false, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(16, 18), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(8, 10), true, false, | 
 |                                false); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // Render surface should have been created now. | 
 |   ASSERT_TRUE(child->render_surface()); | 
 |   ASSERT_EQ(child, child->render_target()); | 
 |  | 
 |   // The child layer's draw transform should refer to its new render surface. | 
 |   // The screen-space transform, however, should still refer to the root. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(surface_sublayer_transform, | 
 |                                   child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(parent_composite_transform, | 
 |                                   child->ScreenSpaceTransform()); | 
 |  | 
 |   // Because the grand_child is the only drawable content, the child's render | 
 |   // surface will tighten its bounds to the grand_child.  The scale at which the | 
 |   // surface's subtree is drawn must be removed from the composite transform. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       surface_sublayer_composite_transform, | 
 |       child->render_target()->render_surface()->draw_transform()); | 
 |  | 
 |   // The screen space is the same as the target since the child surface draws | 
 |   // into the root. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       surface_sublayer_composite_transform, | 
 |       child->render_target()->render_surface()->screen_space_transform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsWhenCannotRenderToSeparateSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |  | 
 |   gfx::Transform parent_transform; | 
 |   parent_transform.Translate(10.0, 10.0); | 
 |  | 
 |   gfx::Transform child_transform; | 
 |   child_transform.Rotate(45.0); | 
 |  | 
 |   // child gets a render surface when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(parent, parent_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child, child_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.0, 2.0), gfx::Size(20, 20), true, | 
 |                                false, false); | 
 |  | 
 |   gfx::Transform expected_grand_child_screen_space_transform; | 
 |   expected_grand_child_screen_space_transform.Translate(10.0, 10.0); | 
 |   expected_grand_child_screen_space_transform.Rotate(45.0); | 
 |   expected_grand_child_screen_space_transform.Translate(2.0, 2.0); | 
 |  | 
 |   // First compute draw properties with separate surfaces enabled. | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // The grand child's draw transform should be its offset wrt the child. | 
 |   gfx::Transform expected_grand_child_draw_transform; | 
 |   expected_grand_child_draw_transform.Translate(2.0, 2.0); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_draw_transform, | 
 |                                   grand_child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_screen_space_transform, | 
 |                                   grand_child->ScreenSpaceTransform()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |  | 
 |   // With separate surfaces disabled, the grand child's draw transform should be | 
 |   // the same as its screen space transform. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_screen_space_transform, | 
 |                                   grand_child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_screen_space_transform, | 
 |                                   grand_child->ScreenSpaceTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForReplica) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |   scoped_ptr<LayerImpl> child_replica = | 
 |       LayerImpl::Create(host_impl()->active_tree(), 100); | 
 |  | 
 |   // One-time setup of root layer | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                true); | 
 |  | 
 |   // Child is set up so that a new render surface should be created. | 
 |   child->SetOpacity(0.5f); | 
 |  | 
 |   gfx::Transform parent_layer_transform; | 
 |   parent_layer_transform.Scale3d(2.0, 2.0, 1.0); | 
 |   gfx::Transform parent_translation_to_anchor; | 
 |   parent_translation_to_anchor.Translate(2.5, 3.0); | 
 |   gfx::Transform parent_composite_transform = | 
 |       parent_translation_to_anchor * parent_layer_transform * | 
 |       Inverse(parent_translation_to_anchor); | 
 |   gfx::Transform replica_layer_transform; | 
 |   replica_layer_transform.Scale3d(3.0, 3.0, 1.0); | 
 |   gfx::Vector2dF parent_composite_scale = | 
 |       MathUtil::ComputeTransform2dScaleComponents(parent_composite_transform, | 
 |                                                   1.f); | 
 |   gfx::Transform surface_sublayer_transform; | 
 |   surface_sublayer_transform.Scale(parent_composite_scale.x(), | 
 |                                    parent_composite_scale.y()); | 
 |   gfx::Transform replica_composite_transform = | 
 |       parent_composite_transform * replica_layer_transform * | 
 |       Inverse(surface_sublayer_transform); | 
 |   child_replica->SetDrawsContent(true); | 
 |   // Child's render surface should not exist yet. | 
 |   ASSERT_FALSE(child->render_surface()); | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, parent_layer_transform, | 
 |                                gfx::Point3F(2.5f, 3.0f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 12), true, false, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(16, 18), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(-0.5f, -0.5f), gfx::Size(1, 1), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child_replica.get(), replica_layer_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(), true, | 
 |                                false, false); | 
 |   child->SetReplicaLayer(std::move(child_replica)); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // Render surface should have been created now. | 
 |   ASSERT_TRUE(child->render_surface()); | 
 |   ASSERT_EQ(child, child->render_target()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       replica_composite_transform, | 
 |       child->render_target()->render_surface()->replica_draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(replica_composite_transform, | 
 |                                   child->render_target() | 
 |                                       ->render_surface() | 
 |                                       ->replica_screen_space_transform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForRenderSurfaceHierarchy) { | 
 |   // This test creates a more complex tree and verifies it all at once. This | 
 |   // covers the following cases: | 
 |   //   - layers that are described w.r.t. a render surface: should have draw | 
 |   //   transforms described w.r.t. that surface | 
 |   //   - A render surface described w.r.t. an ancestor render surface: should | 
 |   //   have a draw transform described w.r.t. that ancestor surface | 
 |   //   - Replicas of a render surface are described w.r.t. the replica's | 
 |   //   transform around its anchor, along with the surface itself. | 
 |   //   - Sanity check on recursion: verify transforms of layers described w.r.t. | 
 |   //   a render surface that is described w.r.t. an ancestor render surface. | 
 |   //   - verifying that each layer has a reference to the correct render surface | 
 |   //   and render target values. | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   parent->SetDrawsContent(true); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(parent); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   LayerImpl* child_of_root = AddChild<LayerImpl>(parent); | 
 |   child_of_root->SetDrawsContent(true); | 
 |   LayerImpl* child_of_rs1 = AddChild<LayerImpl>(render_surface1); | 
 |   child_of_rs1->SetDrawsContent(true); | 
 |   LayerImpl* child_of_rs2 = AddChild<LayerImpl>(render_surface2); | 
 |   child_of_rs2->SetDrawsContent(true); | 
 |   LayerImpl* grand_child_of_root = AddChild<LayerImpl>(child_of_root); | 
 |   grand_child_of_root->SetDrawsContent(true); | 
 |   LayerImpl* grand_child_of_rs1 = AddChild<LayerImpl>(child_of_rs1); | 
 |   grand_child_of_rs1->SetDrawsContent(true); | 
 |   LayerImpl* grand_child_of_rs2 = AddChild<LayerImpl>(child_of_rs2); | 
 |   grand_child_of_rs2->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> replica_of_rs1 = | 
 |       LayerImpl::Create(host_impl()->active_tree(), 101); | 
 |   scoped_ptr<LayerImpl> replica_of_rs2 = | 
 |       LayerImpl::Create(host_impl()->active_tree(), 102); | 
 |  | 
 |   // In combination with descendant draws content, opacity != 1 forces the layer | 
 |   // to have a new render surface. | 
 |   render_surface1->SetOpacity(0.5f); | 
 |   render_surface2->SetOpacity(0.33f); | 
 |  | 
 |   // One-time setup of root layer | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                true); | 
 |  | 
 |   // All layers in the tree are initialized with an anchor at .25 and a size of | 
 |   // (10,10).  matrix "A" is the composite layer transform used in all layers, | 
 |   // Matrix "R" is the composite replica transform used in all replica layers. | 
 |   gfx::Transform translation_to_anchor; | 
 |   translation_to_anchor.Translate(2.5, 0.0); | 
 |   gfx::Transform layer_transform; | 
 |   layer_transform.Translate(1.0, 1.0); | 
 |   gfx::Transform replica_layer_transform; | 
 |   replica_layer_transform.Scale3d(-2.0, 5.0, 1.0); | 
 |  | 
 |   gfx::Transform A = | 
 |       translation_to_anchor * layer_transform * Inverse(translation_to_anchor); | 
 |   gfx::Transform R = A * translation_to_anchor * replica_layer_transform * | 
 |                      Inverse(translation_to_anchor); | 
 |  | 
 |   gfx::Vector2dF surface1_parent_transform_scale = | 
 |       MathUtil::ComputeTransform2dScaleComponents(A, 1.f); | 
 |   gfx::Transform surface1_sublayer_transform; | 
 |   surface1_sublayer_transform.Scale(surface1_parent_transform_scale.x(), | 
 |                                     surface1_parent_transform_scale.y()); | 
 |  | 
 |   // SS1 = transform given to the subtree of render_surface1 | 
 |   gfx::Transform SS1 = surface1_sublayer_transform; | 
 |   // S1 = transform to move from render_surface1 pixels to the layer space of | 
 |   // the owning layer | 
 |   gfx::Transform S1 = Inverse(surface1_sublayer_transform); | 
 |  | 
 |   gfx::Vector2dF surface2_parent_transform_scale = | 
 |       MathUtil::ComputeTransform2dScaleComponents(SS1 * A, 1.f); | 
 |   gfx::Transform surface2_sublayer_transform; | 
 |   surface2_sublayer_transform.Scale(surface2_parent_transform_scale.x(), | 
 |                                     surface2_parent_transform_scale.y()); | 
 |  | 
 |   // SS2 = transform given to the subtree of render_surface2 | 
 |   gfx::Transform SS2 = surface2_sublayer_transform; | 
 |   // S2 = transform to move from render_surface2 pixels to the layer space of | 
 |   // the owning layer | 
 |   gfx::Transform S2 = Inverse(surface2_sublayer_transform); | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting(render_surface2, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting(child_of_root, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(child_of_rs1, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(child_of_rs2, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child_of_root, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child_of_rs1, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child_of_rs2, layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(replica_of_rs1.get(), replica_layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(), true, false, false); | 
 |   SetLayerPropertiesForTesting(replica_of_rs2.get(), replica_layer_transform, | 
 |                                gfx::Point3F(2.5f, 0.f, 0.f), gfx::PointF(), | 
 |                                gfx::Size(), true, false, false); | 
 |  | 
 |   render_surface1->SetReplicaLayer(std::move(replica_of_rs1)); | 
 |   render_surface2->SetReplicaLayer(std::move(replica_of_rs2)); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // Only layers that are associated with render surfaces should have an actual | 
 |   // RenderSurface() value. | 
 |   ASSERT_TRUE(root->render_surface()); | 
 |   ASSERT_FALSE(child_of_root->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_root->render_surface()); | 
 |  | 
 |   ASSERT_TRUE(render_surface1->render_surface()); | 
 |   ASSERT_FALSE(child_of_rs1->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_rs1->render_surface()); | 
 |  | 
 |   ASSERT_TRUE(render_surface2->render_surface()); | 
 |   ASSERT_FALSE(child_of_rs2->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_rs2->render_surface()); | 
 |  | 
 |   // Verify all render target accessors | 
 |   EXPECT_EQ(root, parent->render_target()); | 
 |   EXPECT_EQ(root, child_of_root->render_target()); | 
 |   EXPECT_EQ(root, grand_child_of_root->render_target()); | 
 |  | 
 |   EXPECT_EQ(render_surface1, render_surface1->render_target()); | 
 |   EXPECT_EQ(render_surface1, child_of_rs1->render_target()); | 
 |   EXPECT_EQ(render_surface1, grand_child_of_rs1->render_target()); | 
 |  | 
 |   EXPECT_EQ(render_surface2, render_surface2->render_target()); | 
 |   EXPECT_EQ(render_surface2, child_of_rs2->render_target()); | 
 |   EXPECT_EQ(render_surface2, grand_child_of_rs2->render_target()); | 
 |  | 
 |   // Verify layer draw transforms note that draw transforms are described with | 
 |   // respect to the nearest ancestor render surface but screen space transforms | 
 |   // are described with respect to the root. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A, parent->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A, child_of_root->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A, | 
 |                                   grand_child_of_root->DrawTransform()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS1, render_surface1->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS1 * A, child_of_rs1->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS1 * A * A, | 
 |                                   grand_child_of_rs1->DrawTransform()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS2, render_surface2->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS2 * A, child_of_rs2->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(SS2 * A * A, | 
 |                                   grand_child_of_rs2->DrawTransform()); | 
 |  | 
 |   // Verify layer screen-space transforms | 
 |   // | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A, parent->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A, child_of_root->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A, | 
 |                                   grand_child_of_root->ScreenSpaceTransform()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A, | 
 |                                   render_surface1->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A, | 
 |                                   child_of_rs1->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A * A, | 
 |                                   grand_child_of_rs1->ScreenSpaceTransform()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A, | 
 |                                   render_surface2->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A * A, | 
 |                                   child_of_rs2->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(A * A * A * A * A, | 
 |                                   grand_child_of_rs2->ScreenSpaceTransform()); | 
 |  | 
 |   // Verify render surface transforms. | 
 |   // | 
 |   // Draw transform of render surface 1 is described with respect to root. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * A * S1, render_surface1->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * R * S1, render_surface1->render_surface()->replica_draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * A * S1, render_surface1->render_surface()->screen_space_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * R * S1, | 
 |       render_surface1->render_surface()->replica_screen_space_transform()); | 
 |   // Draw transform of render surface 2 is described with respect to render | 
 |   // surface 1. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       SS1 * A * S2, render_surface2->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       SS1 * R * S2, | 
 |       render_surface2->render_surface()->replica_draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * A * A * S2, | 
 |       render_surface2->render_surface()->screen_space_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       A * A * R * S2, | 
 |       render_surface2->render_surface()->replica_screen_space_transform()); | 
 |  | 
 |   // Sanity check. If these fail there is probably a bug in the test itself.  It | 
 |   // is expected that we correctly set up transforms so that the y-component of | 
 |   // the screen-space transform encodes the "depth" of the layer in the tree. | 
 |   EXPECT_FLOAT_EQ(1.0, parent->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(2.0, | 
 |                   child_of_root->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       3.0, grand_child_of_root->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |  | 
 |   EXPECT_FLOAT_EQ(2.0, | 
 |                   render_surface1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(3.0, child_of_rs1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       4.0, grand_child_of_rs1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |  | 
 |   EXPECT_FLOAT_EQ(3.0, | 
 |                   render_surface2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(4.0, child_of_rs2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       5.0, grand_child_of_rs2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForFlatteningLayer) { | 
 |   // For layers that flatten their subtree, there should be an orthographic | 
 |   // projection (for x and y values) in the middle of the transform sequence. | 
 |   // Note that the way the code is currently implemented, it is not expected to | 
 |   // use a canonical orthographic projection. | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   child->SetDrawsContent(true); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |   LayerImpl* great_grand_child = AddChild<LayerImpl>(grand_child); | 
 |   great_grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform rotation_about_y_axis; | 
 |   rotation_about_y_axis.RotateAboutYAxis(30.0); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, rotation_about_y_axis, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, rotation_about_y_axis, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(great_grand_child, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, false); | 
 |  | 
 |   // No layers in this test should preserve 3d. | 
 |   ASSERT_TRUE(root->should_flatten_transform()); | 
 |   ASSERT_TRUE(child->should_flatten_transform()); | 
 |   ASSERT_TRUE(grand_child->should_flatten_transform()); | 
 |   ASSERT_TRUE(great_grand_child->should_flatten_transform()); | 
 |  | 
 |   gfx::Transform expected_child_draw_transform = rotation_about_y_axis; | 
 |   gfx::Transform expected_child_screen_space_transform = rotation_about_y_axis; | 
 |   gfx::Transform expected_grand_child_draw_transform = | 
 |       rotation_about_y_axis;  // draws onto child's render surface | 
 |   gfx::Transform flattened_rotation_about_y = rotation_about_y_axis; | 
 |   flattened_rotation_about_y.FlattenTo2d(); | 
 |   gfx::Transform expected_grand_child_screen_space_transform = | 
 |       flattened_rotation_about_y * rotation_about_y_axis; | 
 |   gfx::Transform expected_great_grand_child_draw_transform = | 
 |       flattened_rotation_about_y; | 
 |   gfx::Transform expected_great_grand_child_screen_space_transform = | 
 |       flattened_rotation_about_y * flattened_rotation_about_y; | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // The child's draw transform should have been taken by its surface. | 
 |   ASSERT_TRUE(child->render_surface()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_draw_transform, | 
 |                                   child->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_child_screen_space_transform, | 
 |       child->render_surface()->screen_space_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_screen_space_transform, | 
 |                                   child->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_draw_transform, | 
 |                                   grand_child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_grand_child_screen_space_transform, | 
 |                                   grand_child->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_great_grand_child_draw_transform, | 
 |                                   great_grand_child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_great_grand_child_screen_space_transform, | 
 |       great_grand_child->ScreenSpaceTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, LayerFullyContainedWithinClipInTargetSpace) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grand_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   gfx::Transform child_transform; | 
 |   child_transform.Translate(50.0, 50.0); | 
 |   child_transform.RotateAboutZAxis(30.0); | 
 |  | 
 |   gfx::Transform grand_child_transform; | 
 |   grand_child_transform.RotateAboutYAxis(90.0); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(200, 200), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), child_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), grand_child_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false); | 
 |  | 
 |   root->AddChild(child); | 
 |   child->AddChild(grand_child); | 
 |   grand_child->SetShouldFlattenTransform(false); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   // Mapping grand_child's bounds to target space produces a non-empty rect | 
 |   // that is fully contained within the target's bounds, so grand_child should | 
 |   // be considered fully visible. | 
 |   EXPECT_EQ(gfx::Rect(grand_child->bounds()), | 
 |             grand_child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformsForDegenerateIntermediateLayer) { | 
 |   // A layer that is empty in one axis, but not the other, was accidentally | 
 |   // skipping a necessary translation.  Without that translation, the coordinate | 
 |   // space of the layer's draw transform is incorrect. | 
 |   // | 
 |   // Normally this isn't a problem, because the layer wouldn't be drawn anyway, | 
 |   // but if that layer becomes a render surface, then its draw transform is | 
 |   // implicitly inherited by the rest of the subtree, which then is positioned | 
 |   // incorrectly as a result. | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(root); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   // The child height is zero, but has non-zero width that should be accounted | 
 |   // for while computing draw transforms. | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 0), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(child->has_render_surface()); | 
 |   // This is the real test, the rest are sanity checks. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   child->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   grand_child->DrawTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceWithSublayerScale) { | 
 |   const gfx::Transform identity_matrix; | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   gfx::Transform translate; | 
 |   translate.Translate3d(5, 5, 5); | 
 |   SetLayerPropertiesForTesting(render_surface, translate, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, translate, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child, translate, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   // render_surface will have a sublayer scale because of device scale factor. | 
 |   float device_scale_factor = 2.0f; | 
 |   LayerImplList render_surface_layer_list_impl; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root, root->bounds(), translate, &render_surface_layer_list_impl, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.device_scale_factor = device_scale_factor; | 
 |   inputs.property_trees->needs_rebuild = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // Between grand_child and render_surface, we translate by (10, 10) and scale | 
 |   // by a factor of 2. | 
 |   gfx::Vector2dF expected_translation(20.0f, 20.0f); | 
 |   EXPECT_EQ(grand_child->DrawTransform().To2dTranslation(), | 
 |             expected_translation); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformAboveRootLayer) { | 
 |   // Transformations applied at the root of the tree should be forwarded | 
 |   // to child layers instead of applied to the root RenderSurface. | 
 |   const gfx::Transform identity_matrix; | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetDrawsContent(true); | 
 |   LayerImpl* child = AddChild<LayerImpl>(root); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   child->SetScrollClipLayer(root->id()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |  | 
 |   gfx::Transform translate; | 
 |   translate.Translate(50, 50); | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), translate, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     EXPECT_EQ(translate, root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(translate, child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   gfx::Transform scale; | 
 |   scale.Scale(2, 2); | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), scale, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     EXPECT_EQ(scale, root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(scale, child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   gfx::Transform rotate; | 
 |   rotate.Rotate(2); | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), rotate, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     EXPECT_EQ(rotate, root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(rotate, child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   gfx::Transform composite; | 
 |   composite.ConcatTransform(translate); | 
 |   composite.ConcatTransform(scale); | 
 |   composite.ConcatTransform(rotate); | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), composite, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     EXPECT_EQ(composite, root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(composite, child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   // Verify it composes correctly with device scale. | 
 |   float device_scale_factor = 1.5f; | 
 |  | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), translate, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.device_scale_factor = device_scale_factor; | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     gfx::Transform device_scaled_translate = translate; | 
 |     device_scaled_translate.Scale(device_scale_factor, device_scale_factor); | 
 |     EXPECT_EQ(device_scaled_translate, | 
 |               root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(device_scaled_translate, | 
 |               child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   // Verify it composes correctly with page scale. | 
 |   float page_scale_factor = 2.f; | 
 |  | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), translate, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.page_scale_factor = page_scale_factor; | 
 |     inputs.page_scale_layer = root; | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     gfx::Transform page_scaled_translate = translate; | 
 |     page_scaled_translate.Scale(page_scale_factor, page_scale_factor); | 
 |     EXPECT_EQ(page_scaled_translate, | 
 |               root->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(page_scaled_translate, | 
 |               child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 |  | 
 |   // Verify that it composes correctly with transforms directly on root layer. | 
 |   root->SetTransform(composite); | 
 |  | 
 |   { | 
 |     LayerImplList render_surface_layer_list_impl; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root, root->bounds(), composite, &render_surface_layer_list_impl, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.property_trees->needs_rebuild = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |     gfx::Transform compositeSquared = composite; | 
 |     compositeSquared.ConcatTransform(composite); | 
 |     EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |         compositeSquared, root->draw_properties().target_space_transform); | 
 |     EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |         compositeSquared, child->draw_properties().target_space_transform); | 
 |     EXPECT_EQ(identity_matrix, root->render_surface()->draw_transform()); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        RenderSurfaceListForRenderSurfaceWithClippedLayer) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   parent->SetMasksToBounds(true); | 
 |   LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface1); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(30.f, 30.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   // The child layer's content is entirely outside the parent's clip rect, so | 
 |   // the intermediate render surface should not be listed here, even if it was | 
 |   // forced to be created. Render surfaces without children or visible content | 
 |   // are unexpected at draw time (e.g. we might try to create a content texture | 
 |   // of size 0). | 
 |   ASSERT_TRUE(parent->render_surface()); | 
 |   EXPECT_EQ(1U, render_surface_layer_list_impl()->size()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceListForTransparentChild) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface1); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   render_surface1->SetOpacity(0.f); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   parent->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       parent, parent->bounds(), &render_surface_layer_list, | 
 |       parent->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // Since the layer is transparent, render_surface1->render_surface() should | 
 |   // not have gotten added anywhere.  Also, the drawable content rect should not | 
 |   // have been extended by the children. | 
 |   ASSERT_TRUE(parent->render_surface()); | 
 |   EXPECT_EQ(0U, parent->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(1U, render_surface_layer_list.size()); | 
 |   EXPECT_EQ(parent->id(), render_surface_layer_list.at(0)->id()); | 
 |   EXPECT_EQ(gfx::Rect(), parent->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        RenderSurfaceListForTransparentChildWithBackgroundFilter) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface1); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   render_surface1->SetOpacity(0.f); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   child->SetDrawsContent(true); | 
 |   FilterOperations filters; | 
 |   filters.Append(FilterOperation::CreateBlurFilter(1.5f)); | 
 |   render_surface1->SetBackgroundFilters(filters); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   parent->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       parent, parent->bounds(), &render_surface_layer_list, | 
 |       parent->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // The layer is fully transparent, but has a background filter, so it | 
 |   // shouldn't be skipped. | 
 |   ASSERT_TRUE(parent->render_surface()); | 
 |   EXPECT_EQ(1U, parent->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(2U, render_surface_layer_list.size()); | 
 |   EXPECT_EQ(gfx::RectF(0, 0, 10, 10), | 
 |             parent->render_surface()->DrawableContentRect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceListForFilter) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   gfx::Transform scale_matrix; | 
 |   scale_matrix.Scale(2.0f, 2.0f); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(), true, false, true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(0, 0), gfx::Size(25, 25), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(25, 25), gfx::Size(25, 25), true, | 
 |                                false, true); | 
 |   FilterOperations filters; | 
 |   filters.Append(FilterOperation::CreateBlurFilter(10.0f)); | 
 |   parent->SetFilters(filters); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   parent->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root, root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   ASSERT_TRUE(parent->render_surface()); | 
 |   EXPECT_EQ(2U, parent->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(4U, render_surface_layer_list.size()); | 
 |   EXPECT_EQ(gfx::RectF(-29, -29, 158, 158), | 
 |             parent->render_surface()->DrawableContentRect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceForBlendMode) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   const SkXfermode::Mode blend_mode = SkXfermode::kMultiply_Mode; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |  | 
 |   child->SetBlendMode(blend_mode); | 
 |   child->SetOpacity(0.5f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   // Since the child layer has a blend mode other than normal, it should get | 
 |   // its own render surface. Also, layer's draw_properties should contain the | 
 |   // default blend mode, since the render surface becomes responsible for | 
 |   // applying the blend mode. | 
 |   ASSERT_TRUE(child->render_surface()); | 
 |   EXPECT_EQ(1.0f, child->draw_opacity()); | 
 |   EXPECT_EQ(0.5f, child->render_surface()->draw_opacity()); | 
 |   EXPECT_EQ(SkXfermode::kSrcOver_Mode, child->draw_blend_mode()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceDrawOpacity) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* not_surface = AddChild<LayerImpl>(surface1); | 
 |   LayerImpl* surface2 = AddChild<LayerImpl>(not_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(not_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   surface1->SetDrawsContent(true); | 
 |   surface2->SetDrawsContent(true); | 
 |  | 
 |   surface1->SetOpacity(0.5f); | 
 |   not_surface->SetOpacity(0.5f); | 
 |   surface2->SetOpacity(0.5f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(surface1->render_surface()); | 
 |   ASSERT_FALSE(not_surface->render_surface()); | 
 |   ASSERT_TRUE(surface2->render_surface()); | 
 |   EXPECT_EQ(0.5f, surface1->render_surface()->draw_opacity()); | 
 |   // surface2's draw opacity should include the opacity of not-surface and | 
 |   // itself, but not the opacity of surface1. | 
 |   EXPECT_EQ(0.25f, surface2->render_surface()->draw_opacity()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DrawOpacityWhenCannotRenderToSeparateSurface) { | 
 |   // Tests that when separate surfaces are disabled, a layer's draw opacity is | 
 |   // the product of all ancestor layer opacties and the layer's own opacity. | 
 |   // (Rendering will still be incorrect in situations where we really do need | 
 |   // surfaces to apply opacity, such as when we have overlapping layers with an | 
 |   // ancestor whose opacity is <1.) | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(child2); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   parent->SetDrawsContent(true); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   // child1 and grand_child get render surfaces when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |  | 
 |   child1->SetOpacity(0.5f); | 
 |   grand_child->SetOpacity(0.5f); | 
 |   leaf_node1->SetOpacity(0.5f); | 
 |   leaf_node2->SetOpacity(0.5f); | 
 |  | 
 |   // With surfaces enabled, each layer's draw opacity is the product of layer | 
 |   // opacities on the path from the layer to its render target, not including | 
 |   // the opacity of the layer that owns the target surface (since that opacity | 
 |   // is applied by the surface). | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(1.f, root->draw_opacity()); | 
 |   EXPECT_EQ(1.f, parent->draw_opacity()); | 
 |   EXPECT_EQ(1.f, child1->draw_opacity()); | 
 |   EXPECT_EQ(1.f, child2->draw_opacity()); | 
 |   EXPECT_EQ(1.f, grand_child->draw_opacity()); | 
 |   EXPECT_EQ(0.5f, leaf_node1->draw_opacity()); | 
 |   EXPECT_EQ(0.5f, leaf_node2->draw_opacity()); | 
 |  | 
 |   // With surfaces disabled, each layer's draw opacity is the product of layer | 
 |   // opacities on the path from the layer to the root. | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(1.f, root->draw_opacity()); | 
 |   EXPECT_EQ(1.f, parent->draw_opacity()); | 
 |   EXPECT_EQ(0.5f, child1->draw_opacity()); | 
 |   EXPECT_EQ(1.f, child2->draw_opacity()); | 
 |   EXPECT_EQ(0.25f, grand_child->draw_opacity()); | 
 |   EXPECT_EQ(0.125f, leaf_node1->draw_opacity()); | 
 |   EXPECT_EQ(0.5f, leaf_node2->draw_opacity()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ForceRenderSurface) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface1); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |  | 
 |   child->SetDrawsContent(true); | 
 |   render_surface1->SetForceRenderSurface(true); | 
 |  | 
 |   { | 
 |     ExecuteCalculateDrawPropertiesWithPropertyTrees(parent); | 
 |  | 
 |     // The root layer always creates a render surface | 
 |     EXPECT_TRUE(parent->has_render_surface()); | 
 |     EXPECT_TRUE(render_surface1->has_render_surface()); | 
 |   } | 
 |  | 
 |   { | 
 |     render_surface1->SetForceRenderSurface(false); | 
 |     render_surface1->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |     ExecuteCalculateDrawPropertiesWithPropertyTrees(parent); | 
 |     EXPECT_TRUE(parent->has_render_surface()); | 
 |     EXPECT_FALSE(render_surface1->has_render_surface()); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfacesFlattenScreenSpaceTransform) { | 
 |   // Render surfaces act as a flattening point for their subtree, so should | 
 |   // always flatten the target-to-screen space transform seen by descendants. | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |  | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform rotation_about_y_axis; | 
 |   rotation_about_y_axis.RotateAboutYAxis(30.0); | 
 |   // Make |parent| have a render surface. | 
 |   parent->SetOpacity(0.9f); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, rotation_about_y_axis, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   grand_child->SetShouldFlattenTransform(false); | 
 |  | 
 |   // Only grand_child should preserve 3d. | 
 |   EXPECT_TRUE(root->should_flatten_transform()); | 
 |   EXPECT_TRUE(parent->should_flatten_transform()); | 
 |   EXPECT_TRUE(child->should_flatten_transform()); | 
 |   EXPECT_FALSE(grand_child->should_flatten_transform()); | 
 |  | 
 |   gfx::Transform expected_child_draw_transform = identity_matrix; | 
 |   gfx::Transform expected_grand_child_draw_transform = identity_matrix; | 
 |  | 
 |   gfx::Transform flattened_rotation_about_y = rotation_about_y_axis; | 
 |   flattened_rotation_about_y.FlattenTo2d(); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(parent->render_surface()); | 
 |   EXPECT_FALSE(child->render_surface()); | 
 |   EXPECT_FALSE(grand_child->render_surface()); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_matrix, | 
 |                                   grand_child->DrawTransform()); | 
 |  | 
 |   // The screen-space transform inherited by |child| and |grand_child| should | 
 |   // have been flattened at their render target. In particular, the fact that | 
 |   // |grand_child| happens to preserve 3d shouldn't affect this flattening. | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(flattened_rotation_about_y, | 
 |                                   child->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(flattened_rotation_about_y, | 
 |                                   grand_child->ScreenSpaceTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipRectCullsRenderSurfaces) { | 
 |   // The entire subtree of layers that are outside the clip rect should be | 
 |   // culled away, and should not affect the render_surface_layer_list. | 
 |   // | 
 |   // The test tree is set up as follows: | 
 |   //  - all layers except the leaf_nodes are forced to be a new render surface | 
 |   //  that have something to draw. | 
 |   //  - parent is a large container layer. | 
 |   //  - child has masksToBounds=true to cause clipping. | 
 |   //  - grand_child is positioned outside of the child's bounds | 
 |   //  - great_grand_child is also kept outside child's bounds. | 
 |   // | 
 |   // In this configuration, grand_child and great_grand_child are completely | 
 |   // outside the clip rect, and they should never get scheduled on the list of | 
 |   // render surfaces. | 
 |  | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   LayerImpl* great_grand_child = AddChild<LayerImpl>(grand_child); | 
 |  | 
 |   // leaf_node1 ensures that parent and child are kept on the | 
 |   // render_surface_layer_list, even though grand_child and great_grand_child | 
 |   // should be clipped. | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(child); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(great_grand_child); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(45.f, 45.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(great_grand_child, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |  | 
 |   child->SetMasksToBounds(true); | 
 |   child->SetOpacity(0.4f); | 
 |   grand_child->SetOpacity(0.5f); | 
 |   great_grand_child->SetOpacity(0.4f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   ASSERT_EQ(2U, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(parent->id(), render_surface_layer_list_impl()->at(0)->id()); | 
 |   EXPECT_EQ(child->id(), render_surface_layer_list_impl()->at(1)->id()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipRectCullsSurfaceWithoutVisibleContent) { | 
 |   // When a render surface has a clip rect, it is used to clip the content rect | 
 |   // of the surface. | 
 |  | 
 |   // The test tree is set up as follows: | 
 |   //  - parent is a container layer that masksToBounds=true to cause clipping. | 
 |   //  - child is a render surface, which has a clip rect set to the bounds of | 
 |   //  the parent. | 
 |   //  - grand_child is a render surface, and the only visible content in child. | 
 |   //  It is positioned outside of the clip rect from parent. | 
 |  | 
 |   // In this configuration, grand_child should be outside the clipped | 
 |   // content rect of the child, making grand_child not appear in the | 
 |   // render_surface_layer_list. | 
 |  | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   LayerImpl* leaf_node = AddChild<LayerImpl>(grand_child); | 
 |   leaf_node->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(200.f, 200.f), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(leaf_node, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   parent->SetMasksToBounds(true); | 
 |   child->SetOpacity(0.4f); | 
 |   grand_child->SetOpacity(0.4f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   // We should cull child and grand_child from the | 
 |   // render_surface_layer_list. | 
 |   ASSERT_EQ(1U, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(parent->id(), render_surface_layer_list_impl()->at(0)->id()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, IsClippedIsSetCorrectlyLayerImpl) { | 
 |   // Tests that LayerImpl's IsClipped() property is set to true when: | 
 |   //  - the layer clips its subtree, e.g. masks to bounds, | 
 |   //  - the layer is clipped by an ancestor that contributes to the same | 
 |   //    render target, | 
 |   //  - a surface is clipped by an ancestor that contributes to the same | 
 |   //    render target. | 
 |   // | 
 |   // In particular, for a layer that owns a render surface: | 
 |   //  - the render surface inherits any clip from ancestors, and does NOT | 
 |   //    pass that clipped status to the layer itself. | 
 |   //  - but if the layer itself masks to bounds, it is considered clipped | 
 |   //    and propagates the clip to the subtree. | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   parent->SetDrawsContent(true); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   child1->SetDrawsContent(true); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   child2->SetDrawsContent(true); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child1); | 
 |   grand_child->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(child2); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |  | 
 |   // Case 1: nothing is clipped except the root render surface. | 
 |   { | 
 |     ExecuteCalculateDrawProperties(root); | 
 |  | 
 |     ASSERT_TRUE(root->render_surface()); | 
 |     ASSERT_TRUE(child2->render_surface()); | 
 |  | 
 |     EXPECT_FALSE(root->is_clipped()); | 
 |     EXPECT_TRUE(root->render_surface()->is_clipped()); | 
 |     EXPECT_FALSE(parent->is_clipped()); | 
 |     EXPECT_FALSE(child1->is_clipped()); | 
 |     EXPECT_FALSE(child2->is_clipped()); | 
 |     EXPECT_FALSE(child2->render_surface()->is_clipped()); | 
 |     EXPECT_FALSE(grand_child->is_clipped()); | 
 |     EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |     EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |   } | 
 |  | 
 |   // Case 2: parent masksToBounds, so the parent, child1, and child2's | 
 |   // surface are clipped. But layers that contribute to child2's surface are | 
 |   // not clipped explicitly because child2's surface already accounts for | 
 |   // that clip. | 
 |   { | 
 |     parent->SetMasksToBounds(true); | 
 |     host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |  | 
 |     ExecuteCalculateDrawProperties(root); | 
 |  | 
 |     ASSERT_TRUE(root->render_surface()); | 
 |     ASSERT_TRUE(child2->render_surface()); | 
 |  | 
 |     EXPECT_FALSE(root->is_clipped()); | 
 |     EXPECT_TRUE(root->render_surface()->is_clipped()); | 
 |     EXPECT_TRUE(parent->is_clipped()); | 
 |     EXPECT_TRUE(child1->is_clipped()); | 
 |     EXPECT_FALSE(child2->is_clipped()); | 
 |     EXPECT_TRUE(child2->render_surface()->is_clipped()); | 
 |     EXPECT_TRUE(grand_child->is_clipped()); | 
 |     EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |     EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |  | 
 |     parent->SetMasksToBounds(false); | 
 |   } | 
 |  | 
 |   // Case 3: child2 masksToBounds. The layer and subtree are clipped, and | 
 |   // child2's render surface is not clipped. | 
 |   { | 
 |     child2->SetMasksToBounds(true); | 
 |     host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |  | 
 |     ExecuteCalculateDrawProperties(root); | 
 |  | 
 |     ASSERT_TRUE(root->render_surface()); | 
 |     ASSERT_TRUE(child2->render_surface()); | 
 |  | 
 |     EXPECT_FALSE(root->is_clipped()); | 
 |     EXPECT_TRUE(root->render_surface()->is_clipped()); | 
 |     EXPECT_FALSE(parent->is_clipped()); | 
 |     EXPECT_FALSE(child1->is_clipped()); | 
 |     EXPECT_TRUE(child2->is_clipped()); | 
 |     EXPECT_FALSE(child2->render_surface()->is_clipped()); | 
 |     EXPECT_FALSE(grand_child->is_clipped()); | 
 |     EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |     EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, IsClippedWhenCannotRenderToSeparateSurface) { | 
 |   // Tests that when separate surfaces are disabled, is_clipped is true exactly | 
 |   // when a layer or its ancestor has a clip; in particular, if a layer | 
 |   // is_clipped, so is its entire subtree (since there are no render surfaces | 
 |   // that can reset is_clipped). | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(child2); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   parent->SetDrawsContent(true); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   // child1 and grand_child get render surfaces when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |  | 
 |   // Case 1: Nothing is clipped. In this case, is_clipped is always false, with | 
 |   // or without surfaces. | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |  | 
 |   // Case 2: The root is clipped. With surfaces, this only persists until the | 
 |   // next render surface. Without surfaces, the entire tree is clipped. | 
 |   root->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRUE(root->is_clipped()); | 
 |   EXPECT_TRUE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_TRUE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_TRUE(root->is_clipped()); | 
 |   EXPECT_TRUE(parent->is_clipped()); | 
 |   EXPECT_TRUE(child1->is_clipped()); | 
 |   EXPECT_TRUE(child2->is_clipped()); | 
 |   EXPECT_TRUE(grand_child->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |  | 
 |   root->SetMasksToBounds(false); | 
 |  | 
 |   // Case 3: The parent is clipped. Again, with surfaces, this only persists | 
 |   // until the next render surface. Without surfaces, parent's entire subtree is | 
 |   // clipped. | 
 |   parent->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_TRUE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_TRUE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_TRUE(parent->is_clipped()); | 
 |   EXPECT_TRUE(child1->is_clipped()); | 
 |   EXPECT_TRUE(child2->is_clipped()); | 
 |   EXPECT_TRUE(grand_child->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |  | 
 |   parent->SetMasksToBounds(false); | 
 |  | 
 |   // Case 4: child1 is clipped. With surfaces, only child1 is_clipped, since it | 
 |   // has no non-surface children. Without surfaces, child1's entire subtree is | 
 |   // clipped. | 
 |   child1->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_TRUE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node1->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_TRUE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_TRUE(grand_child->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |   EXPECT_FALSE(leaf_node2->is_clipped()); | 
 |  | 
 |   child1->SetMasksToBounds(false); | 
 |  | 
 |   // Case 5: Only the leaf nodes are clipped. The behavior with and without | 
 |   // surfaces is the same. | 
 |   leaf_node1->SetMasksToBounds(true); | 
 |   leaf_node2->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |   EXPECT_FALSE(parent->is_clipped()); | 
 |   EXPECT_FALSE(child1->is_clipped()); | 
 |   EXPECT_FALSE(child2->is_clipped()); | 
 |   EXPECT_FALSE(grand_child->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node1->is_clipped()); | 
 |   EXPECT_TRUE(leaf_node2->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DrawableContentRectForLayers) { | 
 |   // Verify that layers get the appropriate DrawableContentRect when their | 
 |   // parent masksToBounds is true. | 
 |   // | 
 |   //   grand_child1 - completely inside the region; DrawableContentRect should | 
 |   //   be the layer rect expressed in target space. | 
 |   //   grand_child2 - partially clipped but NOT masksToBounds; the clip rect | 
 |   //   will be the intersection of layer bounds and the mask region. | 
 |   //   grand_child3 - partially clipped and masksToBounds; the | 
 |   //   DrawableContentRect will still be the intersection of layer bounds and | 
 |   //   the mask region. | 
 |   //   grand_child4 - outside parent's clip rect; the DrawableContentRect should | 
 |   //   be empty. | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child1 = AddChild<LayerImpl>(child); | 
 |   grand_child1->SetDrawsContent(true); | 
 |   LayerImpl* grand_child2 = AddChild<LayerImpl>(child); | 
 |   grand_child2->SetDrawsContent(true); | 
 |   LayerImpl* grand_child3 = AddChild<LayerImpl>(child); | 
 |   grand_child3->SetDrawsContent(true); | 
 |   LayerImpl* grand_child4 = AddChild<LayerImpl>(child); | 
 |   grand_child4->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(grand_child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(15.f, 15.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(grand_child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(15.f, 15.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(grand_child4, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(45.f, 45.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |  | 
 |   child->SetMasksToBounds(true); | 
 |   grand_child3->SetMasksToBounds(true); | 
 |  | 
 |   // Force child to be a render surface. | 
 |   child->SetOpacity(0.4f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 10, 10), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(15, 15, 5, 5), grand_child3->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(15, 15, 5, 5), grand_child3->drawable_content_rect()); | 
 |   EXPECT_TRUE(grand_child4->drawable_content_rect().IsEmpty()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipRectIsPropagatedCorrectlyToSurfaces) { | 
 |   // Verify that render surfaces (and their layers) get the appropriate | 
 |   // clip rects when their parent masksToBounds is true. | 
 |   // | 
 |   // Layers that own render surfaces (at least for now) do not inherit any | 
 |   // clipping; instead the surface will enforce the clip for the entire subtree. | 
 |   // They may still have a clip rect of their own layer bounds, however, if | 
 |   // masksToBounds was true. | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child1 = AddChild<LayerImpl>(child); | 
 |   LayerImpl* grand_child2 = AddChild<LayerImpl>(child); | 
 |   LayerImpl* grand_child3 = AddChild<LayerImpl>(child); | 
 |   LayerImpl* grand_child4 = AddChild<LayerImpl>(child); | 
 |   // the leaf nodes ensure that these grand_children become render surfaces for | 
 |   // this test. | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child1); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(grand_child2); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node3 = AddChild<LayerImpl>(grand_child3); | 
 |   leaf_node3->SetDrawsContent(true); | 
 |   LayerImpl* leaf_node4 = AddChild<LayerImpl>(grand_child4); | 
 |   leaf_node4->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(grand_child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(10, 10), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(grand_child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(15.f, 15.f), gfx::Size(10, 10), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(grand_child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(15.f, 15.f), gfx::Size(10, 10), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(grand_child4, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(45.f, 45.f), gfx::Size(10, 10), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(leaf_node4, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   child->SetMasksToBounds(true); | 
 |   grand_child3->SetMasksToBounds(true); | 
 |   grand_child4->SetMasksToBounds(true); | 
 |  | 
 |   // Force everyone to be a render surface. | 
 |   child->SetOpacity(0.4f); | 
 |   grand_child1->SetOpacity(0.5f); | 
 |   grand_child2->SetOpacity(0.5f); | 
 |   grand_child3->SetOpacity(0.5f); | 
 |   grand_child4->SetOpacity(0.5f); | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   ASSERT_TRUE(grand_child1->render_surface()); | 
 |   ASSERT_TRUE(grand_child2->render_surface()); | 
 |   ASSERT_TRUE(grand_child3->render_surface()); | 
 |  | 
 |   // Surfaces are clipped by their parent, but un-affected by the owning layer's | 
 |   // masksToBounds. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 20, 20), | 
 |             grand_child1->render_surface()->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 20, 20), | 
 |             grand_child2->render_surface()->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 20, 20), | 
 |             grand_child3->render_surface()->clip_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipRectWhenCannotRenderToSeparateSurface) { | 
 |   // Tests that when separate surfaces are disabled, a layer's clip_rect is the | 
 |   // intersection of all ancestor clips in screen space; in particular, if a | 
 |   // layer masks to bounds, it contributes to the clip_rect of all layers in its | 
 |   // subtree (since there are no render surfaces that can reset the clip_rect). | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(child2); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   parent->SetDrawsContent(true); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   // child1 and grand_child get render surfaces when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(400, 400), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(4.f, 4.f), gfx::Size(800, 800), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(3.f, 3.f), gfx::Size(800, 800), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(8.f, 8.f), gfx::Size(1500, 1500), | 
 |                                true, false); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(16.f, 16.f), gfx::Size(2000, 2000), | 
 |                                true, false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(9.f, 9.f), gfx::Size(2000, 2000), | 
 |                                true, false); | 
 |  | 
 |   // Case 1: Nothing is clipped. In this case, each layer's clip rect is its | 
 |   // bounds in target space. The only thing that changes when surfaces are | 
 |   // disabled is that target space is always screen space. | 
 |   root->SetForceRenderSurface(true); | 
 |   child1->SetForceRenderSurface(true); | 
 |   grand_child->SetForceRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_FALSE(parent->has_render_surface()); | 
 |   EXPECT_TRUE(child1->has_render_surface()); | 
 |   EXPECT_FALSE(child2->has_render_surface()); | 
 |   EXPECT_TRUE(grand_child->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node1->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node2->has_render_surface()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 800, 800), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 2000, 2000), leaf_node2->clip_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 800, 800), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 800, 800), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 1500, 1500), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30, 2000, 2000), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 2000, 2000), leaf_node2->clip_rect()); | 
 |  | 
 |   // Case 2: The root is clipped. In this case, layers that draw into the root | 
 |   // render surface are clipped by the root's bounds. | 
 |   root->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetForceRenderSurface(true); | 
 |   child1->SetForceRenderSurface(true); | 
 |   grand_child->SetForceRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_FALSE(parent->has_render_surface()); | 
 |   EXPECT_TRUE(child1->has_render_surface()); | 
 |   EXPECT_FALSE(child2->has_render_surface()); | 
 |   EXPECT_TRUE(grand_child->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node1->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node2->has_render_surface()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), leaf_node2->clip_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), leaf_node2->clip_rect()); | 
 |  | 
 |   root->SetMasksToBounds(false); | 
 |  | 
 |   // Case 3: The parent and child1 are clipped. When surfaces are enabled, the | 
 |   // parent clip rect only contributes to the subtree rooted at child2, since | 
 |   // the subtree rooted at child1 renders into a separate surface. Similarly, | 
 |   // child1's clip rect doesn't contribute to its descendants, since its only | 
 |   // child is a render surface. However, without surfaces, these clip rects | 
 |   // contribute to all descendants. | 
 |   parent->SetMasksToBounds(true); | 
 |   child1->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetForceRenderSurface(true); | 
 |   child1->SetForceRenderSurface(true); | 
 |   grand_child->SetForceRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_FALSE(parent->has_render_surface()); | 
 |   EXPECT_TRUE(child1->has_render_surface()); | 
 |   EXPECT_FALSE(child2->has_render_surface()); | 
 |   EXPECT_TRUE(grand_child->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node1->has_render_surface()); | 
 |   EXPECT_FALSE(leaf_node2->has_render_surface()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), leaf_node2->clip_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 396, 396), child1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), child2->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 396, 396), grand_child->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 396, 396), leaf_node1->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), leaf_node2->clip_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SurfacesDisabledAndReEnabled) { | 
 |   // Tests that draw properties are computed correctly when we disable and then | 
 |   // re-enable separate surfaces. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   LayerImpl* leaf_node = AddChild<LayerImpl>(grand_child); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   parent->SetDrawsContent(true); | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   leaf_node->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   // child and grand_child get render surfaces when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(400, 400), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(4.f, 4.f), gfx::Size(800, 800), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(8.f, 8.f), gfx::Size(1500, 1500), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(leaf_node, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(16.f, 16.f), gfx::Size(2000, 2000), | 
 |                                true, false, false); | 
 |  | 
 |   parent->SetMasksToBounds(true); | 
 |   child->SetMasksToBounds(true); | 
 |  | 
 |   root->SetHasRenderSurface(true); | 
 |   child->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |  | 
 |   gfx::Transform expected_leaf_draw_transform_with_surfaces; | 
 |   expected_leaf_draw_transform_with_surfaces.Translate(16.0, 16.0); | 
 |  | 
 |   gfx::Transform expected_leaf_draw_transform_without_surfaces; | 
 |   expected_leaf_draw_transform_without_surfaces.Translate(30.0, 30.0); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node->drawable_content_rect()); | 
 |   EXPECT_EQ(expected_leaf_draw_transform_with_surfaces, | 
 |             leaf_node->DrawTransform()); | 
 |  | 
 |   root->SetHasRenderSurface(true); | 
 |   child->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 396, 396), leaf_node->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30, 372, 372), leaf_node->drawable_content_rect()); | 
 |   EXPECT_EQ(expected_leaf_draw_transform_without_surfaces, | 
 |             leaf_node->DrawTransform()); | 
 |  | 
 |   root->SetHasRenderSurface(true); | 
 |   child->SetHasRenderSurface(true); | 
 |   grand_child->SetHasRenderSurface(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node->drawable_content_rect()); | 
 |   EXPECT_EQ(expected_leaf_draw_transform_with_surfaces, | 
 |             leaf_node->DrawTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, AnimationsForRenderSurfaceHierarchy) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child_of_rs1 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* grand_child_of_rs1 = AddChild<LayerImpl>(child_of_rs1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* child_of_rs2 = AddChild<LayerImpl>(render_surface2); | 
 |   LayerImpl* grand_child_of_rs2 = AddChild<LayerImpl>(child_of_rs2); | 
 |   LayerImpl* child_of_root = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child_of_root = AddChild<LayerImpl>(child_of_root); | 
 |  | 
 |   parent->SetDrawsContent(true); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   child_of_rs1->SetDrawsContent(true); | 
 |   grand_child_of_rs1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   child_of_rs2->SetDrawsContent(true); | 
 |   grand_child_of_rs2->SetDrawsContent(true); | 
 |   child_of_root->SetDrawsContent(true); | 
 |   grand_child_of_root->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform layer_transform; | 
 |   layer_transform.Translate(1.0, 1.0); | 
 |  | 
 |   SetLayerPropertiesForTesting( | 
 |       parent, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting( | 
 |       render_surface1, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting( | 
 |       render_surface2, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting( | 
 |       child_of_root, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting( | 
 |       child_of_rs1, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting( | 
 |       child_of_rs2, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting( | 
 |       grand_child_of_root, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting( | 
 |       grand_child_of_rs1, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting( | 
 |       grand_child_of_rs2, layer_transform, gfx::Point3F(0.25f, 0.f, 0.f), | 
 |       gfx::PointF(2.5f, 0.f), gfx::Size(10, 10), true, false, false); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     // Put an animated opacity on the render surface. | 
 |     AddOpacityTransitionToLayerWithPlayer( | 
 |         render_surface1->id(), timeline_impl(), 10.0, 1.f, 0.f, false); | 
 |  | 
 |     // Also put an animated opacity on a layer without descendants. | 
 |     AddOpacityTransitionToLayerWithPlayer( | 
 |         grand_child_of_root->id(), timeline_impl(), 10.0, 1.f, 0.f, false); | 
 |  | 
 |     // Put a transform animation on the render surface. | 
 |     AddAnimatedTransformToLayerWithPlayer(render_surface2->id(), | 
 |                                           timeline_impl(), 10.0, 30, 0); | 
 |  | 
 |     // Also put transform animations on grand_child_of_root, and | 
 |     // grand_child_of_rs2 | 
 |     AddAnimatedTransformToLayerWithPlayer(grand_child_of_root->id(), | 
 |                                           timeline_impl(), 10.0, 30, 0); | 
 |     AddAnimatedTransformToLayerWithPlayer(grand_child_of_rs2->id(), | 
 |                                           timeline_impl(), 10.0, 30, 0); | 
 |   } else { | 
 |     // Put an animated opacity on the render surface. | 
 |     AddOpacityTransitionToController( | 
 |         render_surface1->layer_animation_controller(), 10.0, 1.f, 0.f, false); | 
 |  | 
 |     // Also put an animated opacity on a layer without descendants. | 
 |     AddOpacityTransitionToController( | 
 |         grand_child_of_root->layer_animation_controller(), 10.0, 1.f, 0.f, | 
 |         false); | 
 |  | 
 |     // Put a transform animation on the render surface. | 
 |     AddAnimatedTransformToController( | 
 |         render_surface2->layer_animation_controller(), 10.0, 30, 0); | 
 |  | 
 |     // Also put transform animations on grand_child_of_root, and | 
 |     // grand_child_of_rs2 | 
 |     AddAnimatedTransformToController( | 
 |         grand_child_of_root->layer_animation_controller(), 10.0, 30, 0); | 
 |     AddAnimatedTransformToController( | 
 |         grand_child_of_rs2->layer_animation_controller(), 10.0, 30, 0); | 
 |   } | 
 |  | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   // Only layers that are associated with render surfaces should have an actual | 
 |   // RenderSurface() value. | 
 |   ASSERT_TRUE(parent->render_surface()); | 
 |   ASSERT_FALSE(child_of_root->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_root->render_surface()); | 
 |  | 
 |   ASSERT_TRUE(render_surface1->render_surface()); | 
 |   ASSERT_FALSE(child_of_rs1->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_rs1->render_surface()); | 
 |  | 
 |   ASSERT_TRUE(render_surface2->render_surface()); | 
 |   ASSERT_FALSE(child_of_rs2->render_surface()); | 
 |   ASSERT_FALSE(grand_child_of_rs2->render_surface()); | 
 |  | 
 |   // Verify all render target accessors | 
 |   EXPECT_EQ(parent, parent->render_target()); | 
 |   EXPECT_EQ(parent, child_of_root->render_target()); | 
 |   EXPECT_EQ(parent, grand_child_of_root->render_target()); | 
 |  | 
 |   EXPECT_EQ(render_surface1, render_surface1->render_target()); | 
 |   EXPECT_EQ(render_surface1, child_of_rs1->render_target()); | 
 |   EXPECT_EQ(render_surface1, grand_child_of_rs1->render_target()); | 
 |  | 
 |   EXPECT_EQ(render_surface2, render_surface2->render_target()); | 
 |   EXPECT_EQ(render_surface2, child_of_rs2->render_target()); | 
 |   EXPECT_EQ(render_surface2, grand_child_of_rs2->render_target()); | 
 |  | 
 |   // Verify screen_space_transform_is_animating values | 
 |   EXPECT_FALSE(parent->screen_space_transform_is_animating()); | 
 |   EXPECT_FALSE(child_of_root->screen_space_transform_is_animating()); | 
 |   EXPECT_TRUE(grand_child_of_root->screen_space_transform_is_animating()); | 
 |   EXPECT_FALSE(render_surface1->screen_space_transform_is_animating()); | 
 |   EXPECT_FALSE(child_of_rs1->screen_space_transform_is_animating()); | 
 |   EXPECT_FALSE(grand_child_of_rs1->screen_space_transform_is_animating()); | 
 |   EXPECT_TRUE(render_surface2->screen_space_transform_is_animating()); | 
 |   EXPECT_TRUE(child_of_rs2->screen_space_transform_is_animating()); | 
 |   EXPECT_TRUE(grand_child_of_rs2->screen_space_transform_is_animating()); | 
 |  | 
 |   // Sanity check. If these fail there is probably a bug in the test itself. | 
 |   // It is expected that we correctly set up transforms so that the y-component | 
 |   // of the screen-space transform encodes the "depth" of the layer in the tree. | 
 |   EXPECT_FLOAT_EQ(1.0, parent->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(2.0, | 
 |                   child_of_root->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       3.0, grand_child_of_root->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |  | 
 |   EXPECT_FLOAT_EQ(2.0, | 
 |                   render_surface1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(3.0, child_of_rs1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       4.0, grand_child_of_rs1->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |  | 
 |   EXPECT_FLOAT_EQ(3.0, | 
 |                   render_surface2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ(4.0, child_of_rs2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 |   EXPECT_FLOAT_EQ( | 
 |       5.0, grand_child_of_rs2->ScreenSpaceTransform().matrix().get(1, 3)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, LargeTransforms) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |  | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform large_transform; | 
 |   large_transform.Scale(SkDoubleToMScalar(1e37), SkDoubleToMScalar(1e37)); | 
 |  | 
 |   gfx::Transform identity; | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting(child, large_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, large_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(), grand_child->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        ScreenSpaceTransformIsAnimatingWithDelayedAnimation) { | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   LayerImpl* great_grand_child = AddChild<LayerImpl>(grand_child); | 
 |  | 
 |   parent->SetDrawsContent(true); | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   great_grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity; | 
 |  | 
 |   SetLayerPropertiesForTesting(parent, identity, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, true); | 
 |   SetLayerPropertiesForTesting(child, identity, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(10, 10), true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(great_grand_child, identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   // Add a transform animation with a start delay to |grand_child|. | 
 |   scoped_ptr<Animation> animation = Animation::Create( | 
 |       scoped_ptr<AnimationCurve>(new FakeTransformTransition(1.0)), 0, 1, | 
 |       Animation::TRANSFORM); | 
 |   animation->set_fill_mode(Animation::FILL_MODE_NONE); | 
 |   animation->set_time_offset(base::TimeDelta::FromMilliseconds(-1000)); | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimationToLayerWithPlayer(grand_child->id(), timeline_impl(), | 
 |                                   std::move(animation)); | 
 |   } else { | 
 |     grand_child->layer_animation_controller()->AddAnimation( | 
 |         std::move(animation)); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(parent); | 
 |  | 
 |   EXPECT_FALSE(parent->screen_space_transform_is_animating()); | 
 |   EXPECT_FALSE(child->screen_space_transform_is_animating()); | 
 |  | 
 |   EXPECT_FALSE(grand_child->TransformIsAnimating()); | 
 |   EXPECT_TRUE(grand_child->HasPotentiallyRunningTransformAnimation()); | 
 |   EXPECT_TRUE(grand_child->screen_space_transform_is_animating()); | 
 |   EXPECT_TRUE(great_grand_child->screen_space_transform_is_animating()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectForIdentityTransform) { | 
 |   // Test the calculateVisibleRect() function works correctly for identity | 
 |   // transforms. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // Case 1: Layer is contained within the surface. | 
 |   gfx::Rect layer_content_rect = gfx::Rect(10, 10, 30, 30); | 
 |   gfx::Rect expected = gfx::Rect(10, 10, 30, 30); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 2: Layer is outside the surface rect. | 
 |   layer_content_rect = gfx::Rect(120, 120, 30, 30); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_TRUE(actual.IsEmpty()); | 
 |  | 
 |   // Case 3: Layer is partially overlapping the surface rect. | 
 |   layer_content_rect = gfx::Rect(80, 80, 30, 30); | 
 |   expected = gfx::Rect(80, 80, 20, 20); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectForTranslations) { | 
 |   // Test the calculateVisibleRect() function works correctly for scaling | 
 |   // transforms. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(0, 0, 30, 30); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // Case 1: Layer is contained within the surface. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(10.0, 10.0); | 
 |   gfx::Rect expected = gfx::Rect(0, 0, 30, 30); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 2: Layer is outside the surface rect. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(120.0, 120.0); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_TRUE(actual.IsEmpty()); | 
 |  | 
 |   // Case 3: Layer is partially overlapping the surface rect. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(80.0, 80.0); | 
 |   expected = gfx::Rect(0, 0, 20, 20); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectFor2DRotations) { | 
 |   // Test the calculateVisibleRect() function works correctly for rotations | 
 |   // about z-axis (i.e. 2D rotations).  Remember that calculateVisibleRect() | 
 |   // should return the g in the layer's space. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(0, 0, 30, 30); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // Case 1: Layer is contained within the surface. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(50.0, 50.0); | 
 |   layer_to_surface_transform.Rotate(45.0); | 
 |   gfx::Rect expected = gfx::Rect(0, 0, 30, 30); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 2: Layer is outside the surface rect. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(-50.0, 0.0); | 
 |   layer_to_surface_transform.Rotate(45.0); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_TRUE(actual.IsEmpty()); | 
 |  | 
 |   // Case 3: The layer is rotated about its top-left corner. In surface space, | 
 |   // the layer is oriented diagonally, with the left half outside of the render | 
 |   // surface. In this case, the g should still be the entire layer | 
 |   // (remember the g is computed in layer space); both the top-left | 
 |   // and bottom-right corners of the layer are still visible. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Rotate(45.0); | 
 |   expected = gfx::Rect(0, 0, 30, 30); | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 4: The layer is rotated about its top-left corner, and translated | 
 |   // upwards. In surface space, the layer is oriented diagonally, with only the | 
 |   // top corner of the surface overlapping the layer. In layer space, the render | 
 |   // surface overlaps the right side of the layer. The g should be | 
 |   // the layer's right half. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(0.0, -sqrt(2.0) * 15.0); | 
 |   layer_to_surface_transform.Rotate(45.0); | 
 |   expected = gfx::Rect(15, 0, 15, 30);  // Right half of layer bounds. | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectFor3dOrthographicTransform) { | 
 |   // Test that the calculateVisibleRect() function works correctly for 3d | 
 |   // transforms. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // Case 1: Orthographic projection of a layer rotated about y-axis by 45 | 
 |   // degrees, should be fully contained in the render surface. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.RotateAboutYAxis(45.0); | 
 |   gfx::Rect expected = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 2: Orthographic projection of a layer rotated about y-axis by 45 | 
 |   // degrees, but shifted to the side so only the right-half the layer would be | 
 |   // visible on the surface. | 
 |   // 100 is the un-rotated layer width; divided by sqrt(2) is the rotated width. | 
 |   SkMScalar half_width_of_rotated_layer = | 
 |       SkDoubleToMScalar((100.0 / sqrt(2.0)) * 0.5); | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(-half_width_of_rotated_layer, 0.0); | 
 |   layer_to_surface_transform.RotateAboutYAxis(45.0);  // Rotates about the left | 
 |                                                       // edge of the layer. | 
 |   expected = gfx::Rect(50, 0, 50, 100);  // Tight half of the layer. | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectFor3dPerspectiveTransform) { | 
 |   // Test the calculateVisibleRect() function works correctly when the layer has | 
 |   // a perspective projection onto the target surface. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(-50, -50, 200, 200); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // Case 1: Even though the layer is twice as large as the surface, due to | 
 |   // perspective foreshortening, the layer will fit fully in the surface when | 
 |   // its translated more than the perspective amount. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |  | 
 |   // The following sequence of transforms applies the perspective about the | 
 |   // center of the surface. | 
 |   layer_to_surface_transform.Translate(50.0, 50.0); | 
 |   layer_to_surface_transform.ApplyPerspectiveDepth(9.0); | 
 |   layer_to_surface_transform.Translate(-50.0, -50.0); | 
 |  | 
 |   // This translate places the layer in front of the surface's projection plane. | 
 |   layer_to_surface_transform.Translate3d(0.0, 0.0, -27.0); | 
 |  | 
 |   gfx::Rect expected = gfx::Rect(-50, -50, 200, 200); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Case 2: same projection as before, except that the layer is also translated | 
 |   // to the side, so that only the right half of the layer should be visible. | 
 |   // | 
 |   // Explanation of expected result: The perspective ratio is (z distance | 
 |   // between layer and camera origin) / (z distance between projection plane and | 
 |   // camera origin) == ((-27 - 9) / 9) Then, by similar triangles, if we want to | 
 |   // move a layer by translating -50 units in projected surface units (so that | 
 |   // only half of it is visible), then we would need to translate by (-36 / 9) * | 
 |   // -50 == -200 in the layer's units. | 
 |   layer_to_surface_transform.Translate3d(-200.0, 0.0, 0.0); | 
 |   expected = gfx::Rect(gfx::Point(50, -50), | 
 |                        gfx::Size(100, 200));  // The right half of the layer's | 
 |                                               // bounding rect. | 
 |   actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleRectFor3dOrthographicIsNotClippedBehindSurface) { | 
 |   // There is currently no explicit concept of an orthographic projection plane | 
 |   // in our code (nor in the CSS spec to my knowledge). Therefore, layers that | 
 |   // are technically behind the surface in an orthographic world should not be | 
 |   // clipped when they are flattened to the surface. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // This sequence of transforms effectively rotates the layer about the y-axis | 
 |   // at the center of the layer. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.Translate(50.0, 0.0); | 
 |   layer_to_surface_transform.RotateAboutYAxis(45.0); | 
 |   layer_to_surface_transform.Translate(-50.0, 0.0); | 
 |  | 
 |   gfx::Rect expected = gfx::Rect(0, 0, 100, 100); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectFor3dPerspectiveWhenClippedByW) { | 
 |   // Test the calculateVisibleRect() function works correctly when projecting a | 
 |   // surface onto a layer, but the layer is partially behind the camera (not | 
 |   // just behind the projection plane). In this case, the cartesian coordinates | 
 |   // may seem to be valid, but actually they are not. The visible rect needs to | 
 |   // be properly clipped by the w = 0 plane in homogeneous coordinates before | 
 |   // converting to cartesian coordinates. | 
 |  | 
 |   gfx::Rect target_surface_rect = gfx::Rect(-50, -50, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(-10, -1, 20, 2); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |  | 
 |   // The layer is positioned so that the right half of the layer should be in | 
 |   // front of the camera, while the other half is behind the surface's | 
 |   // projection plane. The following sequence of transforms applies the | 
 |   // perspective and rotation about the center of the layer. | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.ApplyPerspectiveDepth(1.0); | 
 |   layer_to_surface_transform.Translate3d(-2.0, 0.0, 1.0); | 
 |   layer_to_surface_transform.RotateAboutYAxis(45.0); | 
 |  | 
 |   // Sanity check that this transform does indeed cause w < 0 when applying the | 
 |   // transform, otherwise this code is not testing the intended scenario. | 
 |   bool clipped; | 
 |   MathUtil::MapQuad(layer_to_surface_transform, | 
 |                     gfx::QuadF(gfx::RectF(layer_content_rect)), | 
 |                     &clipped); | 
 |   ASSERT_TRUE(clipped); | 
 |  | 
 |   int expected_x_position = 0; | 
 |   int expected_width = 10; | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected_x_position, actual.x()); | 
 |   EXPECT_EQ(expected_width, actual.width()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectForPerspectiveUnprojection) { | 
 |   // To determine visible rect in layer space, there needs to be an | 
 |   // un-projection from surface space to layer space. When the original | 
 |   // transform was a perspective projection that was clipped, it returns a rect | 
 |   // that encloses the clipped bounds.  Un-projecting this new rect may require | 
 |   // clipping again. | 
 |  | 
 |   // This sequence of transforms causes one corner of the layer to protrude | 
 |   // across the w = 0 plane, and should be clipped. | 
 |   gfx::Rect target_surface_rect = gfx::Rect(-50, -50, 100, 100); | 
 |   gfx::Rect layer_content_rect = gfx::Rect(-10, -10, 20, 20); | 
 |   gfx::Transform layer_to_surface_transform; | 
 |   layer_to_surface_transform.MakeIdentity(); | 
 |   layer_to_surface_transform.ApplyPerspectiveDepth(1.0); | 
 |   layer_to_surface_transform.Translate3d(0.0, 0.0, -5.0); | 
 |   layer_to_surface_transform.RotateAboutYAxis(45.0); | 
 |   layer_to_surface_transform.RotateAboutXAxis(80.0); | 
 |  | 
 |   // Sanity check that un-projection does indeed cause w < 0, otherwise this | 
 |   // code is not testing the intended scenario. | 
 |   bool clipped; | 
 |   gfx::RectF clipped_rect = MathUtil::MapClippedRect( | 
 |       layer_to_surface_transform, gfx::RectF(layer_content_rect)); | 
 |   MathUtil::ProjectQuad( | 
 |       Inverse(layer_to_surface_transform), gfx::QuadF(clipped_rect), &clipped); | 
 |   ASSERT_TRUE(clipped); | 
 |  | 
 |   // Only the corner of the layer is not visible on the surface because of being | 
 |   // clipped. But, the net result of rounding visible region to an axis-aligned | 
 |   // rect is that the entire layer should still be considered visible. | 
 |   gfx::Rect expected = gfx::Rect(-10, -10, 20, 20); | 
 |   gfx::Rect actual = LayerTreeHostCommon::CalculateVisibleRect( | 
 |       target_surface_rect, layer_content_rect, layer_to_surface_transform); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleRectsForPositionedRootLayerClippedByViewport) { | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   // Root layer is positioned at (60, 70). The default device viewport size | 
 |   // is (0, 0, 100x100) in target space. So the root layer's visible rect | 
 |   // will be clipped by the viewport to be (0, 0, 40x30) in layer's space. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(60, 70), gfx::Size(100, 100), true, | 
 |                                false, true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |   // In target space, not clipped. | 
 |   EXPECT_EQ(gfx::Rect(60, 70, 100, 100), root->drawable_content_rect()); | 
 |   // In layer space, clipped. | 
 |   EXPECT_EQ(gfx::Rect(40, 30), root->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DrawableAndVisibleContentRectsForSimpleLayers) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child1_layer = AddChildToRoot<LayerImpl>(); | 
 |   child1_layer->SetDrawsContent(true); | 
 |   LayerImpl* child2_layer = AddChildToRoot<LayerImpl>(); | 
 |   child2_layer->SetDrawsContent(true); | 
 |   LayerImpl* child3_layer = AddChildToRoot<LayerImpl>(); | 
 |   child3_layer->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible_layer_rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |  | 
 |   // layer visible_layer_rects are clipped by their target surface. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1_layer->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 25, 25), child2_layer->visible_layer_rect()); | 
 |   EXPECT_TRUE(child3_layer->visible_layer_rect().IsEmpty()); | 
 |  | 
 |   // layer drawable_content_rects are not clipped. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1_layer->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(75, 75, 50, 50), child2_layer->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(125, 125, 50, 50), child3_layer->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsForLayersClippedByLayer) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child1 = AddChild<LayerImpl>(child); | 
 |   grand_child1->SetDrawsContent(true); | 
 |   LayerImpl* grand_child2 = AddChild<LayerImpl>(child); | 
 |   grand_child2->SetDrawsContent(true); | 
 |   LayerImpl* grand_child3 = AddChild<LayerImpl>(child); | 
 |   grand_child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(grand_child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(grand_child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   child->SetMasksToBounds(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible content rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), child->visible_layer_rect()); | 
 |  | 
 |   // All grandchild visible content rects should be clipped by child. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 25, 25), grand_child2->visible_layer_rect()); | 
 |   EXPECT_TRUE(grand_child3->visible_layer_rect().IsEmpty()); | 
 |  | 
 |   // All grandchild DrawableContentRects should also be clipped by child. | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 50, 50), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(75, 75, 25, 25), grand_child2->drawable_content_rect()); | 
 |   EXPECT_TRUE(grand_child3->drawable_content_rect().IsEmpty()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleContentRectWithClippingAndScaling) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grand_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   root->AddChild(child); | 
 |   child->AddChild(grand_child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform child_scale_matrix; | 
 |   child_scale_matrix.Scale(0.25f, 0.25f); | 
 |   gfx::Transform grand_child_scale_matrix; | 
 |   grand_child_scale_matrix.Scale(0.246f, 0.246f); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), child_scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), grand_child_scale_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false); | 
 |  | 
 |   child->SetMasksToBounds(true); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   // The visible rect is expanded to integer coordinates in target space before | 
 |   // being projected back to layer space, where it is once again expanded to | 
 |   // integer coordinates. | 
 |   EXPECT_EQ(gfx::Rect(49, 49), grand_child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsForLayersInUnclippedRenderSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface); | 
 |   child1->SetDrawsContent(true); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(render_surface); | 
 |   child2->SetDrawsContent(true); | 
 |   LayerImpl* child3 = AddChild<LayerImpl>(render_surface); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(render_surface->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible content rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), render_surface->visible_layer_rect()); | 
 |  | 
 |   // An unclipped surface grows its DrawableContentRect to include all drawable | 
 |   // regions of the subtree. | 
 |   EXPECT_EQ(gfx::RectF(5.f, 5.f, 170.f, 170.f), | 
 |             render_surface->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // All layers that draw content into the unclipped surface are also unclipped. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child3->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 50, 50), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(75, 75, 50, 50), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(125, 125, 50, 50), child3->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleRectsWhenCannotRenderToSeparateSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(parent); | 
 |   LayerImpl* grand_child1 = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* grand_child2 = AddChild<LayerImpl>(child2); | 
 |   LayerImpl* leaf_node1 = AddChild<LayerImpl>(grand_child1); | 
 |   LayerImpl* leaf_node2 = AddChild<LayerImpl>(grand_child2); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   parent->SetDrawsContent(true); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   grand_child1->SetDrawsContent(true); | 
 |   grand_child2->SetDrawsContent(true); | 
 |   leaf_node1->SetDrawsContent(true); | 
 |   leaf_node2->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   // child1 and child2 get render surfaces when surfaces are enabled. | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(400, 400), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(4.f, 4.f), gfx::Size(800, 800), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(3.f, 3.f), gfx::Size(800, 800), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(grand_child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(8.f, 8.f), gfx::Size(1500, 1500), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(grand_child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(7.f, 7.f), gfx::Size(1500, 1500), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(leaf_node1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(16.f, 16.f), gfx::Size(2000, 2000), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(leaf_node2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(9.f, 9.f), gfx::Size(2000, 2000), | 
 |                                true, false, false); | 
 |  | 
 |   // Case 1: No layers clip. Visible rects are clipped by the viewport, but the | 
 |   // viewport clip doesn't apply to layers that draw into unclipped surfaces. | 
 |   // Each layer's drawable content rect is its bounds in target space; the only | 
 |   // thing that changes with surfaces disabled is that target space is always | 
 |   // screen space. | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   child2->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2000, 2000), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2000, 2000), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(8, 8, 1500, 1500), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(7, 7, 1500, 1500), grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(24, 24, 2000, 2000), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node2->drawable_content_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(94, 94), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(95, 95), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(86, 86), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(88, 88), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(70, 70), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(79, 79), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 800, 800), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 800, 800), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 1500, 1500), | 
 |             grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(12, 12, 1500, 1500), | 
 |             grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30, 2000, 2000), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(21, 21, 2000, 2000), leaf_node2->drawable_content_rect()); | 
 |  | 
 |   // Case 2: The parent clips. In this case, neither surface is unclipped, so | 
 |   // all visible layer rects are clipped by the intersection of all ancestor | 
 |   // clips, whether or not surfaces are disabled. However, drawable content | 
 |   // rects are clipped only until the next render surface is reached, so | 
 |   // descendants of parent have their drawable content rects clipped only when | 
 |   // surfaces are disabled. | 
 |   parent->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   child2->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(94, 94), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(95, 95), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(86, 86), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(88, 88), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(70, 70), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(79, 79), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(8, 8, 1500, 1500), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(7, 7, 1500, 1500), grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(24, 24, 2000, 2000), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 2000, 2000), leaf_node2->drawable_content_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(94, 94), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(95, 95), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(86, 86), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(88, 88), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(70, 70), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(79, 79), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 396, 396), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 397, 397), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 388, 388), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(12, 12, 390, 390), grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30, 372, 372), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(21, 21, 381, 381), leaf_node2->drawable_content_rect()); | 
 |  | 
 |   parent->SetMasksToBounds(false); | 
 |  | 
 |   // Case 3: child1 and grand_child2 clip. In this case, descendants of these | 
 |   // layers have their visible rects clipped by them; without surfaces, these | 
 |   // rects are also clipped by the viewport. Similarly, descendants of these | 
 |   // layers have their drawable content rects clipped by them. | 
 |   child1->SetMasksToBounds(true); | 
 |   grand_child2->SetMasksToBounds(true); | 
 |   host_impl()->active_tree()->property_trees()->needs_rebuild = true; | 
 |   root->SetHasRenderSurface(true); | 
 |   child1->SetHasRenderSurface(true); | 
 |   child2->SetHasRenderSurface(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(792, 792), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1500, 1500), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(776, 776), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(1491, 1491), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(800, 800), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(8, 8, 792, 792), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(7, 7, 1500, 1500), grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(24, 24, 776, 776), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(16, 16, 1491, 1491), leaf_node2->drawable_content_rect()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithoutSeparateSurfaces(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(98, 98), parent->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(94, 94), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(95, 95), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(86, 86), grand_child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(88, 88), grand_child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(70, 70), leaf_node1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(79, 79), leaf_node2->visible_layer_rect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(100, 100), root->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 400, 400), parent->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(6, 6, 800, 800), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 800, 800), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(14, 14, 792, 792), grand_child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(12, 12, 1500, 1500), | 
 |             grand_child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30, 776, 776), leaf_node1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(21, 21, 1491, 1491), leaf_node2->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleContentRectsForClippedSurfaceWithEmptyClip) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(root); | 
 |   LayerImpl* child3 = AddChild<LayerImpl>(root); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   LayerImplList render_surface_layer_list_impl; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   // Now set the root render surface an empty clip. | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root, gfx::Size(), &render_surface_layer_list_impl, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |  | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |   ASSERT_TRUE(root->render_surface()); | 
 |   EXPECT_FALSE(root->is_clipped()); | 
 |  | 
 |   gfx::Rect empty; | 
 |   EXPECT_EQ(empty, root->render_surface()->clip_rect()); | 
 |   EXPECT_TRUE(root->render_surface()->is_clipped()); | 
 |  | 
 |   // Visible content rect calculation will check if the target surface is | 
 |   // clipped or not. An empty clip rect does not indicate the render surface | 
 |   // is unclipped. | 
 |   EXPECT_EQ(empty, child1->visible_layer_rect()); | 
 |   EXPECT_EQ(empty, child2->visible_layer_rect()); | 
 |   EXPECT_EQ(empty, child3->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsForLayersWithUninvertibleTransform) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   // Case 1: a truly degenerate matrix | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform uninvertible_matrix(0.0, 0.0, 0.0, 0.0, 0.0, 0.0); | 
 |   ASSERT_FALSE(uninvertible_matrix.IsInvertible()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, uninvertible_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(child->visible_layer_rect().IsEmpty()); | 
 |   EXPECT_TRUE(child->drawable_content_rect().IsEmpty()); | 
 |  | 
 |   // Case 2: a matrix with flattened z, uninvertible and not visible according | 
 |   // to the CSS spec. | 
 |   uninvertible_matrix.MakeIdentity(); | 
 |   uninvertible_matrix.matrix().set(2, 2, 0.0); | 
 |   ASSERT_FALSE(uninvertible_matrix.IsInvertible()); | 
 |  | 
 |   SetLayerPropertiesForTesting(child, uninvertible_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(child->visible_layer_rect().IsEmpty()); | 
 |   EXPECT_TRUE(child->drawable_content_rect().IsEmpty()); | 
 |  | 
 |   // Case 3: a matrix with flattened z, also uninvertible and not visible. | 
 |   uninvertible_matrix.MakeIdentity(); | 
 |   uninvertible_matrix.Translate(500.0, 0.0); | 
 |   uninvertible_matrix.matrix().set(2, 2, 0.0); | 
 |   ASSERT_FALSE(uninvertible_matrix.IsInvertible()); | 
 |  | 
 |   SetLayerPropertiesForTesting(child, uninvertible_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(child->visible_layer_rect().IsEmpty()); | 
 |   EXPECT_TRUE(child->drawable_content_rect().IsEmpty()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleContentRectForLayerWithUninvertibleDrawTransform) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   gfx::Transform perspective; | 
 |   perspective.ApplyPerspectiveDepth(SkDoubleToMScalar(1e-12)); | 
 |  | 
 |   gfx::Transform rotation; | 
 |   rotation.RotateAboutYAxis(45.0); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, perspective, gfx::Point3F(), | 
 |                                gfx::PointF(10.f, 10.f), gfx::Size(100, 100), | 
 |                                false, true, false); | 
 |   SetLayerPropertiesForTesting(grand_child, rotation, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // Though all layers have invertible transforms, matrix multiplication using | 
 |   // floating-point math makes the draw transform uninvertible. | 
 |   EXPECT_FALSE(grand_child->DrawTransform().IsInvertible()); | 
 |  | 
 |   // CalcDrawProps only skips a subtree when a layer's own transform is | 
 |   // uninvertible, not when its draw transform is invertible, since CDP makes | 
 |   // skipping decisions before computing a layer's draw transform. Property | 
 |   // trees make skipping decisions after computing draw transforms, so could be | 
 |   // made to skip layers with an uninvertible draw transform (once CDP is | 
 |   // deleted). | 
 |   EXPECT_EQ(gfx::Rect(grand_child->bounds()), | 
 |             grand_child->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        OcclusionForLayerWithUninvertibleDrawTransform) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   scoped_ptr<OutputSurface> output_surface = FakeOutputSurface::Create3d(); | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.active_tree(), 2); | 
 |   scoped_ptr<LayerImpl> grand_child = | 
 |       LayerImpl::Create(host_impl.active_tree(), 3); | 
 |   scoped_ptr<LayerImpl> occluding_child = | 
 |       LayerImpl::Create(host_impl.active_tree(), 4); | 
 |   child->SetDrawsContent(true); | 
 |   grand_child->SetDrawsContent(true); | 
 |   occluding_child->SetDrawsContent(true); | 
 |   occluding_child->SetContentsOpaque(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform perspective; | 
 |   perspective.ApplyPerspectiveDepth(SkDoubleToMScalar(1e-12)); | 
 |  | 
 |   gfx::Transform rotation; | 
 |   rotation.RotateAboutYAxis(45.0); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1000, 1000), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(child.get(), perspective, gfx::Point3F(), | 
 |                                gfx::PointF(10.f, 10.f), gfx::Size(300, 300), | 
 |                                false, true, false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), rotation, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(200, 200), false, true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(occluding_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(200, 200), false, false, false); | 
 |  | 
 |   host_impl.SetViewportSize(root->bounds()); | 
 |  | 
 |   child->AddChild(std::move(grand_child)); | 
 |   root->AddChild(std::move(child)); | 
 |   root->AddChild(std::move(occluding_child)); | 
 |   host_impl.active_tree()->SetRootLayer(std::move(root)); | 
 |   host_impl.SetVisible(true); | 
 |   host_impl.InitializeRenderer(output_surface.get()); | 
 |   host_impl.active_tree()->BuildPropertyTreesForTesting(); | 
 |   bool update_lcd_text = false; | 
 |   host_impl.active_tree()->UpdateDrawProperties(update_lcd_text); | 
 |  | 
 |   LayerImpl* grand_child_ptr = | 
 |       host_impl.active_tree()->root_layer()->children()[0]->children()[0].get(); | 
 |  | 
 |   // Though all layers have invertible transforms, matrix multiplication using | 
 |   // floating-point math makes the draw transform uninvertible. | 
 |   EXPECT_FALSE(grand_child_ptr->DrawTransform().IsInvertible()); | 
 |  | 
 |   // Since |grand_child| has an uninvertible draw transform, it is treated as | 
 |   // unoccluded (even though |occluding_child| comes later in draw order, and | 
 |   // hence potentially occludes it). | 
 |   gfx::Rect layer_bounds = gfx::Rect(grand_child_ptr->bounds()); | 
 |   EXPECT_EQ( | 
 |       layer_bounds, | 
 |       grand_child_ptr->draw_properties() | 
 |           .occlusion_in_content_space.GetUnoccludedContentRect(layer_bounds)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        SingularTransformDoesNotPreventClearingDrawProperties) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   root->AddChild(child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform uninvertible_matrix(0.0, 0.0, 0.0, 0.0, 0.0, 0.0); | 
 |   ASSERT_FALSE(uninvertible_matrix.IsInvertible()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), | 
 |                                uninvertible_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), | 
 |                                gfx::Size(50, 50), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   child->set_sorted_for_recursion(true); | 
 |  | 
 |   TransformOperations start_transform_operations; | 
 |   start_transform_operations.AppendScale(1.f, 0.f, 0.f); | 
 |  | 
 |   TransformOperations end_transform_operations; | 
 |   end_transform_operations.AppendScale(1.f, 1.f, 0.f); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(root->id(), timeline(), 10.0, | 
 |                                           start_transform_operations, | 
 |                                           end_transform_operations); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(root.get(), 10.0, start_transform_operations, | 
 |                                 end_transform_operations); | 
 |   } | 
 |   EXPECT_TRUE(root->TransformIsAnimating()); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_FALSE(child->sorted_for_recursion()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        SingularNonAnimatingTransformDoesNotPreventClearingDrawProperties) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform uninvertible_matrix(0.0, 0.0, 0.0, 0.0, 0.0, 0.0); | 
 |   ASSERT_FALSE(uninvertible_matrix.IsInvertible()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), | 
 |                                uninvertible_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   root->set_sorted_for_recursion(true); | 
 |  | 
 |   EXPECT_FALSE(root->TransformIsAnimating()); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_FALSE(root->sorted_for_recursion()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsForLayersInClippedRenderSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface); | 
 |   child1->SetDrawsContent(true); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(render_surface); | 
 |   child2->SetDrawsContent(true); | 
 |   LayerImpl* child3 = AddChild<LayerImpl>(render_surface); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(render_surface->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible content rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), render_surface->visible_layer_rect()); | 
 |  | 
 |   // A clipped surface grows its DrawableContentRect to include all drawable | 
 |   // regions of the subtree, but also gets clamped by the ancestor's clip. | 
 |   EXPECT_EQ(gfx::RectF(5.f, 5.f, 95.f, 95.f), | 
 |             render_surface->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // All layers that draw content into the surface have their visible content | 
 |   // rect clipped by the surface clip rect. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 25, 25), child2->visible_layer_rect()); | 
 |   EXPECT_TRUE(child3->visible_layer_rect().IsEmpty()); | 
 |  | 
 |   // But the DrawableContentRects are unclipped. | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 50, 50), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(75, 75, 50, 50), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(125, 125, 50, 50), child3->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsForSurfaceHierarchy) { | 
 |   // Check that clipping does not propagate down surfaces. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface2); | 
 |   child1->SetDrawsContent(true); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(render_surface2); | 
 |   child2->SetDrawsContent(true); | 
 |   LayerImpl* child3 = AddChild<LayerImpl>(render_surface2); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(7, 13), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(render_surface1->render_surface()); | 
 |   ASSERT_TRUE(render_surface2->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible content rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), render_surface1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), render_surface2->visible_layer_rect()); | 
 |  | 
 |   // A clipped surface grows its DrawableContentRect to include all drawable | 
 |   // regions of the subtree, but also gets clamped by the ancestor's clip. | 
 |   EXPECT_EQ(gfx::RectF(5.f, 5.f, 95.f, 95.f), | 
 |             render_surface1->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // render_surface1 lives in the "unclipped universe" of render_surface1, and | 
 |   // is only implicitly clipped by render_surface1's content rect. So, | 
 |   // render_surface2 grows to enclose all drawable content of its subtree. | 
 |   EXPECT_EQ(gfx::RectF(5.f, 5.f, 170.f, 170.f), | 
 |             render_surface2->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // All layers that draw content into render_surface2 think they are unclipped. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child3->visible_layer_rect()); | 
 |  | 
 |   // DrawableContentRects are also unclipped. | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 50, 50), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(75, 75, 50, 50), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(125, 125, 50, 50), child3->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleRectsForClippedDescendantsOfUnclippedSurfaces) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(child2); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(500, 500), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(700, 700), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1000, 1000), true, | 
 |                                false, true); | 
 |  | 
 |   child1->SetMasksToBounds(true); | 
 |   child2->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(500, 500), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), render_surface2->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleRectsWhenClipChildIsBetweenTwoRenderSurfaces) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   render_surface1->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(20, 20), render_surface1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), clip_child->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), render_surface2->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipRectOfSurfaceWhoseParentIsAClipChild) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   render_surface1->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), render_surface2->render_surface()->clip_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceContentRectWhenLayerNotDrawn) { | 
 |   // Test that only drawn layers contribute to render surface content rect. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(surface); | 
 |  | 
 |   test_layer->SetDrawsContent(false); | 
 |   surface->SetDrawsContent(true); | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(200, 200), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(test_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(150, 150), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), surface->render_surface()->content_rect()); | 
 |  | 
 |   test_layer->SetDrawsContent(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(150, 150), surface->render_surface()->content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectsMultipleSurfaces) { | 
 |   // Tests visible rects computation when we have unclipped_surface-> | 
 |   // surface_with_unclipped_descendants->clipped_surface, checks that the bounds | 
 |   // of surface_with_unclipped_descendants doesn't propagate to the | 
 |   // clipped_surface below it. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* unclipped_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(unclipped_surface); | 
 |   LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface); | 
 |   LayerImpl* clipped_surface = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   unclipped_surface->SetDrawsContent(true); | 
 |   unclipped_desc_surface->SetDrawsContent(true); | 
 |   clipped_surface->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(unclipped_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clipped_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(30, 30), unclipped_surface->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(20, 20), unclipped_desc_surface->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), clipped_surface->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RootClipPropagationToClippedSurface) { | 
 |   // Tests visible rects computation when we have unclipped_surface-> | 
 |   // surface_with_unclipped_descendants->clipped_surface, checks that the bounds | 
 |   // of root propagate to the clipped_surface. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* unclipped_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(unclipped_surface); | 
 |   LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface); | 
 |   LayerImpl* clipped_surface = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   unclipped_surface->SetDrawsContent(true); | 
 |   unclipped_desc_surface->SetDrawsContent(true); | 
 |   clipped_surface->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(unclipped_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clipped_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   unclipped_desc_surface->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), unclipped_surface->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), unclipped_desc_surface->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), clipped_surface->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsWithTransformOnUnclippedSurface) { | 
 |   // Layers that have non-axis aligned bounds (due to transforms) have an | 
 |   // expanded, axis-aligned DrawableContentRect and visible content rect. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface); | 
 |   child1->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform child_rotation; | 
 |   child_rotation.Rotate(45.0); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting( | 
 |       child1, child_rotation, gfx::Point3F(25, 25, 0.f), | 
 |       gfx::PointF(25.f, 25.f), gfx::Size(50, 50), true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(render_surface->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::RectF(100.f, 100.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // Layers that do not draw content should have empty visible content rects. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), render_surface->visible_layer_rect()); | 
 |  | 
 |   // The unclipped surface grows its DrawableContentRect to include all drawable | 
 |   // regions of the subtree. | 
 |   int diagonal_radius = ceil(sqrt(2.0) * 25.0); | 
 |   gfx::Rect expected_surface_drawable_content = | 
 |       gfx::Rect(50 - diagonal_radius, | 
 |                 50 - diagonal_radius, | 
 |                 diagonal_radius * 2, | 
 |                 diagonal_radius * 2); | 
 |   EXPECT_EQ(gfx::RectF(expected_surface_drawable_content), | 
 |             render_surface->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // All layers that draw content into the unclipped surface are also unclipped. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(expected_surface_drawable_content, child1->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        DrawableAndVisibleContentRectsWithTransformOnClippedSurface) { | 
 |   // Layers that have non-axis aligned bounds (due to transforms) have an | 
 |   // expanded, axis-aligned DrawableContentRect and visible content rect. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface); | 
 |   child1->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform child_rotation; | 
 |   child_rotation.Rotate(45.0); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(3, 4), true, false, | 
 |                                true); | 
 |  | 
 |   SetLayerPropertiesForTesting( | 
 |       child1, child_rotation, gfx::Point3F(25, 25, 0.f), | 
 |       gfx::PointF(25.f, 25.f), gfx::Size(50, 50), true, false, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(render_surface->render_surface()); | 
 |  | 
 |   // The clipped surface clamps the DrawableContentRect that encloses the | 
 |   // rotated layer. | 
 |   int diagonal_radius = ceil(sqrt(2.0) * 25.0); | 
 |   gfx::Rect unclipped_surface_content = gfx::Rect(50 - diagonal_radius, | 
 |                                                   50 - diagonal_radius, | 
 |                                                   diagonal_radius * 2, | 
 |                                                   diagonal_radius * 2); | 
 |   gfx::RectF expected_surface_drawable_content( | 
 |       gfx::IntersectRects(unclipped_surface_content, gfx::Rect(50, 50))); | 
 |   EXPECT_EQ(expected_surface_drawable_content, | 
 |             render_surface->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // On the clipped surface, only a quarter  of the child1 is visible, but when | 
 |   // rotating it back to  child1's content space, the actual enclosing rect ends | 
 |   // up covering the full left half of child1. | 
 |   // | 
 |   // Given the floating point math, this number is a little bit fuzzy. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 26, 50), child1->visible_layer_rect()); | 
 |  | 
 |   // The child's DrawableContentRect is unclipped. | 
 |   EXPECT_EQ(unclipped_surface_content, child1->drawable_content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DrawableAndVisibleContentRectsInHighDPI) { | 
 |   LayerImpl* root = root_layer(); | 
 |   FakePictureLayerImpl* render_surface1 = | 
 |       AddChildToRoot<FakePictureLayerImpl>(); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   FakePictureLayerImpl* render_surface2 = | 
 |       AddChild<FakePictureLayerImpl>(render_surface1); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   FakePictureLayerImpl* child1 = | 
 |       AddChild<FakePictureLayerImpl>(render_surface2); | 
 |   child1->SetDrawsContent(true); | 
 |   FakePictureLayerImpl* child2 = | 
 |       AddChild<FakePictureLayerImpl>(render_surface2); | 
 |   child2->SetDrawsContent(true); | 
 |   FakePictureLayerImpl* child3 = | 
 |       AddChild<FakePictureLayerImpl>(render_surface2); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(3, 4), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(7, 13), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(5.f, 5.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(75.f, 75.f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(125.f, 125.f), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |  | 
 |   float device_scale_factor = 2.f; | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root, device_scale_factor); | 
 |  | 
 |   ASSERT_TRUE(render_surface1->render_surface()); | 
 |   ASSERT_TRUE(render_surface2->render_surface()); | 
 |  | 
 |   // drawable_content_rects for all layers and surfaces are scaled by | 
 |   // device_scale_factor. | 
 |   EXPECT_EQ(gfx::RectF(200.f, 200.f), | 
 |             root->render_surface()->DrawableContentRect()); | 
 |   EXPECT_EQ(gfx::RectF(10.f, 10.f, 190.f, 190.f), | 
 |             render_surface1->render_surface()->DrawableContentRect()); | 
 |  | 
 |   // render_surface2 lives in the "unclipped universe" of render_surface1, and | 
 |   // is only implicitly clipped by render_surface1. | 
 |   EXPECT_EQ(gfx::RectF(10.f, 10.f, 350.f, 350.f), | 
 |             render_surface2->render_surface()->DrawableContentRect()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(10, 10, 100, 100), child1->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(150, 150, 100, 100), child2->drawable_content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(250, 250, 100, 100), child3->drawable_content_rect()); | 
 |  | 
 |   // The root layer does not actually draw content of its own. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0), root->visible_layer_rect()); | 
 |  | 
 |   // All layer visible content rects are not expressed in content space of each | 
 |   // layer, so they are not scaled by the device_scale_factor. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 3, 4), render_surface1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 7, 13), render_surface2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child2->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), child3->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, BackFaceCullingWithoutPreserves3d) { | 
 |   // Verify the behavior of back-face culling when there are no preserve-3d | 
 |   // layers. Note that 3d transforms still apply in this case, but they are | 
 |   // "flattened" to each parent layer according to current W3C spec. | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> front_facing_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> back_facing_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> front_facing_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> back_facing_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> | 
 |       front_facing_child_of_front_facing_surface = | 
 |           make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> | 
 |       back_facing_child_of_front_facing_surface = | 
 |           make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> | 
 |       front_facing_child_of_back_facing_surface = | 
 |           make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> | 
 |       back_facing_child_of_back_facing_surface = | 
 |           make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   parent->AddChild(front_facing_child); | 
 |   parent->AddChild(back_facing_child); | 
 |   parent->AddChild(front_facing_surface); | 
 |   parent->AddChild(back_facing_surface); | 
 |   front_facing_surface->AddChild(front_facing_child_of_front_facing_surface); | 
 |   front_facing_surface->AddChild(back_facing_child_of_front_facing_surface); | 
 |   back_facing_surface->AddChild(front_facing_child_of_back_facing_surface); | 
 |   back_facing_surface->AddChild(back_facing_child_of_back_facing_surface); | 
 |  | 
 |   host()->SetRootLayer(parent); | 
 |  | 
 |   // Nothing is double-sided | 
 |   front_facing_child->SetDoubleSided(false); | 
 |   back_facing_child->SetDoubleSided(false); | 
 |   front_facing_surface->SetDoubleSided(false); | 
 |   back_facing_surface->SetDoubleSided(false); | 
 |   front_facing_child_of_front_facing_surface->SetDoubleSided(false); | 
 |   back_facing_child_of_front_facing_surface->SetDoubleSided(false); | 
 |   front_facing_child_of_back_facing_surface->SetDoubleSided(false); | 
 |   back_facing_child_of_back_facing_surface->SetDoubleSided(false); | 
 |  | 
 |   gfx::Transform backface_matrix; | 
 |   backface_matrix.Translate(50.0, 50.0); | 
 |   backface_matrix.RotateAboutYAxis(180.0); | 
 |   backface_matrix.Translate(-50.0, -50.0); | 
 |  | 
 |   // Having a descendant and opacity will force these to have render surfaces. | 
 |   front_facing_surface->SetOpacity(0.5f); | 
 |   back_facing_surface->SetOpacity(0.5f); | 
 |  | 
 |   // Nothing preserves 3d. According to current W3C CSS gfx::Transforms spec, | 
 |   // these layers should blindly use their own local transforms to determine | 
 |   // back-face culling. | 
 |   SetLayerPropertiesForTesting(parent.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(front_facing_child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(back_facing_child.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(front_facing_surface.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(back_facing_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(front_facing_child_of_front_facing_surface.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(back_facing_child_of_front_facing_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(front_facing_child_of_back_facing_surface.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(back_facing_child_of_back_facing_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(parent.get()); | 
 |  | 
 |   // Verify which render surfaces were created. | 
 |   EXPECT_FALSE(front_facing_child->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child->has_render_surface()); | 
 |   EXPECT_TRUE(front_facing_surface->has_render_surface()); | 
 |   EXPECT_TRUE(back_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE( | 
 |       front_facing_child_of_front_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child_of_front_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(front_facing_child_of_back_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child_of_back_facing_surface->has_render_surface()); | 
 |  | 
 |   EXPECT_EQ(4u, update_layer_list().size()); | 
 |   EXPECT_TRUE(UpdateLayerListContains(front_facing_child->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains(front_facing_surface->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains( | 
 |       front_facing_child_of_front_facing_surface->id())); | 
 |   EXPECT_TRUE( | 
 |       UpdateLayerListContains(front_facing_child_of_back_facing_surface->id())); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, BackFaceCullingWithPreserves3d) { | 
 |   // Verify the behavior of back-face culling when preserves-3d transform style | 
 |   // is used. | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   LayerImpl* parent = root_layer(); | 
 |   LayerImpl* front_facing_child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* back_facing_child = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* front_facing_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* back_facing_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* front_facing_child_of_front_facing_surface = | 
 |       AddChild<LayerImpl>(front_facing_surface); | 
 |   LayerImpl* back_facing_child_of_front_facing_surface = | 
 |       AddChild<LayerImpl>(front_facing_surface); | 
 |   LayerImpl* front_facing_child_of_back_facing_surface = | 
 |       AddChild<LayerImpl>(back_facing_surface); | 
 |   LayerImpl* back_facing_child_of_back_facing_surface = | 
 |       AddChild<LayerImpl>(back_facing_surface); | 
 |   // Opacity will not force creation of render surfaces in this case because of | 
 |   // the preserve-3d transform style. Instead, an example of when a surface | 
 |   // would be created with preserve-3d is when there is a replica layer. | 
 |   LayerImpl* dummy_replica_layer1 = | 
 |       AddReplicaLayer<LayerImpl>(front_facing_surface); | 
 |   LayerImpl* dummy_replica_layer2 = | 
 |       AddReplicaLayer<LayerImpl>(back_facing_surface); | 
 |  | 
 |   // Nothing is double-sided | 
 |   front_facing_child->SetDoubleSided(false); | 
 |   back_facing_child->SetDoubleSided(false); | 
 |   front_facing_surface->SetDoubleSided(false); | 
 |   back_facing_surface->SetDoubleSided(false); | 
 |   front_facing_child_of_front_facing_surface->SetDoubleSided(false); | 
 |   back_facing_child_of_front_facing_surface->SetDoubleSided(false); | 
 |   front_facing_child_of_back_facing_surface->SetDoubleSided(false); | 
 |   back_facing_child_of_back_facing_surface->SetDoubleSided(false); | 
 |  | 
 |   // Everything draws content. | 
 |   front_facing_child->SetDrawsContent(true); | 
 |   back_facing_child->SetDrawsContent(true); | 
 |   front_facing_surface->SetDrawsContent(true); | 
 |   back_facing_surface->SetDrawsContent(true); | 
 |   front_facing_child_of_front_facing_surface->SetDrawsContent(true); | 
 |   back_facing_child_of_front_facing_surface->SetDrawsContent(true); | 
 |   front_facing_child_of_back_facing_surface->SetDrawsContent(true); | 
 |   back_facing_child_of_back_facing_surface->SetDrawsContent(true); | 
 |   dummy_replica_layer1->SetDrawsContent(true); | 
 |   dummy_replica_layer2->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform backface_matrix; | 
 |   backface_matrix.Translate(50.0, 50.0); | 
 |   backface_matrix.RotateAboutYAxis(180.0); | 
 |   backface_matrix.Translate(-50.0, -50.0); | 
 |  | 
 |   // Each surface creates its own new 3d rendering context (as defined by W3C | 
 |   // spec).  According to current W3C CSS gfx::Transforms spec, layers in a 3d | 
 |   // rendering context should use the transform with respect to that context. | 
 |   // This 3d rendering context occurs when (a) parent's transform style is flat | 
 |   // and (b) the layer's transform style is preserve-3d. | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, | 
 |                                false);  // parent transform style is flat. | 
 |   SetLayerPropertiesForTesting(front_facing_child, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(back_facing_child, backface_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false); | 
 |   // surface transform style is preserve-3d. | 
 |   SetLayerPropertiesForTesting(front_facing_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), false, true); | 
 |   // surface transform style is preserve-3d. | 
 |   SetLayerPropertiesForTesting(back_facing_surface, backface_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), false, true); | 
 |   SetLayerPropertiesForTesting(front_facing_child_of_front_facing_surface, | 
 |                                identity_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, true); | 
 |   SetLayerPropertiesForTesting(back_facing_child_of_front_facing_surface, | 
 |                                backface_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, true); | 
 |   SetLayerPropertiesForTesting(front_facing_child_of_back_facing_surface, | 
 |                                identity_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, true); | 
 |   SetLayerPropertiesForTesting(back_facing_child_of_back_facing_surface, | 
 |                                backface_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, true); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(parent); | 
 |  | 
 |   // Verify which render surfaces were created and used. | 
 |   EXPECT_FALSE(front_facing_child->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child->has_render_surface()); | 
 |   EXPECT_TRUE(front_facing_surface->has_render_surface()); | 
 |   // We expect that a has_render_surface was created but not used. | 
 |   EXPECT_TRUE(back_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE( | 
 |       front_facing_child_of_front_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child_of_front_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(front_facing_child_of_back_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(back_facing_child_of_back_facing_surface->has_render_surface()); | 
 |  | 
 |   EXPECT_EQ(3u, update_layer_list_impl()->size()); | 
 |  | 
 |   EXPECT_TRUE(UpdateLayerListImplContains(front_facing_child->id())); | 
 |   EXPECT_TRUE(UpdateLayerListImplContains(front_facing_surface->id())); | 
 |   EXPECT_TRUE(UpdateLayerListImplContains( | 
 |       front_facing_child_of_front_facing_surface->id())); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, BackFaceCullingWithAnimatingTransforms) { | 
 |   // Verify that layers are appropriately culled when their back face is showing | 
 |   // and they are not double sided, while animations are going on. | 
 |   // | 
 |   // Layers that are animating do not get culled on the main thread, as their | 
 |   // transforms should be treated as "unknown" so we can not be sure that their | 
 |   // back face is really showing. | 
 |   const gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> animating_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child_of_animating_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> animating_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child2 = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   parent->AddChild(child); | 
 |   parent->AddChild(animating_surface); | 
 |   animating_surface->AddChild(child_of_animating_surface); | 
 |   parent->AddChild(animating_child); | 
 |   parent->AddChild(child2); | 
 |  | 
 |   host()->SetRootLayer(parent); | 
 |  | 
 |   // Nothing is double-sided | 
 |   child->SetDoubleSided(false); | 
 |   child2->SetDoubleSided(false); | 
 |   animating_surface->SetDoubleSided(false); | 
 |   child_of_animating_surface->SetDoubleSided(false); | 
 |   animating_child->SetDoubleSided(false); | 
 |  | 
 |   gfx::Transform backface_matrix; | 
 |   backface_matrix.Translate(50.0, 50.0); | 
 |   backface_matrix.RotateAboutYAxis(180.0); | 
 |   backface_matrix.Translate(-50.0, -50.0); | 
 |  | 
 |   // Make our render surface. | 
 |   animating_surface->SetForceRenderSurface(true); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     // Animate the transform on the render surface. | 
 |     AddAnimatedTransformToLayerWithPlayer(animating_surface->id(), timeline(), | 
 |                                           10.0, 30, 0); | 
 |     // This is just an animating layer, not a surface. | 
 |     AddAnimatedTransformToLayerWithPlayer(animating_child->id(), timeline(), | 
 |                                           10.0, 30, 0); | 
 |   } else { | 
 |     // Animate the transform on the render surface. | 
 |     AddAnimatedTransformToController( | 
 |         animating_surface->layer_animation_controller(), 10.0, 30, 0); | 
 |     // This is just an animating layer, not a surface. | 
 |     AddAnimatedTransformToController( | 
 |         animating_child->layer_animation_controller(), 10.0, 30, 0); | 
 |   } | 
 |  | 
 |   SetLayerPropertiesForTesting(parent.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(animating_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child_of_animating_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(animating_child.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   LayerTreeHostCommon::CalcDrawPropsMainInputs inputs(parent.get(), | 
 |                                                       parent->bounds()); | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   EXPECT_FALSE(child->has_render_surface()); | 
 |   EXPECT_TRUE(animating_surface->has_render_surface()); | 
 |   EXPECT_FALSE(child_of_animating_surface->has_render_surface()); | 
 |   EXPECT_FALSE(animating_child->has_render_surface()); | 
 |   EXPECT_FALSE(child2->has_render_surface()); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(parent.get()); | 
 |  | 
 |   EXPECT_EQ(4u, update_layer_list().size()); | 
 |  | 
 |   // The non-animating child be culled from the layer list for the parent render | 
 |   // surface. | 
 |   EXPECT_TRUE(UpdateLayerListContains(animating_surface->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains(animating_child->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains(child2->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains(child_of_animating_surface->id())); | 
 |  | 
 |   EXPECT_FALSE(child2->visible_rect_from_property_trees().IsEmpty()); | 
 |  | 
 |   // The animating layers should have a visible content rect that represents the | 
 |   // area of the front face that is within the viewport. | 
 |   EXPECT_EQ(animating_child->visible_rect_from_property_trees(), | 
 |             gfx::Rect(animating_child->bounds())); | 
 |   EXPECT_EQ(animating_surface->visible_rect_from_property_trees(), | 
 |             gfx::Rect(animating_surface->bounds())); | 
 |   // And layers in the subtree of the animating layer should have valid visible | 
 |   // content rects also. | 
 |   EXPECT_EQ(child_of_animating_surface->visible_rect_from_property_trees(), | 
 |             gfx::Rect(child_of_animating_surface->bounds())); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |      BackFaceCullingWithPreserves3dForFlatteningSurface) { | 
 |   // Verify the behavior of back-face culling for a render surface that is | 
 |   // created when it flattens its subtree, and its parent has preserves-3d. | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> front_facing_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> back_facing_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child1 = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child2 = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   parent->AddChild(front_facing_surface); | 
 |   parent->AddChild(back_facing_surface); | 
 |   front_facing_surface->AddChild(child1); | 
 |   back_facing_surface->AddChild(child2); | 
 |  | 
 |   host()->SetRootLayer(parent); | 
 |  | 
 |   // RenderSurfaces are not double-sided | 
 |   front_facing_surface->SetDoubleSided(false); | 
 |   back_facing_surface->SetDoubleSided(false); | 
 |  | 
 |   gfx::Transform backface_matrix; | 
 |   backface_matrix.Translate(50.0, 50.0); | 
 |   backface_matrix.RotateAboutYAxis(180.0); | 
 |   backface_matrix.Translate(-50.0, -50.0); | 
 |  | 
 |   SetLayerPropertiesForTesting(parent.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                false, | 
 |                                true);  // parent transform style is preserve3d. | 
 |   SetLayerPropertiesForTesting(front_facing_surface.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                true);  // surface transform style is flat. | 
 |   SetLayerPropertiesForTesting(back_facing_surface.get(), | 
 |                                backface_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                true);  // surface transform style is flat. | 
 |   SetLayerPropertiesForTesting(child1.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   front_facing_surface->Set3dSortingContextId(1); | 
 |   back_facing_surface->Set3dSortingContextId(1); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(parent.get()); | 
 |  | 
 |   // Verify which render surfaces were created and used. | 
 |   EXPECT_TRUE(front_facing_surface->has_render_surface()); | 
 |  | 
 |   // We expect the render surface to have been created, but remain unused. | 
 |   EXPECT_TRUE(back_facing_surface->has_render_surface()); | 
 |   EXPECT_FALSE(child1->has_render_surface()); | 
 |   EXPECT_FALSE(child2->has_render_surface()); | 
 |  | 
 |   EXPECT_EQ(2u, update_layer_list().size()); | 
 |   EXPECT_TRUE(UpdateLayerListContains(front_facing_surface->id())); | 
 |   EXPECT_TRUE(UpdateLayerListContains(child1->id())); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonScalingTest, LayerTransformsInHighDPI) { | 
 |   // Verify draw and screen space transforms of layers not in a surface. | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   LayerImpl* parent = root_layer(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                true); | 
 |  | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   child->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(10, 10), false, | 
 |                                true, false); | 
 |  | 
 |   LayerImpl* child2 = AddChildToRoot<LayerImpl>(); | 
 |   child2->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(5, 5), false, | 
 |                                true, false); | 
 |  | 
 |   float device_scale_factor = 2.5f; | 
 |   gfx::Size viewport_size(100, 100); | 
 |   ExecuteCalculateDrawProperties(parent, device_scale_factor); | 
 |  | 
 |   EXPECT_IDEAL_SCALE_EQ(device_scale_factor, parent); | 
 |   EXPECT_IDEAL_SCALE_EQ(device_scale_factor, child); | 
 |   EXPECT_IDEAL_SCALE_EQ(device_scale_factor, child2); | 
 |  | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   // Verify parent transforms | 
 |   gfx::Transform expected_parent_transform; | 
 |   expected_parent_transform.Scale(device_scale_factor, device_scale_factor); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_parent_transform, | 
 |                                   parent->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_parent_transform, | 
 |                                   parent->DrawTransform()); | 
 |  | 
 |   // Verify results of transformed parent rects | 
 |   gfx::RectF parent_bounds(gfx::SizeF(parent->bounds())); | 
 |  | 
 |   gfx::RectF parent_draw_rect = | 
 |       MathUtil::MapClippedRect(parent->DrawTransform(), parent_bounds); | 
 |   gfx::RectF parent_screen_space_rect = | 
 |       MathUtil::MapClippedRect(parent->ScreenSpaceTransform(), parent_bounds); | 
 |  | 
 |   gfx::RectF expected_parent_draw_rect(gfx::SizeF(parent->bounds())); | 
 |   expected_parent_draw_rect.Scale(device_scale_factor); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_parent_draw_rect, parent_draw_rect); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_parent_draw_rect, parent_screen_space_rect); | 
 |  | 
 |   // Verify child and child2 transforms. They should match. | 
 |   gfx::Transform expected_child_transform; | 
 |   expected_child_transform.Scale(device_scale_factor, device_scale_factor); | 
 |   expected_child_transform.Translate(child->position().x(), | 
 |                                      child->position().y()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_transform, | 
 |                                   child->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_transform, | 
 |                                   child->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_transform, | 
 |                                   child2->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_transform, | 
 |                                   child2->ScreenSpaceTransform()); | 
 |  | 
 |   // Verify results of transformed child and child2 rects. They should | 
 |   // match. | 
 |   gfx::RectF child_bounds(gfx::SizeF(child->bounds())); | 
 |  | 
 |   gfx::RectF child_draw_rect = | 
 |       MathUtil::MapClippedRect(child->DrawTransform(), child_bounds); | 
 |   gfx::RectF child_screen_space_rect = | 
 |       MathUtil::MapClippedRect(child->ScreenSpaceTransform(), child_bounds); | 
 |  | 
 |   gfx::RectF child2_draw_rect = | 
 |       MathUtil::MapClippedRect(child2->DrawTransform(), child_bounds); | 
 |   gfx::RectF child2_screen_space_rect = | 
 |       MathUtil::MapClippedRect(child2->ScreenSpaceTransform(), child_bounds); | 
 |  | 
 |   gfx::RectF expected_child_draw_rect(child->position(), | 
 |                                       gfx::SizeF(child->bounds())); | 
 |   expected_child_draw_rect.Scale(device_scale_factor); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_child_draw_rect, child_draw_rect); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_child_draw_rect, child_screen_space_rect); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_child_draw_rect, child2_draw_rect); | 
 |   EXPECT_FLOAT_RECT_EQ(expected_child_draw_rect, child2_screen_space_rect); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonScalingTest, SurfaceLayerTransformsInHighDPI) { | 
 |   // Verify draw and screen space transforms of layers in a surface. | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform perspective_matrix; | 
 |   perspective_matrix.ApplyPerspectiveDepth(2.0); | 
 |  | 
 |   gfx::Transform scale_small_matrix; | 
 |   scale_small_matrix.Scale(SK_MScalar1 / 10.f, SK_MScalar1 / 12.f); | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                false); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                false); | 
 |  | 
 |   LayerImpl* perspective_surface = AddChild<LayerImpl>(parent); | 
 |   SetLayerPropertiesForTesting(perspective_surface, | 
 |                                perspective_matrix * scale_small_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(2.f, 2.f), | 
 |                                gfx::Size(10, 10), false, true, true); | 
 |   perspective_surface->SetDrawsContent(true); | 
 |  | 
 |   LayerImpl* scale_surface = AddChild<LayerImpl>(parent); | 
 |   SetLayerPropertiesForTesting(scale_surface, scale_small_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(2.f, 2.f), | 
 |                                gfx::Size(10, 10), false, true, true); | 
 |   scale_surface->SetDrawsContent(true); | 
 |  | 
 |   float device_scale_factor = 2.5f; | 
 |   float page_scale_factor = 3.f; | 
 |   ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor, | 
 |                                  root); | 
 |  | 
 |   EXPECT_IDEAL_SCALE_EQ(device_scale_factor * page_scale_factor, parent); | 
 |   EXPECT_IDEAL_SCALE_EQ(device_scale_factor * page_scale_factor, | 
 |                         perspective_surface); | 
 |   // Ideal scale is the max 2d scale component of the combined transform up to | 
 |   // the nearest render target. Here this includes the layer transform as well | 
 |   // as the device and page scale factors. | 
 |   gfx::Transform transform = scale_small_matrix; | 
 |   transform.Scale(device_scale_factor * page_scale_factor, | 
 |                   device_scale_factor * page_scale_factor); | 
 |   gfx::Vector2dF scales = | 
 |       MathUtil::ComputeTransform2dScaleComponents(transform, 0.f); | 
 |   float max_2d_scale = std::max(scales.x(), scales.y()); | 
 |   EXPECT_IDEAL_SCALE_EQ(max_2d_scale, scale_surface); | 
 |  | 
 |   // The ideal scale will draw 1:1 with its render target space along | 
 |   // the larger-scale axis. | 
 |   gfx::Vector2dF target_space_transform_scales = | 
 |       MathUtil::ComputeTransform2dScaleComponents( | 
 |           scale_surface->draw_properties().target_space_transform, 0.f); | 
 |   EXPECT_FLOAT_EQ(max_2d_scale, | 
 |                   std::max(target_space_transform_scales.x(), | 
 |                            target_space_transform_scales.y())); | 
 |  | 
 |   EXPECT_EQ(3u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   gfx::Transform expected_parent_draw_transform; | 
 |   expected_parent_draw_transform.Scale(device_scale_factor * page_scale_factor, | 
 |                                        device_scale_factor * page_scale_factor); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_parent_draw_transform, | 
 |                                   parent->DrawTransform()); | 
 |  | 
 |   // The scale for the perspective surface is not known, so it is rendered 1:1 | 
 |   // with the screen, and then scaled during drawing. | 
 |   gfx::Transform expected_perspective_surface_draw_transform; | 
 |   expected_perspective_surface_draw_transform.Translate( | 
 |       device_scale_factor * page_scale_factor * | 
 |       perspective_surface->position().x(), | 
 |       device_scale_factor * page_scale_factor * | 
 |       perspective_surface->position().y()); | 
 |   expected_perspective_surface_draw_transform.PreconcatTransform( | 
 |       perspective_matrix); | 
 |   expected_perspective_surface_draw_transform.PreconcatTransform( | 
 |       scale_small_matrix); | 
 |   gfx::Transform expected_perspective_surface_layer_draw_transform; | 
 |   expected_perspective_surface_layer_draw_transform.Scale( | 
 |       device_scale_factor * page_scale_factor, | 
 |       device_scale_factor * page_scale_factor); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_perspective_surface_draw_transform, | 
 |       perspective_surface->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_perspective_surface_layer_draw_transform, | 
 |       perspective_surface->DrawTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonScalingTest, SmallIdealScale) { | 
 |   gfx::Transform parent_scale_matrix; | 
 |   SkMScalar initial_parent_scale = 1.75; | 
 |   parent_scale_matrix.Scale(initial_parent_scale, initial_parent_scale); | 
 |  | 
 |   gfx::Transform child_scale_matrix; | 
 |   SkMScalar initial_child_scale = 0.25; | 
 |   child_scale_matrix.Scale(initial_child_scale, initial_child_scale); | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetBounds(gfx::Size(100, 100)); | 
 |  | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, parent_scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                false); | 
 |  | 
 |   LayerImpl* child_scale = AddChild<LayerImpl>(parent); | 
 |   child_scale->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child_scale, child_scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(10, 10), false, | 
 |                                true, false); | 
 |  | 
 |   float device_scale_factor = 2.5f; | 
 |   float page_scale_factor = 0.01f; | 
 |  | 
 |   { | 
 |     ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor, | 
 |                                    root); | 
 |  | 
 |     // The ideal scale is able to go below 1. | 
 |     float expected_ideal_scale = | 
 |         device_scale_factor * page_scale_factor * initial_parent_scale; | 
 |     EXPECT_LT(expected_ideal_scale, 1.f); | 
 |     EXPECT_IDEAL_SCALE_EQ(expected_ideal_scale, parent); | 
 |  | 
 |     expected_ideal_scale = device_scale_factor * page_scale_factor * | 
 |                            initial_parent_scale * initial_child_scale; | 
 |     EXPECT_LT(expected_ideal_scale, 1.f); | 
 |     EXPECT_IDEAL_SCALE_EQ(expected_ideal_scale, child_scale); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonScalingTest, IdealScaleForAnimatingLayer) { | 
 |   gfx::Transform parent_scale_matrix; | 
 |   SkMScalar initial_parent_scale = 1.75; | 
 |   parent_scale_matrix.Scale(initial_parent_scale, initial_parent_scale); | 
 |  | 
 |   gfx::Transform child_scale_matrix; | 
 |   SkMScalar initial_child_scale = 1.25; | 
 |   child_scale_matrix.Scale(initial_child_scale, initial_child_scale); | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetBounds(gfx::Size(100, 100)); | 
 |  | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, parent_scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), false, true, | 
 |                                false); | 
 |  | 
 |   LayerImpl* child_scale = AddChild<LayerImpl>(parent); | 
 |   child_scale->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child_scale, child_scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(10, 10), false, | 
 |                                true, false); | 
 |  | 
 |   { | 
 |     ExecuteCalculateDrawProperties(root); | 
 |  | 
 |     EXPECT_IDEAL_SCALE_EQ(initial_parent_scale, parent); | 
 |     // Animating layers compute ideal scale in the same way as when | 
 |     // they are static. | 
 |     EXPECT_IDEAL_SCALE_EQ(initial_child_scale * initial_parent_scale, | 
 |                           child_scale); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceTransformsInHighDPI) { | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   LayerImpl* parent = root_layer(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), false, true, | 
 |                                true); | 
 |  | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   child->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(10, 10), false, | 
 |                                true, true); | 
 |  | 
 |   gfx::Transform replica_transform; | 
 |   replica_transform.Scale(1.0, -1.0); | 
 |  | 
 |   scoped_ptr<LayerImpl> replica = | 
 |       LayerImpl::Create(host_impl()->active_tree(), 7); | 
 |   SetLayerPropertiesForTesting(replica.get(), replica_transform, gfx::Point3F(), | 
 |                                gfx::PointF(2.f, 2.f), gfx::Size(10, 10), false, | 
 |                                true, false); | 
 |   child->SetReplicaLayer(std::move(replica)); | 
 |  | 
 |   // This layer should end up in the same surface as child, with the same draw | 
 |   // and screen space transforms. | 
 |   LayerImpl* duplicate_child_non_owner = AddChild<LayerImpl>(child); | 
 |   duplicate_child_non_owner->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(duplicate_child_non_owner, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                false, true, false); | 
 |  | 
 |   float device_scale_factor = 1.5f; | 
 |   ExecuteCalculateDrawProperties(parent, device_scale_factor); | 
 |  | 
 |   // We should have two render surfaces. The root's render surface and child's | 
 |   // render surface (it needs one because it has a replica layer). | 
 |   EXPECT_EQ(2u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   gfx::Transform expected_parent_transform; | 
 |   expected_parent_transform.Scale(device_scale_factor, device_scale_factor); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_parent_transform, | 
 |                                   parent->ScreenSpaceTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_parent_transform, | 
 |                                   parent->DrawTransform()); | 
 |  | 
 |   gfx::Transform expected_draw_transform; | 
 |   expected_draw_transform.Scale(device_scale_factor, device_scale_factor); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_draw_transform, | 
 |                                   child->DrawTransform()); | 
 |  | 
 |   gfx::Transform expected_screen_space_transform; | 
 |   expected_screen_space_transform.Scale(device_scale_factor, | 
 |                                         device_scale_factor); | 
 |   expected_screen_space_transform.Translate(child->position().x(), | 
 |                                             child->position().y()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_screen_space_transform, | 
 |                                   child->ScreenSpaceTransform()); | 
 |  | 
 |   gfx::Transform expected_duplicate_child_draw_transform = | 
 |       child->DrawTransform(); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(child->DrawTransform(), | 
 |                                   duplicate_child_non_owner->DrawTransform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       child->ScreenSpaceTransform(), | 
 |       duplicate_child_non_owner->ScreenSpaceTransform()); | 
 |   EXPECT_EQ(child->drawable_content_rect(), | 
 |             duplicate_child_non_owner->drawable_content_rect()); | 
 |   EXPECT_EQ(child->bounds(), duplicate_child_non_owner->bounds()); | 
 |  | 
 |   gfx::Transform expected_render_surface_draw_transform; | 
 |   expected_render_surface_draw_transform.Translate( | 
 |       device_scale_factor * child->position().x(), | 
 |       device_scale_factor * child->position().y()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_render_surface_draw_transform, | 
 |                                   child->render_surface()->draw_transform()); | 
 |  | 
 |   gfx::Transform expected_surface_draw_transform; | 
 |   expected_surface_draw_transform.Translate(device_scale_factor * 2.f, | 
 |                                             device_scale_factor * 2.f); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(expected_surface_draw_transform, | 
 |                                   child->render_surface()->draw_transform()); | 
 |  | 
 |   gfx::Transform expected_surface_screen_space_transform; | 
 |   expected_surface_screen_space_transform.Translate(device_scale_factor * 2.f, | 
 |                                                     device_scale_factor * 2.f); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_surface_screen_space_transform, | 
 |       child->render_surface()->screen_space_transform()); | 
 |  | 
 |   gfx::Transform expected_replica_draw_transform; | 
 |   expected_replica_draw_transform.matrix().set(1, 1, -1.0); | 
 |   expected_replica_draw_transform.matrix().set(0, 3, 6.0); | 
 |   expected_replica_draw_transform.matrix().set(1, 3, 6.0); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_replica_draw_transform, | 
 |       child->render_surface()->replica_draw_transform()); | 
 |  | 
 |   gfx::Transform expected_replica_screen_space_transform; | 
 |   expected_replica_screen_space_transform.matrix().set(1, 1, -1.0); | 
 |   expected_replica_screen_space_transform.matrix().set(0, 3, 6.0); | 
 |   expected_replica_screen_space_transform.matrix().set(1, 3, 6.0); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_replica_screen_space_transform, | 
 |       child->render_surface()->replica_screen_space_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_replica_screen_space_transform, | 
 |       child->render_surface()->replica_screen_space_transform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |      RenderSurfaceTransformsInHighDPIAccurateScaleZeroPosition) { | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   LayerImpl* parent = root_layer(); | 
 |   parent->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(33, 31), false, true, | 
 |                                true); | 
 |  | 
 |   LayerImpl* child = AddChildToRoot<LayerImpl>(); | 
 |   child->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(13, 11), false, true, | 
 |                                true); | 
 |  | 
 |   gfx::Transform replica_transform; | 
 |   replica_transform.Scale(1.0, -1.0); | 
 |   scoped_ptr<LayerImpl> replica = | 
 |       LayerImpl::Create(host_impl()->active_tree(), 7); | 
 |   SetLayerPropertiesForTesting(replica.get(), replica_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(13, 11), false, true, | 
 |                                false); | 
 |   child->SetReplicaLayer(std::move(replica)); | 
 |  | 
 |   float device_scale_factor = 1.7f; | 
 |   ExecuteCalculateDrawProperties(parent, device_scale_factor); | 
 |  | 
 |   // We should have two render surfaces. The root's render surface and child's | 
 |   // render surface (it needs one because it has a replica layer). | 
 |   EXPECT_EQ(2u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_transform, | 
 |                                   child->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(identity_transform, | 
 |                                   child->render_surface()->draw_transform()); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       identity_transform, child->render_surface()->screen_space_transform()); | 
 |  | 
 |   gfx::Transform expected_replica_draw_transform; | 
 |   expected_replica_draw_transform.matrix().set(1, 1, -1.0); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_replica_draw_transform, | 
 |       child->render_surface()->replica_draw_transform()); | 
 |  | 
 |   gfx::Transform expected_replica_screen_space_transform; | 
 |   expected_replica_screen_space_transform.matrix().set(1, 1, -1.0); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       expected_replica_screen_space_transform, | 
 |       child->render_surface()->replica_screen_space_transform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SubtreeSearch) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> mask_layer = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> replica_layer = Layer::Create(layer_settings()); | 
 |  | 
 |   grand_child->SetReplicaLayer(replica_layer.get()); | 
 |   child->AddChild(grand_child.get()); | 
 |   child->SetMaskLayer(mask_layer.get()); | 
 |   root->AddChild(child.get()); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   int nonexistent_id = -1; | 
 |   EXPECT_EQ(root.get(), | 
 |             LayerTreeHostCommon::FindLayerInSubtree(root.get(), root->id())); | 
 |   EXPECT_EQ(child.get(), | 
 |             LayerTreeHostCommon::FindLayerInSubtree(root.get(), child->id())); | 
 |   EXPECT_EQ( | 
 |       grand_child.get(), | 
 |       LayerTreeHostCommon::FindLayerInSubtree(root.get(), grand_child->id())); | 
 |   EXPECT_EQ( | 
 |       mask_layer.get(), | 
 |       LayerTreeHostCommon::FindLayerInSubtree(root.get(), mask_layer->id())); | 
 |   EXPECT_EQ( | 
 |       replica_layer.get(), | 
 |       LayerTreeHostCommon::FindLayerInSubtree(root.get(), replica_layer->id())); | 
 |   EXPECT_EQ( | 
 |       0, LayerTreeHostCommon::FindLayerInSubtree(root.get(), nonexistent_id)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransparentChildRenderSurfaceCreation) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grand_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(100, 100), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(10, 10), | 
 |                                true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(10, 10), | 
 |                                true, | 
 |                                false); | 
 |  | 
 |   root->AddChild(child); | 
 |   child->AddChild(grand_child); | 
 |   child->SetOpacity(0.5f); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_FALSE(child->has_render_surface()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, OpacityAnimatingOnPendingTree) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(host()->settings(), &task_runner_provider, | 
 |                                   &shared_bitmap_manager, &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.pending_tree(), 1); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.pending_tree(), 2); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   child->SetDrawsContent(true); | 
 |   child->SetOpacity(0.0f); | 
 |  | 
 |   // Add opacity animation. | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     scoped_refptr<AnimationTimeline> timeline = | 
 |         AnimationTimeline::Create(AnimationIdProvider::NextTimelineId()); | 
 |     host_impl.animation_host()->AddAnimationTimeline(timeline); | 
 |  | 
 |     AddOpacityTransitionToLayerWithPlayer(child->id(), timeline, 10.0, 0.0f, | 
 |                                           1.0f, false); | 
 |   } else { | 
 |     AddOpacityTransitionToController(child->layer_animation_controller(), 10.0, | 
 |                                      0.0f, 1.0f, false); | 
 |   } | 
 |  | 
 |   root->AddChild(std::move(child)); | 
 |   root->SetHasRenderSurface(true); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // We should have one render surface and two layers. The child | 
 |   // layer should be included even though it is transparent. | 
 |   ASSERT_EQ(1u, render_surface_layer_list.size()); | 
 |   ASSERT_EQ(2u, root->render_surface()->layer_list().size()); | 
 |  | 
 |   // If the root itself is hidden, the child should not be drawn even if it has | 
 |   // an animating opacity. | 
 |   root->SetOpacity(0.f); | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   LayerImplList render_surface_layer_list2; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs2( | 
 |       root.get(), root->bounds(), &render_surface_layer_list2, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs2.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs2); | 
 |  | 
 |   LayerImpl* child_ptr = root->layer_tree_impl()->LayerById(2); | 
 |   EffectTree tree = root->layer_tree_impl()->property_trees()->effect_tree; | 
 |   EffectNode* node = tree.Node(child_ptr->effect_tree_index()); | 
 |   EXPECT_FALSE(node->data.is_drawn); | 
 | } | 
 |  | 
 | class LayerTreeSettingsForLCDTextTest : public LayerTreeSettings { | 
 |  public: | 
 |   LayerTreeSettingsForLCDTextTest() { | 
 |     use_compositor_animation_timelines = true; | 
 |   } | 
 | }; | 
 |  | 
 | using LCDTextTestParam = std::tr1::tuple<bool, bool, bool>; | 
 | class LCDTextTest : public LayerTreeHostCommonTestBase, | 
 |                     public testing::TestWithParam<LCDTextTestParam> { | 
 |  public: | 
 |   LCDTextTest() | 
 |       : LayerTreeHostCommonTestBase(LayerTreeSettingsForLCDTextTest()), | 
 |         host_impl_(LayerTreeSettingsForLCDTextTest(), | 
 |                    &task_runner_provider_, | 
 |                    &shared_bitmap_manager_, | 
 |                    &task_graph_runner_), | 
 |         root_(nullptr), | 
 |         child_(nullptr), | 
 |         grand_child_(nullptr) {} | 
 |  | 
 |   scoped_refptr<AnimationTimeline> timeline() { return timeline_; } | 
 |  | 
 |  protected: | 
 |   void SetUp() override { | 
 |     if (layer_settings().use_compositor_animation_timelines) { | 
 |       timeline_ = | 
 |           AnimationTimeline::Create(AnimationIdProvider::NextTimelineId()); | 
 |       host_impl_.animation_host()->AddAnimationTimeline(timeline_); | 
 |     } | 
 |  | 
 |     can_use_lcd_text_ = std::tr1::get<0>(GetParam()); | 
 |     layers_always_allowed_lcd_text_ = std::tr1::get<1>(GetParam()); | 
 |  | 
 |     scoped_ptr<LayerImpl> root_ptr = | 
 |         LayerImpl::Create(host_impl_.active_tree(), 1); | 
 |     scoped_ptr<LayerImpl> child_ptr = | 
 |         LayerImpl::Create(host_impl_.active_tree(), 2); | 
 |     scoped_ptr<LayerImpl> grand_child_ptr = | 
 |         LayerImpl::Create(host_impl_.active_tree(), 3); | 
 |  | 
 |     // Stash raw pointers to look at later. | 
 |     root_ = root_ptr.get(); | 
 |     child_ = child_ptr.get(); | 
 |     grand_child_ = grand_child_ptr.get(); | 
 |  | 
 |     child_->AddChild(std::move(grand_child_ptr)); | 
 |     root_->AddChild(std::move(child_ptr)); | 
 |     host_impl_.active_tree()->SetRootLayer(std::move(root_ptr)); | 
 |  | 
 |     root_->SetContentsOpaque(true); | 
 |     child_->SetContentsOpaque(true); | 
 |     grand_child_->SetContentsOpaque(true); | 
 |  | 
 |     root_->SetDrawsContent(true); | 
 |     child_->SetDrawsContent(true); | 
 |     grand_child_->SetDrawsContent(true); | 
 |  | 
 |     gfx::Transform identity_matrix; | 
 |     SetLayerPropertiesForTesting(root_, identity_matrix, gfx::Point3F(), | 
 |                                  gfx::PointF(), gfx::Size(1, 1), true, false, | 
 |                                  true); | 
 |     SetLayerPropertiesForTesting(child_, identity_matrix, gfx::Point3F(), | 
 |                                  gfx::PointF(), gfx::Size(1, 1), true, false, | 
 |                                  std::tr1::get<2>(GetParam())); | 
 |     SetLayerPropertiesForTesting(grand_child_, identity_matrix, gfx::Point3F(), | 
 |                                  gfx::PointF(), gfx::Size(1, 1), true, false, | 
 |                                  false); | 
 |   } | 
 |  | 
 |   bool can_use_lcd_text_; | 
 |   bool layers_always_allowed_lcd_text_; | 
 |  | 
 |   FakeImplTaskRunnerProvider task_runner_provider_; | 
 |   TestSharedBitmapManager shared_bitmap_manager_; | 
 |   TestTaskGraphRunner task_graph_runner_; | 
 |   FakeLayerTreeHostImpl host_impl_; | 
 |   scoped_refptr<AnimationTimeline> timeline_; | 
 |  | 
 |   LayerImpl* root_; | 
 |   LayerImpl* child_; | 
 |   LayerImpl* grand_child_; | 
 | }; | 
 |  | 
 | TEST_P(LCDTextTest, CanUseLCDText) { | 
 |   bool expect_lcd_text = can_use_lcd_text_ || layers_always_allowed_lcd_text_; | 
 |   bool expect_not_lcd_text = layers_always_allowed_lcd_text_; | 
 |  | 
 |   // Case 1: Identity transform. | 
 |   gfx::Transform identity_matrix; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 2: Integral translation. | 
 |   gfx::Transform integral_translation; | 
 |   integral_translation.Translate(1.0, 2.0); | 
 |   child_->SetTransform(integral_translation); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 3: Non-integral translation. | 
 |   gfx::Transform non_integral_translation; | 
 |   non_integral_translation.Translate(1.5, 2.5); | 
 |   child_->SetTransform(non_integral_translation); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 4: Rotation. | 
 |   gfx::Transform rotation; | 
 |   rotation.Rotate(10.0); | 
 |   child_->SetTransform(rotation); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 5: Scale. | 
 |   gfx::Transform scale; | 
 |   scale.Scale(2.0, 2.0); | 
 |   child_->SetTransform(scale); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 6: Skew. | 
 |   gfx::Transform skew; | 
 |   skew.Skew(10.0, 0.0); | 
 |   child_->SetTransform(skew); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 7: Translucent. | 
 |   child_->SetTransform(identity_matrix); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   child_->SetOpacity(0.5f); | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 8: Sanity check: restore transform and opacity. | 
 |   child_->SetTransform(identity_matrix); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   child_->SetOpacity(1.f); | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 9: Non-opaque content. | 
 |   child_->SetContentsOpaque(false); | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Case 10: Sanity check: restore content opaqueness. | 
 |   child_->SetContentsOpaque(true); | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 | } | 
 |  | 
 | TEST_P(LCDTextTest, CanUseLCDTextWithAnimation) { | 
 |   bool expect_lcd_text = can_use_lcd_text_ || layers_always_allowed_lcd_text_; | 
 |   bool expect_not_lcd_text = layers_always_allowed_lcd_text_; | 
 |  | 
 |   // Sanity check: Make sure can_use_lcd_text_ is set on each node. | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Add opacity animation. | 
 |   child_->SetOpacity(0.9f); | 
 |   child_->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddOpacityTransitionToLayerWithPlayer(child_->id(), timeline(), 10.0, 0.9f, | 
 |                                           0.1f, false); | 
 |   } else { | 
 |     AddOpacityTransitionToController(child_->layer_animation_controller(), 10.0, | 
 |                                      0.9f, 0.1f, false); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   // Text LCD should be adjusted while animation is active. | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, grand_child_->can_use_lcd_text()); | 
 | } | 
 |  | 
 | TEST_P(LCDTextTest, CanUseLCDTextWithAnimationContentsOpaque) { | 
 |   bool expect_lcd_text = can_use_lcd_text_ || layers_always_allowed_lcd_text_; | 
 |   bool expect_not_lcd_text = layers_always_allowed_lcd_text_; | 
 |  | 
 |   // Sanity check: Make sure can_use_lcd_text_ is set on each node. | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 |  | 
 |   // Mark contents non-opaque within the first animation frame. | 
 |   child_->SetContentsOpaque(false); | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddOpacityTransitionToLayerWithPlayer(child_->id(), timeline(), 10.0, 0.9f, | 
 |                                           0.1f, false); | 
 |   } else { | 
 |     AddOpacityTransitionToController(child_->layer_animation_controller(), 10.0, | 
 |                                      0.9f, 0.1f, false); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(root_, 1.f, 1.f, NULL, can_use_lcd_text_, | 
 |                                  layers_always_allowed_lcd_text_); | 
 |   // LCD text should be disabled for non-opaque layers even during animations. | 
 |   EXPECT_EQ(expect_lcd_text, root_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_not_lcd_text, child_->can_use_lcd_text()); | 
 |   EXPECT_EQ(expect_lcd_text, grand_child_->can_use_lcd_text()); | 
 | } | 
 |  | 
 | INSTANTIATE_TEST_CASE_P(LayerTreeHostCommonTest, | 
 |                         LCDTextTest, | 
 |                         testing::Combine(testing::Bool(), | 
 |                                          testing::Bool(), | 
 |                                          testing::Bool())); | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SubtreeHidden_SingleLayerImpl) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.pending_tree(), 1); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.pending_tree(), 2); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(40, 40), true, false, | 
 |                                false); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> grand_child = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 3); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   grand_child->SetDrawsContent(true); | 
 |   grand_child->SetHideLayerAndSubtree(true); | 
 |  | 
 |   child->AddChild(std::move(grand_child)); | 
 |   root->AddChild(std::move(child)); | 
 |   root->SetHasRenderSurface(true); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // We should have one render surface and two layers. The grand child has | 
 |   // hidden itself. | 
 |   ASSERT_EQ(1u, render_surface_layer_list.size()); | 
 |   ASSERT_EQ(2u, root->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(1, root->render_surface()->layer_list().at(0)->id()); | 
 |   EXPECT_EQ(2, root->render_surface()->layer_list().at(1)->id()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SubtreeHidden_TwoLayersImpl) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.pending_tree(), 1); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.pending_tree(), 2); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(40, 40), true, false, | 
 |                                false); | 
 |   child->SetDrawsContent(true); | 
 |   child->SetHideLayerAndSubtree(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> grand_child = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 3); | 
 |   SetLayerPropertiesForTesting(grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   child->AddChild(std::move(grand_child)); | 
 |   root->AddChild(std::move(child)); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // We should have one render surface and one layers. The child has | 
 |   // hidden itself and the grand child. | 
 |   ASSERT_EQ(1u, render_surface_layer_list.size()); | 
 |   ASSERT_EQ(1u, root->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(1, root->render_surface()->layer_list().at(0)->id()); | 
 | } | 
 |  | 
 | void EmptyCopyOutputCallback(scoped_ptr<CopyOutputResult> result) {} | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SubtreeHiddenWithCopyRequest) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.pending_tree(), 1); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_grand_parent = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 2); | 
 |   SetLayerPropertiesForTesting(copy_grand_parent.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   copy_grand_parent->SetDrawsContent(true); | 
 |   LayerImpl* copy_grand_parent_layer = copy_grand_parent.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_parent = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 3); | 
 |   SetLayerPropertiesForTesting(copy_parent.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, true); | 
 |   copy_parent->SetDrawsContent(true); | 
 |   LayerImpl* copy_parent_layer = copy_parent.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_request = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 4); | 
 |   SetLayerPropertiesForTesting(copy_request.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, true); | 
 |   copy_request->SetDrawsContent(true); | 
 |   LayerImpl* copy_layer = copy_request.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_child = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 5); | 
 |   SetLayerPropertiesForTesting(copy_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, false); | 
 |   copy_child->SetDrawsContent(true); | 
 |   LayerImpl* copy_child_layer = copy_child.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_grand_child = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 6); | 
 |   SetLayerPropertiesForTesting(copy_grand_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, false); | 
 |   copy_child->SetDrawsContent(true); | 
 |   LayerImpl* copy_grand_child_layer = copy_grand_child.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_grand_parent_sibling_before = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 7); | 
 |   SetLayerPropertiesForTesting(copy_grand_parent_sibling_before.get(), | 
 |                                identity_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(40, 40), true, false, false); | 
 |   copy_grand_parent_sibling_before->SetDrawsContent(true); | 
 |   LayerImpl* copy_grand_parent_sibling_before_layer = | 
 |       copy_grand_parent_sibling_before.get(); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_grand_parent_sibling_after = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 8); | 
 |   SetLayerPropertiesForTesting(copy_grand_parent_sibling_after.get(), | 
 |                                identity_matrix, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(40, 40), true, false, false); | 
 |   copy_grand_parent_sibling_after->SetDrawsContent(true); | 
 |   LayerImpl* copy_grand_parent_sibling_after_layer = | 
 |       copy_grand_parent_sibling_after.get(); | 
 |  | 
 |   copy_child->AddChild(std::move(copy_grand_child)); | 
 |   copy_request->AddChild(std::move(copy_child)); | 
 |   copy_parent->AddChild(std::move(copy_request)); | 
 |   copy_grand_parent->AddChild(std::move(copy_parent)); | 
 |   root->AddChild(std::move(copy_grand_parent_sibling_before)); | 
 |   root->AddChild(std::move(copy_grand_parent)); | 
 |   root->AddChild(std::move(copy_grand_parent_sibling_after)); | 
 |  | 
 |   // Hide the copy_grand_parent and its subtree. But make a copy request in that | 
 |   // hidden subtree on copy_layer. Also hide the copy grand child and its | 
 |   // subtree. | 
 |   copy_grand_parent_layer->SetHideLayerAndSubtree(true); | 
 |   copy_grand_parent_sibling_before_layer->SetHideLayerAndSubtree(true); | 
 |   copy_grand_parent_sibling_after_layer->SetHideLayerAndSubtree(true); | 
 |   copy_grand_child_layer->SetHideLayerAndSubtree(true); | 
 |  | 
 |   std::vector<scoped_ptr<CopyOutputRequest>> copy_requests; | 
 |   copy_requests.push_back( | 
 |       CopyOutputRequest::CreateRequest(base::Bind(&EmptyCopyOutputCallback))); | 
 |   copy_layer->PassCopyRequests(©_requests); | 
 |   EXPECT_TRUE(copy_layer->HasCopyRequest()); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   EXPECT_GT(root->num_copy_requests_in_target_subtree(), 0); | 
 |   EXPECT_GT(copy_grand_parent_layer->num_copy_requests_in_target_subtree(), 0); | 
 |   EXPECT_GT(copy_parent_layer->num_copy_requests_in_target_subtree(), 0); | 
 |   EXPECT_GT(copy_layer->num_copy_requests_in_target_subtree(), 0); | 
 |  | 
 |   // We should have four render surfaces, one for the root, one for the grand | 
 |   // parent since it has opacity and two drawing descendants, one for the parent | 
 |   // since it owns a surface, and one for the copy_layer. | 
 |   ASSERT_EQ(4u, render_surface_layer_list.size()); | 
 |   EXPECT_EQ(root->id(), render_surface_layer_list.at(0)->id()); | 
 |   EXPECT_EQ(copy_grand_parent_layer->id(), | 
 |             render_surface_layer_list.at(1)->id()); | 
 |   EXPECT_EQ(copy_parent_layer->id(), render_surface_layer_list.at(2)->id()); | 
 |   EXPECT_EQ(copy_layer->id(), render_surface_layer_list.at(3)->id()); | 
 |  | 
 |   // The root render surface should have 2 contributing layers. | 
 |   ASSERT_EQ(2u, root->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(root->id(), root->render_surface()->layer_list().at(0)->id()); | 
 |   EXPECT_EQ(copy_grand_parent_layer->id(), | 
 |             root->render_surface()->layer_list().at(1)->id()); | 
 |  | 
 |   // Nothing actually draws into the copy parent, so only the copy_layer will | 
 |   // appear in its list, since it needs to be drawn for the copy request. | 
 |   ASSERT_EQ(1u, copy_parent_layer->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(copy_layer->id(), | 
 |             copy_layer->render_surface()->layer_list().at(0)->id()); | 
 |  | 
 |   // The copy_layer's render surface should have two contributing layers. | 
 |   ASSERT_EQ(2u, copy_layer->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(copy_layer->id(), | 
 |             copy_layer->render_surface()->layer_list().at(0)->id()); | 
 |   EXPECT_EQ(copy_child_layer->id(), | 
 |             copy_layer->render_surface()->layer_list().at(1)->id()); | 
 |  | 
 |   // copy_grand_parent, copy_parent shouldn't be drawn because they are hidden, | 
 |   // but the copy_layer and copy_child should be drawn for the copy request. | 
 |   // copy grand child should not be drawn as its hidden even in the copy | 
 |   // request. | 
 |   EffectTree tree = root->layer_tree_impl()->property_trees()->effect_tree; | 
 |   EffectNode* node = tree.Node(copy_grand_parent_layer->effect_tree_index()); | 
 |   EXPECT_FALSE(node->data.is_drawn); | 
 |   node = tree.Node(copy_parent_layer->effect_tree_index()); | 
 |   EXPECT_FALSE(node->data.is_drawn); | 
 |   node = tree.Node(copy_layer->effect_tree_index()); | 
 |   EXPECT_TRUE(node->data.is_drawn); | 
 |   node = tree.Node(copy_child_layer->effect_tree_index()); | 
 |   EXPECT_TRUE(node->data.is_drawn); | 
 |   node = tree.Node(copy_grand_child_layer->effect_tree_index()); | 
 |   EXPECT_FALSE(node->data.is_drawn); | 
 |  | 
 |   // Though copy_layer is drawn, it shouldn't contribute to drawn surface as its | 
 |   // actually hidden. | 
 |   EXPECT_FALSE(copy_layer->render_surface()->contributes_to_drawn_surface()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClippedOutCopyRequest) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.pending_tree(), 1); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_parent = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 2); | 
 |   SetLayerPropertiesForTesting(copy_parent.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(), true, | 
 |                                false, false); | 
 |   copy_parent->SetDrawsContent(true); | 
 |   copy_parent->SetMasksToBounds(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_layer = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 3); | 
 |   SetLayerPropertiesForTesting(copy_layer.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, true); | 
 |   copy_layer->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> copy_child = | 
 |       LayerImpl::Create(host_impl.pending_tree(), 4); | 
 |   SetLayerPropertiesForTesting(copy_child.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, false); | 
 |   copy_child->SetDrawsContent(true); | 
 |  | 
 |   std::vector<scoped_ptr<CopyOutputRequest>> copy_requests; | 
 |   copy_requests.push_back( | 
 |       CopyOutputRequest::CreateRequest(base::Bind(&EmptyCopyOutputCallback))); | 
 |   copy_layer->PassCopyRequests(©_requests); | 
 |   EXPECT_TRUE(copy_layer->HasCopyRequest()); | 
 |  | 
 |   copy_layer->AddChild(std::move(copy_child)); | 
 |   copy_parent->AddChild(std::move(copy_layer)); | 
 |   root->AddChild(std::move(copy_parent)); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // We should have two render surface, as the others are clipped out. | 
 |   ASSERT_EQ(2u, render_surface_layer_list.size()); | 
 |   EXPECT_EQ(root->id(), render_surface_layer_list.at(0)->id()); | 
 |  | 
 |   // The root render surface should only have 2 contributing layer, since the | 
 |   // other layers are empty/clipped away. | 
 |   ASSERT_EQ(2u, root->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(root->id(), root->render_surface()->layer_list().at(0)->id()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleContentRectInsideSurface) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(50, 50), | 
 |                                true, | 
 |                                false); | 
 |   root->SetIsDrawable(true); | 
 |  | 
 |   // The surface is moved slightly outside of the viewport. | 
 |   scoped_refptr<Layer> surface = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(surface.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(-10, -20), | 
 |                                gfx::Size(), | 
 |                                true, | 
 |                                false); | 
 |   surface->SetForceRenderSurface(true); | 
 |  | 
 |   scoped_refptr<Layer> surface_child = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(surface_child.get(), | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(50, 50), | 
 |                                true, | 
 |                                false); | 
 |   surface_child->SetIsDrawable(true); | 
 |  | 
 |   surface->AddChild(surface_child); | 
 |   root->AddChild(surface); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   LayerTreeHostCommon::CalcDrawPropsMainInputs inputs(root.get(), | 
 |                                                       root->bounds()); | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // The visible_layer_rect for the |surface_child| should not be clipped by | 
 |   // the viewport. | 
 |   EXPECT_EQ(gfx::Rect(50, 50).ToString(), | 
 |             surface_child->visible_rect_from_property_trees().ToString()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformedClipParent) { | 
 |   // Ensure that a transform between the layer and its render surface is not a | 
 |   // problem. Constructs the following layer tree. | 
 |   // | 
 |   //   root (a render surface) | 
 |   //     + render_surface | 
 |   //       + clip_parent (scaled) | 
 |   //         + intervening_clipping_layer | 
 |   //           + clip_child | 
 |   // | 
 |   // The render surface should be resized correctly and the clip child should | 
 |   // inherit the right clip rect. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(render_surface); | 
 |   clip_parent->SetDrawsContent(true); | 
 |   LayerImpl* intervening = AddChild<LayerImpl>(clip_parent); | 
 |   intervening->SetDrawsContent(true); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(intervening); | 
 |   clip_child->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   intervening->SetMasksToBounds(true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |  | 
 |   gfx::Transform scale_transform; | 
 |   scale_transform.Scale(2, 2); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_parent, scale_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 1.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(intervening, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 1.f), gfx::Size(5, 5), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 1.f), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   ASSERT_TRUE(root->render_surface()); | 
 |   ASSERT_TRUE(render_surface->render_surface()); | 
 |  | 
 |   // Ensure that we've inherited our clip parent's clip and weren't affected | 
 |   // by the intervening clip layer. | 
 |   ASSERT_EQ(gfx::Rect(1, 1, 20, 20).ToString(), | 
 |             clip_parent->clip_rect().ToString()); | 
 |   ASSERT_EQ(clip_parent->clip_rect().ToString(), | 
 |             clip_child->clip_rect().ToString()); | 
 |   ASSERT_EQ(gfx::Rect(3, 3, 10, 10).ToString(), | 
 |             intervening->clip_rect().ToString()); | 
 |  | 
 |   // Ensure that the render surface reports a content rect that has been grown | 
 |   // to accomodate for the clip child. | 
 |   ASSERT_EQ(gfx::Rect(1, 1, 20, 20).ToString(), | 
 |             render_surface->render_surface()->content_rect().ToString()); | 
 |  | 
 |   // The above check implies the two below, but they nicely demonstrate that | 
 |   // we've grown, despite the intervening layer's clip. | 
 |   ASSERT_TRUE(clip_parent->clip_rect().Contains( | 
 |       render_surface->render_surface()->content_rect())); | 
 |   ASSERT_FALSE(intervening->clip_rect().Contains( | 
 |       render_surface->render_surface()->content_rect())); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipParentWithInterveningRenderSurface) { | 
 |   // Ensure that intervening render surfaces are not a problem in the basic | 
 |   // case. In the following tree, both render surfaces should be resized to | 
 |   // accomodate for the clip child, despite an intervening clip. | 
 |   // | 
 |   //   root (a render surface) | 
 |   //    + clip_parent (masks to bounds) | 
 |   //      + render_surface1 (sets opacity) | 
 |   //        + intervening (masks to bounds) | 
 |   //          + render_surface2 (also sets opacity) | 
 |   //            + clip_child | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* intervening = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(intervening); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface2); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |  | 
 |   clip_child->SetClipParent(clip_parent); | 
 |  | 
 |   intervening->SetMasksToBounds(true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |  | 
 |   gfx::Transform translation_transform; | 
 |   translation_transform.Translate(2, 2); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, translation_transform, | 
 |                                gfx::Point3F(), gfx::PointF(1.f, 1.f), | 
 |                                gfx::Size(40, 40), true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(intervening, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 1.f), gfx::Size(5, 5), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(-10.f, -10.f), gfx::Size(60, 60), | 
 |                                true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |   EXPECT_TRUE(render_surface1->render_surface()); | 
 |   EXPECT_TRUE(render_surface2->render_surface()); | 
 |  | 
 |   // Since the render surfaces could have expanded, they should not clip (their | 
 |   // bounds would no longer be reliable). We should resort to layer clipping | 
 |   // in this case. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0).ToString(), | 
 |             render_surface1->render_surface()->clip_rect().ToString()); | 
 |   EXPECT_FALSE(render_surface1->render_surface()->is_clipped()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0).ToString(), | 
 |             render_surface2->render_surface()->clip_rect().ToString()); | 
 |   EXPECT_FALSE(render_surface2->render_surface()->is_clipped()); | 
 |  | 
 |   // NB: clip rects are in target space. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             render_surface1->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface1->is_clipped()); | 
 |  | 
 |   // This value is inherited from the clipping ancestor layer, 'intervening'. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 5, 5).ToString(), | 
 |             render_surface2->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface2->is_clipped()); | 
 |  | 
 |   // The content rects of both render surfaces should both have expanded to | 
 |   // contain the clip child. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             render_surface1->render_surface()->content_rect().ToString()); | 
 |   EXPECT_EQ(gfx::Rect(-1, -1, 40, 40).ToString(), | 
 |             render_surface2->render_surface()->content_rect().ToString()); | 
 |  | 
 |   // The clip child should have inherited the clip parent's clip (projected to | 
 |   // the right space, of course), and should have the correctly sized visible | 
 |   // content rect. | 
 |   EXPECT_EQ(gfx::Rect(-1, -1, 40, 40).ToString(), | 
 |             clip_child->clip_rect().ToString()); | 
 |   EXPECT_EQ(gfx::Rect(9, 9, 40, 40).ToString(), | 
 |             clip_child->visible_layer_rect().ToString()); | 
 |   EXPECT_TRUE(clip_child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipParentScrolledInterveningLayer) { | 
 |   // Ensure that intervening render surfaces are not a problem, even if there | 
 |   // is a scroll involved. Note, we do _not_ have to consider any other sort | 
 |   // of transform. | 
 |   // | 
 |   //   root (a render surface) | 
 |   //    + clip_parent (masks to bounds) | 
 |   //      + render_surface1 (sets opacity) | 
 |   //        + intervening (masks to bounds AND scrolls) | 
 |   //          + render_surface2 (also sets opacity) | 
 |   //            + clip_child | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* intervening = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(intervening); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface2); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |  | 
 |   clip_child->SetClipParent(clip_parent); | 
 |  | 
 |   intervening->SetMasksToBounds(true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   intervening->SetScrollClipLayer(clip_parent->id()); | 
 |   intervening->SetCurrentScrollOffset(gfx::ScrollOffset(3, 3)); | 
 |  | 
 |   gfx::Transform translation_transform; | 
 |   translation_transform.Translate(2, 2); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, translation_transform, | 
 |                                gfx::Point3F(), gfx::PointF(1.f, 1.f), | 
 |                                gfx::Size(40, 40), true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(intervening, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 1.f), gfx::Size(5, 5), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(10, 10), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(-10.f, -10.f), gfx::Size(60, 60), | 
 |                                true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |   EXPECT_TRUE(render_surface1->render_surface()); | 
 |   EXPECT_TRUE(render_surface2->render_surface()); | 
 |  | 
 |   // Since the render surfaces could have expanded, they should not clip (their | 
 |   // bounds would no longer be reliable). We should resort to layer clipping | 
 |   // in this case. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0).ToString(), | 
 |             render_surface1->render_surface()->clip_rect().ToString()); | 
 |   EXPECT_FALSE(render_surface1->render_surface()->is_clipped()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0).ToString(), | 
 |             render_surface2->render_surface()->clip_rect().ToString()); | 
 |   EXPECT_FALSE(render_surface2->render_surface()->is_clipped()); | 
 |  | 
 |   // NB: clip rects are in target space. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             render_surface1->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface1->is_clipped()); | 
 |  | 
 |   // This value is inherited from the clipping ancestor layer, 'intervening'. | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 3, 3).ToString(), | 
 |             render_surface2->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface2->is_clipped()); | 
 |  | 
 |   // The content rects of both render surfaces should both have expanded to | 
 |   // contain the clip child. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             render_surface1->render_surface()->content_rect().ToString()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 40, 40).ToString(), | 
 |             render_surface2->render_surface()->content_rect().ToString()); | 
 |  | 
 |   // The clip child should have inherited the clip parent's clip (projected to | 
 |   // the right space, of course), and should have the correctly sized visible | 
 |   // content rect. | 
 |   EXPECT_EQ(gfx::Rect(2, 2, 40, 40).ToString(), | 
 |             clip_child->clip_rect().ToString()); | 
 |   EXPECT_EQ(gfx::Rect(12, 12, 40, 40).ToString(), | 
 |             clip_child->visible_layer_rect().ToString()); | 
 |   EXPECT_TRUE(clip_child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DescendantsOfClipChildren) { | 
 |   // Ensures that descendants of the clip child inherit the correct clip. | 
 |   // | 
 |   //   root (a render surface) | 
 |   //    + clip_parent (masks to bounds) | 
 |   //      + intervening (masks to bounds) | 
 |   //        + clip_child | 
 |   //          + child | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(root); | 
 |   LayerImpl* intervening = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(intervening); | 
 |   LayerImpl* child = AddChild<LayerImpl>(clip_child); | 
 |   clip_child->SetDrawsContent(true); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   intervening->SetMasksToBounds(true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(40, 40), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(intervening, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(5, 5), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(60, 60), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |  | 
 |   // Neither the clip child nor its descendant should have inherited the clip | 
 |   // from |intervening|. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             clip_child->clip_rect().ToString()); | 
 |   EXPECT_TRUE(clip_child->is_clipped()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 40, 40).ToString(), | 
 |             child->visible_layer_rect().ToString()); | 
 |   EXPECT_TRUE(child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        SurfacesShouldBeUnaffectedByNonDescendantClipChildren) { | 
 |   // Ensures that non-descendant clip children in the tree do not affect | 
 |   // render surfaces. | 
 |   // | 
 |   //   root (a render surface) | 
 |   //    + clip_parent (masks to bounds) | 
 |   //      + render_surface1 | 
 |   //        + clip_child | 
 |   //      + render_surface2 | 
 |   //        + non_clip_child | 
 |   // | 
 |   // In this example render_surface2 should be unaffected by clip_child. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_parent); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface1); | 
 |   clip_child->SetDrawsContent(true); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(clip_parent); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   LayerImpl* non_clip_child = AddChild<LayerImpl>(render_surface2); | 
 |   non_clip_child->SetDrawsContent(true); | 
 |  | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   render_surface1->SetMasksToBounds(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(15, 15), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(5, 5), | 
 |                                gfx::Size(5, 5), true, false, true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(5, 5), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(-1, 1), gfx::Size(10, 10), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(non_clip_child, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(5, 5), | 
 |                                true, false, false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |   EXPECT_TRUE(render_surface1->render_surface()); | 
 |   EXPECT_TRUE(render_surface2->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 5, 5).ToString(), | 
 |             render_surface1->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface1->is_clipped()); | 
 |  | 
 |   // The render surface should not clip (it has unclipped descendants), instead | 
 |   // it should rely on layer clipping. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 0, 0).ToString(), | 
 |             render_surface1->render_surface()->clip_rect().ToString()); | 
 |   EXPECT_FALSE(render_surface1->render_surface()->is_clipped()); | 
 |  | 
 |   // That said, it should have grown to accomodate the unclipped descendant and | 
 |   // its own size. | 
 |   EXPECT_EQ(gfx::Rect(-1, 0, 6, 5).ToString(), | 
 |             render_surface1->render_surface()->content_rect().ToString()); | 
 |  | 
 |   // This render surface should clip. It has no unclipped descendants. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 5, 5).ToString(), | 
 |             render_surface2->clip_rect().ToString()); | 
 |   EXPECT_TRUE(render_surface2->render_surface()->is_clipped()); | 
 |  | 
 |   // It also shouldn't have grown to accomodate the clip child. | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 5, 5).ToString(), | 
 |             render_surface2->render_surface()->content_rect().ToString()); | 
 |  | 
 |   // Sanity check our num_unclipped_descendants values. | 
 |   EXPECT_EQ(1u, render_surface1->num_unclipped_descendants()); | 
 |   EXPECT_EQ(0u, render_surface2->num_unclipped_descendants()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        CreateRenderSurfaceWhenFlattenInsideRenderingContext) { | 
 |   // Verifies that Render Surfaces are created at the edge of rendering context. | 
 |  | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child1 = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(child1); | 
 |   LayerImpl* child3 = AddChild<LayerImpl>(child2); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   gfx::Point3F transform_origin; | 
 |   gfx::PointF position; | 
 |   gfx::Size bounds(100, 100); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, transform_origin, | 
 |                                position, bounds, true, false); | 
 |   SetLayerPropertiesForTesting(child1, identity_matrix, transform_origin, | 
 |                                position, bounds, false, true); | 
 |   child1->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child2, identity_matrix, transform_origin, | 
 |                                position, bounds, true, false); | 
 |   child2->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child3, identity_matrix, transform_origin, | 
 |                                position, bounds, true, false); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   child2->Set3dSortingContextId(1); | 
 |   child3->Set3dSortingContextId(1); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root); | 
 |  | 
 |   // Verify which render surfaces were created. | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_FALSE(child1->has_render_surface()); | 
 |   EXPECT_TRUE(child2->has_render_surface()); | 
 |   EXPECT_FALSE(child3->has_render_surface()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, CanRenderToSeparateSurface) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |  | 
 |   scoped_ptr<LayerImpl> root = | 
 |       LayerImpl::Create(host_impl.active_tree(), 12345); | 
 |   scoped_ptr<LayerImpl> child1 = | 
 |       LayerImpl::Create(host_impl.active_tree(), 123456); | 
 |   scoped_ptr<LayerImpl> child2 = | 
 |       LayerImpl::Create(host_impl.active_tree(), 1234567); | 
 |   scoped_ptr<LayerImpl> child3 = | 
 |       LayerImpl::Create(host_impl.active_tree(), 12345678); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Point3F transform_origin; | 
 |   gfx::PointF position; | 
 |   gfx::Size bounds(100, 100); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, transform_origin, | 
 |                                position, bounds, true, false, true); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   // This layer structure normally forces render surface due to preserves3d | 
 |   // behavior. | 
 |   SetLayerPropertiesForTesting(child1.get(), identity_matrix, transform_origin, | 
 |                                position, bounds, false, true, false); | 
 |   child1->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child2.get(), identity_matrix, transform_origin, | 
 |                                position, bounds, true, false, true); | 
 |   child2->SetDrawsContent(true); | 
 |   SetLayerPropertiesForTesting(child3.get(), identity_matrix, transform_origin, | 
 |                                position, bounds, true, false, false); | 
 |   child3->SetDrawsContent(true); | 
 |  | 
 |   child2->Set3dSortingContextId(1); | 
 |   child3->Set3dSortingContextId(1); | 
 |  | 
 |   child2->AddChild(std::move(child3)); | 
 |   child1->AddChild(std::move(child2)); | 
 |   root->AddChild(std::move(child1)); | 
 |  | 
 |   { | 
 |     LayerImplList render_surface_layer_list; | 
 |     FakeLayerTreeHostImpl::RecursiveUpdateNumChildren(root.get()); | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root.get(), root->bounds(), &render_surface_layer_list, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.can_render_to_separate_surface = true; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |     EXPECT_EQ(2u, render_surface_layer_list.size()); | 
 |  | 
 |     int count_represents_target_render_surface = 0; | 
 |     int count_represents_contributing_render_surface = 0; | 
 |     int count_represents_itself = 0; | 
 |     LayerIterator end = LayerIterator::End(&render_surface_layer_list); | 
 |     for (LayerIterator it = LayerIterator::Begin(&render_surface_layer_list); | 
 |          it != end; ++it) { | 
 |       if (it.represents_target_render_surface()) | 
 |         count_represents_target_render_surface++; | 
 |       if (it.represents_contributing_render_surface()) | 
 |         count_represents_contributing_render_surface++; | 
 |       if (it.represents_itself()) | 
 |         count_represents_itself++; | 
 |     } | 
 |  | 
 |     // Two render surfaces. | 
 |     EXPECT_EQ(2, count_represents_target_render_surface); | 
 |     // Second render surface contributes to root render surface. | 
 |     EXPECT_EQ(1, count_represents_contributing_render_surface); | 
 |     // All 4 layers represent itself. | 
 |     EXPECT_EQ(4, count_represents_itself); | 
 |   } | 
 |  | 
 |   { | 
 |     LayerImplList render_surface_layer_list; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root.get(), root->bounds(), &render_surface_layer_list, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     inputs.can_render_to_separate_surface = false; | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |     EXPECT_EQ(1u, render_surface_layer_list.size()); | 
 |  | 
 |     int count_represents_target_render_surface = 0; | 
 |     int count_represents_contributing_render_surface = 0; | 
 |     int count_represents_itself = 0; | 
 |     LayerIterator end = LayerIterator::End(&render_surface_layer_list); | 
 |     for (LayerIterator it = LayerIterator::Begin(&render_surface_layer_list); | 
 |          it != end; ++it) { | 
 |       if (it.represents_target_render_surface()) | 
 |         count_represents_target_render_surface++; | 
 |       if (it.represents_contributing_render_surface()) | 
 |         count_represents_contributing_render_surface++; | 
 |       if (it.represents_itself()) | 
 |         count_represents_itself++; | 
 |     } | 
 |  | 
 |     // Only root layer has a render surface. | 
 |     EXPECT_EQ(1, count_represents_target_render_surface); | 
 |     // No layer contributes a render surface to root render surface. | 
 |     EXPECT_EQ(0, count_represents_contributing_render_surface); | 
 |     // All 4 layers represent itself. | 
 |     EXPECT_EQ(4, count_represents_itself); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DoNotIncludeBackfaceInvisibleSurfaces) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* back_facing = AddChild<LayerImpl>(root); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(back_facing); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(back_facing); | 
 |   LayerImpl* child1 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* child2 = AddChild<LayerImpl>(render_surface2); | 
 |   child1->SetDrawsContent(true); | 
 |   child2->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(back_facing, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                false, true, true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                false, true, true); | 
 |   SetLayerPropertiesForTesting(child1, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child2, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |  | 
 |   root->SetShouldFlattenTransform(false); | 
 |   root->Set3dSortingContextId(1); | 
 |   back_facing->Set3dSortingContextId(1); | 
 |   back_facing->SetShouldFlattenTransform(false); | 
 |   render_surface1->SetDoubleSided(false); | 
 |   render_surface2->Set3dSortingContextId(2); | 
 |   render_surface2->SetDoubleSided(false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(render_surface1->sorting_context_id(), root->sorting_context_id()); | 
 |   EXPECT_NE(render_surface2->sorting_context_id(), root->sorting_context_id()); | 
 |   EXPECT_EQ(3u, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(2u, render_surface_layer_list_impl() | 
 |                     ->at(0) | 
 |                     ->render_surface() | 
 |                     ->layer_list() | 
 |                     .size()); | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl() | 
 |                     ->at(1) | 
 |                     ->render_surface() | 
 |                     ->layer_list() | 
 |                     .size()); | 
 |  | 
 |   gfx::Transform rotation_transform = identity_transform; | 
 |   rotation_transform.RotateAboutXAxis(180.0); | 
 |  | 
 |   back_facing->SetTransform(rotation_transform); | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // render_surface1 is in the same 3d rendering context as back_facing and is | 
 |   // not double sided, so it should not be in RSLL. render_surface2 is also not | 
 |   // double-sided, but will still be in RSLL as it's in a different 3d rendering | 
 |   // context. | 
 |   EXPECT_EQ(2u, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl() | 
 |                     ->at(0) | 
 |                     ->render_surface() | 
 |                     ->layer_list() | 
 |                     .size()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DoNotIncludeBackfaceInvisibleLayers) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(root); | 
 |   LayerImpl* grand_child = AddChild<LayerImpl>(child); | 
 |   grand_child->SetDrawsContent(true); | 
 |  | 
 |   child->SetDoubleSided(false); | 
 |   grand_child->SetUseParentBackfaceVisibility(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), false, true, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), false, true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grand_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), false, true, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(grand_child, render_surface_layer_list_impl() | 
 |                              ->at(0) | 
 |                              ->render_surface() | 
 |                              ->layer_list()[0]); | 
 |   gfx::Transform rotation_transform = identity_transform; | 
 |   rotation_transform.RotateAboutXAxis(180.0); | 
 |  | 
 |   child->SetTransform(rotation_transform); | 
 |   child->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(0u, render_surface_layer_list_impl() | 
 |                     ->at(0) | 
 |                     ->render_surface() | 
 |                     ->layer_list() | 
 |                     .size()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClippedByScrollParent) { | 
 |   // Checks that the simple case (being clipped by a scroll parent that would | 
 |   // have been processed before you anyhow) results in the right clips. | 
 |   // | 
 |   // + root | 
 |   //   + scroll_parent_border | 
 |   //   | + scroll_parent_clip | 
 |   //   |   + scroll_parent | 
 |   //   + scroll_child | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* scroll_parent_border = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* scroll_parent_clip = AddChild<LayerImpl>(scroll_parent_border); | 
 |   LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_parent_clip); | 
 |   LayerImpl* scroll_child = AddChild<LayerImpl>(root); | 
 |  | 
 |   scroll_parent->SetDrawsContent(true); | 
 |   scroll_child->SetDrawsContent(true); | 
 |   scroll_parent_clip->SetMasksToBounds(true); | 
 |  | 
 |   scroll_child->SetScrollParent(scroll_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> scroll_children(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_child); | 
 |   scroll_parent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(scroll_parent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 30, 30).ToString(), | 
 |             scroll_child->clip_rect().ToString()); | 
 |   EXPECT_TRUE(scroll_child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ScrollChildAndScrollParentDifferentTargets) { | 
 |   // Tests the computation of draw transform for the scroll child when its | 
 |   // target is different from its scroll parent's target. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* scroll_child_target = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* scroll_child = AddChild<LayerImpl>(scroll_child_target); | 
 |   LayerImpl* scroll_parent_target = AddChild<LayerImpl>(scroll_child_target); | 
 |   LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_parent_target); | 
 |  | 
 |   scroll_parent->SetDrawsContent(true); | 
 |   scroll_child->SetDrawsContent(true); | 
 |  | 
 |   scroll_child->SetScrollParent(scroll_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> scroll_children(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_child); | 
 |   scroll_parent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(scroll_child_target, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(scroll_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_target, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(10, 10), | 
 |                                gfx::Size(50, 50), true, false, true); | 
 |   SetLayerPropertiesForTesting(scroll_parent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(scroll_child->DrawTransform(), identity_transform); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SingularTransformSubtreesDoNotDraw) { | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetDrawsContent(true); | 
 |   LayerImpl* parent = AddChildToRoot<LayerImpl>(); | 
 |   parent->SetDrawsContent(true); | 
 |   LayerImpl* child = AddChild<LayerImpl>(parent); | 
 |   child->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, true, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(parent, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, true, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, true, | 
 |                                true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(3u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   gfx::Transform singular_transform; | 
 |   singular_transform.Scale3d( | 
 |       SkDoubleToMScalar(1.0), SkDoubleToMScalar(1.0), SkDoubleToMScalar(0.0)); | 
 |  | 
 |   child->SetTransform(singular_transform); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(2u, render_surface_layer_list_impl()->size()); | 
 |  | 
 |   // Ensure that the entire subtree under a layer with singular transform does | 
 |   // not get rendered. | 
 |   parent->SetTransform(singular_transform); | 
 |   child->SetTransform(identity_transform); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(1u, render_surface_layer_list_impl()->size()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClippedByOutOfOrderScrollParent) { | 
 |   // Checks that clipping by a scroll parent that follows you in paint order | 
 |   // still results in correct clipping. | 
 |   // | 
 |   // + root | 
 |   //   + scroll_parent_border | 
 |   //     + scroll_parent_clip | 
 |   //       + scroll_parent | 
 |   //   + scroll_child | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* scroll_parent_border = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_parent_clip = AddChild<LayerImpl>(scroll_parent_border); | 
 |   LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_parent_clip); | 
 |   LayerImpl* scroll_child = AddChild<LayerImpl>(root); | 
 |  | 
 |   scroll_parent->SetDrawsContent(true); | 
 |   scroll_child->SetDrawsContent(true); | 
 |  | 
 |   scroll_parent_clip->SetMasksToBounds(true); | 
 |  | 
 |   scroll_child->SetScrollParent(scroll_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> scroll_children(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_child); | 
 |   scroll_parent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(scroll_parent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 30, 30).ToString(), | 
 |             scroll_child->clip_rect().ToString()); | 
 |   EXPECT_TRUE(scroll_child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClippedByOutOfOrderScrollGrandparent) { | 
 |   // Checks that clipping by a scroll parent and scroll grandparent that follow | 
 |   // you in paint order still results in correct clipping. | 
 |   // | 
 |   // + root | 
 |   //   + scroll_child | 
 |   //   + scroll_parent_border | 
 |   //   | + scroll_parent_clip | 
 |   //   |   + scroll_parent | 
 |   //   + scroll_grandparent_border | 
 |   //     + scroll_grandparent_clip | 
 |   //       + scroll_grandparent | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* scroll_child = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_parent_border = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_parent_clip = AddChild<LayerImpl>(scroll_parent_border); | 
 |   LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_parent_clip); | 
 |   LayerImpl* scroll_grandparent_border = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_grandparent_clip = | 
 |       AddChild<LayerImpl>(scroll_grandparent_border); | 
 |   LayerImpl* scroll_grandparent = AddChild<LayerImpl>(scroll_grandparent_clip); | 
 |  | 
 |   scroll_parent->SetDrawsContent(true); | 
 |   scroll_grandparent->SetDrawsContent(true); | 
 |   scroll_child->SetDrawsContent(true); | 
 |  | 
 |   scroll_parent_clip->SetMasksToBounds(true); | 
 |   scroll_grandparent_clip->SetMasksToBounds(true); | 
 |  | 
 |   scroll_child->SetScrollParent(scroll_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> scroll_children(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_child); | 
 |   scroll_parent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   scroll_parent_border->SetScrollParent(scroll_grandparent); | 
 |   scroll_children.reset(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_parent_border); | 
 |   scroll_grandparent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 20, 20).ToString(), | 
 |             scroll_child->clip_rect().ToString()); | 
 |   EXPECT_TRUE(scroll_child->is_clipped()); | 
 |  | 
 |   // Despite the fact that we visited the above layers out of order to get the | 
 |   // correct clip, the layer lists should be unaffected. | 
 |   EXPECT_EQ(3u, root->render_surface()->layer_list().size()); | 
 |   EXPECT_EQ(scroll_child, root->render_surface()->layer_list().at(0)); | 
 |   EXPECT_EQ(scroll_parent, root->render_surface()->layer_list().at(1)); | 
 |   EXPECT_EQ(scroll_grandparent, root->render_surface()->layer_list().at(2)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, OutOfOrderClippingRequiresRSLLSorting) { | 
 |   // Ensures that even if we visit layers out of order, we still produce a | 
 |   // correctly ordered render surface layer list. | 
 |   // + root | 
 |   //   + scroll_child | 
 |   //   + scroll_parent_border | 
 |   //     + scroll_parent_clip | 
 |   //       + scroll_parent | 
 |   //         + render_surface2 | 
 |   //   + scroll_grandparent_border | 
 |   //     + scroll_grandparent_clip | 
 |   //       + scroll_grandparent | 
 |   //         + render_surface1 | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   LayerImpl* scroll_child = AddChild<LayerImpl>(root); | 
 |   scroll_child->SetDrawsContent(true); | 
 |  | 
 |   LayerImpl* scroll_parent_border = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_parent_clip = AddChild<LayerImpl>(scroll_parent_border); | 
 |   LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_parent_clip); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(scroll_parent); | 
 |   LayerImpl* scroll_grandparent_border = AddChild<LayerImpl>(root); | 
 |   LayerImpl* scroll_grandparent_clip = | 
 |       AddChild<LayerImpl>(scroll_grandparent_border); | 
 |   LayerImpl* scroll_grandparent = AddChild<LayerImpl>(scroll_grandparent_clip); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(scroll_grandparent); | 
 |  | 
 |   scroll_parent->SetDrawsContent(true); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   scroll_grandparent->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |  | 
 |   scroll_parent_clip->SetMasksToBounds(true); | 
 |   scroll_grandparent_clip->SetMasksToBounds(true); | 
 |  | 
 |   scroll_child->SetScrollParent(scroll_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> scroll_children(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_child); | 
 |   scroll_parent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   scroll_parent_border->SetScrollParent(scroll_grandparent); | 
 |   scroll_children.reset(new std::set<LayerImpl*>); | 
 |   scroll_children->insert(scroll_parent_border); | 
 |   scroll_grandparent->SetScrollChildren(scroll_children.release()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(20, 20), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_grandparent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(scroll_parent_border, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent_clip, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(scroll_child, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRUE(root->render_surface()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 20, 20).ToString(), | 
 |             scroll_child->clip_rect().ToString()); | 
 |   EXPECT_TRUE(scroll_child->is_clipped()); | 
 |  | 
 |   // Despite the fact that we had to process the layers out of order to get the | 
 |   // right clip, our render_surface_layer_list's order should be unaffected. | 
 |   EXPECT_EQ(3u, render_surface_layer_list_impl()->size()); | 
 |   EXPECT_EQ(root, render_surface_layer_list_impl()->at(0)); | 
 |   EXPECT_EQ(render_surface2, render_surface_layer_list_impl()->at(1)); | 
 |   EXPECT_EQ(render_surface1, render_surface_layer_list_impl()->at(2)); | 
 |   EXPECT_TRUE(render_surface_layer_list_impl()->at(0)->render_surface()); | 
 |   EXPECT_TRUE(render_surface_layer_list_impl()->at(1)->render_surface()); | 
 |   EXPECT_TRUE(render_surface_layer_list_impl()->at(2)->render_surface()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, FixedPositionWithInterveningRenderSurface) { | 
 |   // Ensures that when we have a render surface between a fixed position layer | 
 |   // and its container, we compute the fixed position layer's draw transform | 
 |   // with respect to that intervening render surface, not with respect to its | 
 |   // container's render target. | 
 |   // | 
 |   // + root | 
 |   //   + render_surface | 
 |   //     + fixed | 
 |   //       + child | 
 |   // | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> render_surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> fixed = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(render_surface); | 
 |   render_surface->AddChild(fixed); | 
 |   fixed->AddChild(child); | 
 |  | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |   render_surface->SetForceRenderSurface(true); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), gfx::Transform(), gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(render_surface.get(), gfx::Transform(), | 
 |                                gfx::Point3F(), gfx::PointF(7.f, 9.f), | 
 |                                gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(fixed.get(), gfx::Transform(), gfx::Point3F(), | 
 |                                gfx::PointF(10.f, 15.f), gfx::Size(50, 50), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child.get(), gfx::Transform(), gfx::Point3F(), | 
 |                                gfx::PointF(1.f, 2.f), gfx::Size(50, 50), true, | 
 |                                false); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   TransformTree& tree = host()->property_trees()->transform_tree; | 
 |  | 
 |   gfx::Transform expected_fixed_draw_transform; | 
 |   expected_fixed_draw_transform.Translate(10.f, 15.f); | 
 |   EXPECT_EQ(expected_fixed_draw_transform, | 
 |             DrawTransformFromPropertyTrees(fixed.get(), tree)); | 
 |  | 
 |   gfx::Transform expected_fixed_screen_space_transform; | 
 |   expected_fixed_screen_space_transform.Translate(17.f, 24.f); | 
 |   EXPECT_EQ(expected_fixed_screen_space_transform, | 
 |             ScreenSpaceTransformFromPropertyTrees(fixed.get(), tree)); | 
 |  | 
 |   gfx::Transform expected_child_draw_transform; | 
 |   expected_child_draw_transform.Translate(11.f, 17.f); | 
 |   EXPECT_EQ(expected_child_draw_transform, | 
 |             DrawTransformFromPropertyTrees(child.get(), tree)); | 
 |  | 
 |   gfx::Transform expected_child_screen_space_transform; | 
 |   expected_child_screen_space_transform.Translate(18.f, 26.f); | 
 |   EXPECT_EQ(expected_child_screen_space_transform, | 
 |             ScreenSpaceTransformFromPropertyTrees(child.get(), tree)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ScrollCompensationWithRounding) { | 
 |   // This test verifies that a scrolling layer that gets snapped to | 
 |   // integer coordinates doesn't move a fixed position child. | 
 |   // | 
 |   // + root | 
 |   //   + container | 
 |   //     + scroller | 
 |   //       + fixed | 
 |   // | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<LayerImpl> container = | 
 |       LayerImpl::Create(host_impl.active_tree(), 2); | 
 |   LayerImpl* container_layer = container.get(); | 
 |   scoped_ptr<LayerImpl> scroller = | 
 |       LayerImpl::Create(host_impl.active_tree(), 3); | 
 |   LayerImpl* scroll_layer = scroller.get(); | 
 |   scoped_ptr<LayerImpl> fixed = LayerImpl::Create(host_impl.active_tree(), 4); | 
 |   LayerImpl* fixed_layer = fixed.get(); | 
 |  | 
 |   container->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |  | 
 |   scroller->SetScrollClipLayer(container->id()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   gfx::Transform container_transform; | 
 |   container_transform.Translate3d(10.0, 20.0, 0.0); | 
 |   gfx::Vector2dF container_offset = container_transform.To2dTranslation(); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(container.get(), container_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroller.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(fixed.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   root->SetDrawsContent(true); | 
 |   container->SetDrawsContent(true); | 
 |   scroller->SetDrawsContent(true); | 
 |   fixed->SetDrawsContent(true); | 
 |   scroller->AddChild(std::move(fixed)); | 
 |   container->AddChild(std::move(scroller)); | 
 |   root->AddChild(std::move(container)); | 
 |  | 
 |   // Rounded to integers already. | 
 |   { | 
 |     gfx::Vector2dF scroll_delta(3.0, 5.0); | 
 |     scroll_layer->SetScrollDelta(scroll_delta); | 
 |  | 
 |     LayerImplList render_surface_layer_list; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root.get(), root->bounds(), &render_surface_layer_list, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |     EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |         container_layer->draw_properties().screen_space_transform, | 
 |         fixed_layer->draw_properties().screen_space_transform); | 
 |     EXPECT_VECTOR_EQ( | 
 |         fixed_layer->draw_properties().screen_space_transform.To2dTranslation(), | 
 |         container_offset); | 
 |     EXPECT_VECTOR_EQ(scroll_layer->draw_properties() | 
 |                          .screen_space_transform.To2dTranslation(), | 
 |                      container_offset - scroll_delta); | 
 |   } | 
 |  | 
 |   // Scroll delta requiring rounding. | 
 |   { | 
 |     gfx::Vector2dF scroll_delta(4.1f, 8.1f); | 
 |     scroll_layer->SetScrollDelta(scroll_delta); | 
 |  | 
 |     gfx::Vector2dF rounded_scroll_delta(4.f, 8.f); | 
 |  | 
 |     LayerImplList render_surface_layer_list; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root.get(), root->bounds(), &render_surface_layer_list, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |     EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |         container_layer->draw_properties().screen_space_transform, | 
 |         fixed_layer->draw_properties().screen_space_transform); | 
 |     EXPECT_VECTOR_EQ( | 
 |         fixed_layer->draw_properties().screen_space_transform.To2dTranslation(), | 
 |         container_offset); | 
 |     EXPECT_VECTOR_EQ(scroll_layer->draw_properties() | 
 |                          .screen_space_transform.To2dTranslation(), | 
 |                      container_offset - rounded_scroll_delta); | 
 |   } | 
 |  | 
 |   // Scale is applied earlier in the tree. | 
 |   { | 
 |     gfx::Transform scaled_container_transform = container_transform; | 
 |     scaled_container_transform.Scale3d(2.0, 2.0, 1.0); | 
 |     container_layer->SetTransform(scaled_container_transform); | 
 |     root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |     root->layer_tree_impl() | 
 |         ->property_trees() | 
 |         ->transform_tree.set_source_to_parent_updates_allowed(true); | 
 |  | 
 |     gfx::Vector2dF scroll_delta(4.5f, 8.5f); | 
 |     scroll_layer->SetScrollDelta(scroll_delta); | 
 |  | 
 |     LayerImplList render_surface_layer_list; | 
 |     root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |     LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |         root.get(), root->bounds(), &render_surface_layer_list, | 
 |         root->layer_tree_impl()->current_render_surface_list_id()); | 
 |     LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |     EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |         container_layer->draw_properties().screen_space_transform, | 
 |         fixed_layer->draw_properties().screen_space_transform); | 
 |     EXPECT_VECTOR_EQ( | 
 |         fixed_layer->draw_properties().screen_space_transform.To2dTranslation(), | 
 |         container_offset); | 
 |  | 
 |     container_layer->SetTransform(container_transform); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        ScrollCompensationMainScrollOffsetFractionalPart) { | 
 |   // This test verifies that a scrolling layer that has fractional scroll offset | 
 |   // from main doesn't move a fixed position child. | 
 |   // | 
 |   // + root | 
 |   //   + container | 
 |   //     + scroller | 
 |   //       + fixed | 
 |   // | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   host_impl.CreatePendingTree(); | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<LayerImpl> container = | 
 |       LayerImpl::Create(host_impl.active_tree(), 2); | 
 |   LayerImpl* container_layer = container.get(); | 
 |   scoped_ptr<LayerImpl> scroller = | 
 |       LayerImpl::Create(host_impl.active_tree(), 3); | 
 |   LayerImpl* scroll_layer = scroller.get(); | 
 |   scoped_ptr<LayerImpl> fixed = LayerImpl::Create(host_impl.active_tree(), 4); | 
 |   LayerImpl* fixed_layer = fixed.get(); | 
 |  | 
 |   container->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   container->SetDrawsContent(true); | 
 |   fixed->SetDrawsContent(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |  | 
 |   scroller->SetDrawsContent(true); | 
 |   scroller->SetScrollClipLayer(container->id()); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   gfx::Transform container_transform; | 
 |   container_transform.Translate3d(10.0, 20.0, 0.0); | 
 |   gfx::Vector2dF container_offset = container_transform.To2dTranslation(); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(container.get(), container_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(40, 40), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(scroller.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(0.0, 0.0), | 
 |                                gfx::Size(30, 30), true, false, false); | 
 |  | 
 |   gfx::ScrollOffset scroll_offset(3.3, 4.2); | 
 |   gfx::Vector2dF main_scroll_fractional_part(0.3f, 0.2f); | 
 |   gfx::Vector2dF scroll_delta(0.1f, 0.4f); | 
 |   // Blink only uses the integer part of the scroll_offset for fixed | 
 |   // position layer. | 
 |   SetLayerPropertiesForTesting(fixed.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(3.0f, 4.0f), gfx::Size(50, 50), true, | 
 |                                false, false); | 
 |   scroll_layer->PushScrollOffsetFromMainThread(scroll_offset); | 
 |   scroll_layer->SetScrollDelta(scroll_delta); | 
 |   scroll_layer->SetScrollCompensationAdjustment(main_scroll_fractional_part); | 
 |  | 
 |   scroller->AddChild(std::move(fixed)); | 
 |   container->AddChild(std::move(scroller)); | 
 |   root->AddChild(std::move(container)); | 
 |  | 
 |   LayerImplList render_surface_layer_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root.get(), root->bounds(), &render_surface_layer_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ( | 
 |       container_layer->draw_properties().screen_space_transform, | 
 |       fixed_layer->draw_properties().screen_space_transform); | 
 |   EXPECT_VECTOR_EQ( | 
 |       fixed_layer->draw_properties().screen_space_transform.To2dTranslation(), | 
 |       container_offset); | 
 |  | 
 |   gfx::ScrollOffset effective_scroll_offset = | 
 |       ScrollOffsetWithDelta(scroll_offset, scroll_delta); | 
 |   gfx::Vector2d rounded_effective_scroll_offset = | 
 |       ToRoundedVector2d(ScrollOffsetToVector2dF(effective_scroll_offset)); | 
 |   EXPECT_VECTOR_EQ( | 
 |       scroll_layer->draw_properties().screen_space_transform.To2dTranslation(), | 
 |       container_offset - rounded_effective_scroll_offset); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        ScrollSnappingWithAnimatedScreenSpaceTransform) { | 
 |   // This test verifies that a scrolling layer whose screen space transform is | 
 |   // animating doesn't get snapped to integer coordinates. | 
 |   // | 
 |   // + root | 
 |   //   + animated layer | 
 |   //     + surface | 
 |   //       + container | 
 |   //         + scroller | 
 |   // | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* animated_layer = AddChildToRoot<FakePictureLayerImpl>(); | 
 |   LayerImpl* surface = AddChild<LayerImpl>(animated_layer); | 
 |   LayerImpl* container = AddChild<LayerImpl>(surface); | 
 |   LayerImpl* scroller = AddChild<LayerImpl>(container); | 
 |   scroller->SetScrollClipLayer(container->id()); | 
 |   scroller->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   gfx::Transform start_scale; | 
 |   start_scale.Scale(1.5f, 1.5f); | 
 |   SetLayerPropertiesForTesting(root, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(animated_layer, start_scale, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(surface, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(container, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(scroller, identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |  | 
 |   gfx::Transform end_scale; | 
 |   end_scale.Scale(2.f, 2.f); | 
 |   TransformOperations start_operations; | 
 |   start_operations.AppendMatrix(start_scale); | 
 |   TransformOperations end_operations; | 
 |   end_operations.AppendMatrix(end_scale); | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(animated_layer->id(), timeline_impl(), | 
 |                                           1.0, start_operations, | 
 |                                           end_operations); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(animated_layer, 1.0, start_operations, | 
 |                                 end_operations); | 
 |   } | 
 |   gfx::Vector2dF scroll_delta(5.f, 9.f); | 
 |   scroller->SetScrollDelta(scroll_delta); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   gfx::Vector2dF expected_draw_transform_translation(-7.5f, -13.5f); | 
 |   EXPECT_VECTOR2DF_EQ(expected_draw_transform_translation, | 
 |                       scroller->DrawTransform().To2dTranslation()); | 
 | } | 
 |  | 
 | class AnimationScaleFactorTrackingLayerImpl : public LayerImpl { | 
 |  public: | 
 |   static scoped_ptr<AnimationScaleFactorTrackingLayerImpl> Create( | 
 |       LayerTreeImpl* tree_impl, | 
 |       int id) { | 
 |     return make_scoped_ptr( | 
 |         new AnimationScaleFactorTrackingLayerImpl(tree_impl, id)); | 
 |   } | 
 |  | 
 |   ~AnimationScaleFactorTrackingLayerImpl() override {} | 
 |  | 
 |  private: | 
 |   explicit AnimationScaleFactorTrackingLayerImpl(LayerTreeImpl* tree_impl, | 
 |                                                  int id) | 
 |       : LayerImpl(tree_impl, id) { | 
 |     SetDrawsContent(true); | 
 |   } | 
 | }; | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, MaximumAnimationScaleFactor) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   LayerTreeSettings settings = host()->settings(); | 
 |   settings.layer_transforms_should_scale_layer_contents = true; | 
 |   FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider, | 
 |                                   &shared_bitmap_manager, &task_graph_runner); | 
 |   gfx::Transform identity_matrix; | 
 |   scoped_ptr<AnimationScaleFactorTrackingLayerImpl> grand_parent = | 
 |       AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<AnimationScaleFactorTrackingLayerImpl> parent = | 
 |       AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 2); | 
 |   scoped_ptr<AnimationScaleFactorTrackingLayerImpl> child = | 
 |       AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 3); | 
 |   scoped_ptr<AnimationScaleFactorTrackingLayerImpl> grand_child = | 
 |       AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 4); | 
 |  | 
 |   AnimationScaleFactorTrackingLayerImpl* parent_raw = parent.get(); | 
 |   AnimationScaleFactorTrackingLayerImpl* child_raw = child.get(); | 
 |   AnimationScaleFactorTrackingLayerImpl* grand_child_raw = grand_child.get(); | 
 |  | 
 |   child->AddChild(std::move(grand_child)); | 
 |   parent->AddChild(std::move(child)); | 
 |   grand_parent->AddChild(std::move(parent)); | 
 |  | 
 |   SetLayerPropertiesForTesting(grand_parent.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 2), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(parent_raw, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child_raw, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                false); | 
 |  | 
 |   SetLayerPropertiesForTesting(grand_child_raw, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // No layers have animations. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   TransformOperations translation; | 
 |   translation.AppendTranslate(1.f, 2.f, 3.f); | 
 |  | 
 |   scoped_refptr<AnimationTimeline> timeline; | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     timeline = AnimationTimeline::Create(AnimationIdProvider::NextTimelineId()); | 
 |     host_impl.animation_host()->AddAnimationTimeline(timeline); | 
 |  | 
 |     AddAnimatedTransformToLayerWithPlayer(parent_raw->id(), timeline, 1.0, | 
 |                                           TransformOperations(), translation); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(parent_raw, 1.0, TransformOperations(), | 
 |                                 translation); | 
 |   } | 
 |  | 
 |   // No layers have scale-affecting animations. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   TransformOperations scale; | 
 |   scale.AppendScale(5.f, 4.f, 3.f); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(child_raw->id(), timeline, 1.0, | 
 |                                           TransformOperations(), scale); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(child_raw, 1.0, TransformOperations(), scale); | 
 |   } | 
 |   child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // Only |child| has a scale-affecting animation. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(5.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       5.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(1.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       1.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(grand_parent->id(), timeline, 1.0, | 
 |                                           TransformOperations(), scale); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(grand_parent.get(), 1.0, TransformOperations(), | 
 |                                 scale); | 
 |   } | 
 |   grand_parent->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |grand_parent| and |child| have scale-affecting animations. | 
 |   EXPECT_EQ(5.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(5.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   // We don't support combining animated scales from two nodes; 0.f means | 
 |   // that the maximum scale could not be computed. | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(1.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(1.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(parent_raw->id(), timeline, 1.0, | 
 |                                           TransformOperations(), scale); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(parent_raw, 1.0, TransformOperations(), scale); | 
 |   } | 
 |   parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |grand_parent|, |parent|, and |child| have scale-affecting animations. | 
 |   EXPECT_EQ(5.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(1.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AbortAnimationsOnLayerWithPlayer(grand_parent->id(), timeline, | 
 |                                      Animation::TRANSFORM); | 
 |     AbortAnimationsOnLayerWithPlayer(parent_raw->id(), timeline, | 
 |                                      Animation::TRANSFORM); | 
 |     AbortAnimationsOnLayerWithPlayer(child_raw->id(), timeline, | 
 |                                      Animation::TRANSFORM); | 
 |   } else { | 
 |     grand_parent->layer_animation_controller()->AbortAnimations( | 
 |         Animation::TRANSFORM); | 
 |     parent_raw->layer_animation_controller()->AbortAnimations( | 
 |         Animation::TRANSFORM); | 
 |     child_raw->layer_animation_controller()->AbortAnimations( | 
 |         Animation::TRANSFORM); | 
 |   } | 
 |  | 
 |   TransformOperations perspective; | 
 |   perspective.AppendPerspective(10.f); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(child_raw->id(), timeline, 1.0, | 
 |                                           TransformOperations(), perspective); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(child_raw, 1.0, TransformOperations(), | 
 |                                 perspective); | 
 |   } | 
 |   child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |child| has a scale-affecting animation but computing the maximum of this | 
 |   // animation is not supported. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AbortAnimationsOnLayerWithPlayer(child_raw->id(), timeline, | 
 |                                      Animation::TRANSFORM); | 
 |   } else { | 
 |     child_raw->layer_animation_controller()->AbortAnimations( | 
 |         Animation::TRANSFORM); | 
 |   } | 
 |   gfx::Transform scale_matrix; | 
 |   scale_matrix.Scale(1.f, 2.f); | 
 |   grand_parent->SetTransform(scale_matrix); | 
 |   parent_raw->SetTransform(scale_matrix); | 
 |   grand_parent->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(parent_raw->id(), timeline, 1.0, | 
 |                                           TransformOperations(), scale); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(parent_raw, 1.0, TransformOperations(), scale); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |grand_parent| and |parent| each have scale 2.f. |parent| has a  scale | 
 |   // animation with maximum scale 5.f. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(10.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(10.f, | 
 |             child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       10.f, | 
 |       grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(2.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(2.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       2.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   gfx::Transform perspective_matrix; | 
 |   perspective_matrix.ApplyPerspectiveDepth(2.f); | 
 |   child_raw->SetTransform(perspective_matrix); | 
 |   grand_parent->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |child| has a transform that's neither a translation nor a scale. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(10.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(2.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   parent_raw->SetTransform(perspective_matrix); | 
 |   grand_parent->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |parent| and |child| have transforms that are neither translations nor | 
 |   // scales. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 |  | 
 |   parent_raw->SetTransform(identity_matrix); | 
 |   child_raw->SetTransform(identity_matrix); | 
 |   grand_parent->SetTransform(perspective_matrix); | 
 |   grand_parent->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent.get()); | 
 |  | 
 |   // |grand_parent| has a transform that's neither a translation nor a scale. | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, child_raw->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, grand_child_raw->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   EXPECT_EQ(0.f, | 
 |             grand_parent->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             parent_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ(0.f, | 
 |             child_raw->draw_properties().starting_animation_contents_scale); | 
 |   EXPECT_EQ( | 
 |       0.f, | 
 |       grand_child_raw->draw_properties().starting_animation_contents_scale); | 
 | } | 
 |  | 
 | static int membership_id(LayerImpl* layer) { | 
 |   return layer->draw_properties().last_drawn_render_surface_layer_list_id; | 
 | } | 
 |  | 
 | static void GatherDrawnLayers(LayerImplList* rsll, | 
 |                               std::set<LayerImpl*>* drawn_layers) { | 
 |   for (LayerIterator it = LayerIterator::Begin(rsll), | 
 |                      end = LayerIterator::End(rsll); | 
 |        it != end; ++it) { | 
 |     LayerImpl* layer = *it; | 
 |     if (it.represents_itself()) | 
 |       drawn_layers->insert(layer); | 
 |  | 
 |     if (!it.represents_contributing_render_surface()) | 
 |       continue; | 
 |  | 
 |     if (layer->mask_layer()) | 
 |       drawn_layers->insert(layer->mask_layer()); | 
 |     if (layer->replica_layer() && layer->replica_layer()->mask_layer()) | 
 |       drawn_layers->insert(layer->replica_layer()->mask_layer()); | 
 |   } | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceLayerListMembership) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   scoped_ptr<LayerImpl> grand_parent = | 
 |       LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<LayerImpl> parent = LayerImpl::Create(host_impl.active_tree(), 3); | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.active_tree(), 5); | 
 |   scoped_ptr<LayerImpl> grand_child1 = | 
 |       LayerImpl::Create(host_impl.active_tree(), 7); | 
 |   scoped_ptr<LayerImpl> grand_child2 = | 
 |       LayerImpl::Create(host_impl.active_tree(), 9); | 
 |  | 
 |   LayerImpl* grand_parent_raw = grand_parent.get(); | 
 |   LayerImpl* parent_raw = parent.get(); | 
 |   LayerImpl* child_raw = child.get(); | 
 |   LayerImpl* grand_child1_raw = grand_child1.get(); | 
 |   LayerImpl* grand_child2_raw = grand_child2.get(); | 
 |  | 
 |   child->AddChild(std::move(grand_child1)); | 
 |   child->AddChild(std::move(grand_child2)); | 
 |   parent->AddChild(std::move(child)); | 
 |   grand_parent->AddChild(std::move(parent)); | 
 |  | 
 |   SetLayerPropertiesForTesting(grand_parent_raw, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 2), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(parent_raw, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                false); | 
 |  | 
 |   SetLayerPropertiesForTesting(child_raw, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 2), true, false, | 
 |                                false); | 
 |  | 
 |   SetLayerPropertiesForTesting(grand_child1_raw, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 2), | 
 |                                true, false, false); | 
 |  | 
 |   SetLayerPropertiesForTesting(grand_child2_raw, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 2), | 
 |                                true, false, false); | 
 |  | 
 |   // Start with nothing being drawn. | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   int member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   std::set<LayerImpl*> expected; | 
 |   std::set<LayerImpl*> actual; | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // If we force render surface, but none of the layers are in the layer list, | 
 |   // then this layer should not appear in RSLL. | 
 |   grand_child1_raw->SetHasRenderSurface(true); | 
 |   grand_child1_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // However, if we say that this layer also draws content, it will appear in | 
 |   // RSLL. | 
 |   grand_child1_raw->SetDrawsContent(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_child1_raw); | 
 |  | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Now child is forced to have a render surface, and one if its children draws | 
 |   // content. | 
 |   grand_child1_raw->SetDrawsContent(false); | 
 |   grand_child1_raw->SetHasRenderSurface(false); | 
 |   grand_child1_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   child_raw->SetHasRenderSurface(true); | 
 |   grand_child2_raw->SetDrawsContent(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_child2_raw); | 
 |  | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Add a mask layer to child. | 
 |   child_raw->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 6)); | 
 |   child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw->mask_layer())); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_child2_raw); | 
 |   expected.insert(child_raw->mask_layer()); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_child2_raw); | 
 |   expected.insert(child_raw->mask_layer()); | 
 |  | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Add replica mask layer. | 
 |   scoped_ptr<LayerImpl> replica_layer = | 
 |       LayerImpl::Create(host_impl.active_tree(), 20); | 
 |   replica_layer->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 21)); | 
 |   child_raw->SetReplicaLayer(std::move(replica_layer)); | 
 |   child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw->mask_layer())); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw->replica_layer()->mask_layer())); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_child2_raw); | 
 |   expected.insert(child_raw->mask_layer()); | 
 |   expected.insert(child_raw->replica_layer()->mask_layer()); | 
 |  | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   child_raw->TakeReplicaLayer(); | 
 |  | 
 |   // With nothing drawing, we should have no layers. | 
 |   grand_child2_raw->SetDrawsContent(false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw)); | 
 |   EXPECT_NE(member_id, membership_id(child_raw->mask_layer())); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   // Child itself draws means that we should have the child and the mask in the | 
 |   // list. | 
 |   child_raw->SetDrawsContent(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_NE(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_NE(member_id, membership_id(parent_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw->mask_layer())); | 
 |   EXPECT_NE(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_NE(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(child_raw); | 
 |   expected.insert(child_raw->mask_layer()); | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 |  | 
 |   child_raw->TakeMaskLayer(); | 
 |   child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |  | 
 |   // Now everyone's a member! | 
 |   grand_parent_raw->SetDrawsContent(true); | 
 |   parent_raw->SetDrawsContent(true); | 
 |   child_raw->SetDrawsContent(true); | 
 |   grand_child1_raw->SetDrawsContent(true); | 
 |   grand_child2_raw->SetDrawsContent(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(grand_parent_raw); | 
 |   member_id = render_surface_layer_list_count(); | 
 |  | 
 |   EXPECT_EQ(member_id, membership_id(grand_parent_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(parent_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(child_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child1_raw)); | 
 |   EXPECT_EQ(member_id, membership_id(grand_child2_raw)); | 
 |  | 
 |   expected.clear(); | 
 |   expected.insert(grand_parent_raw); | 
 |   expected.insert(parent_raw); | 
 |   expected.insert(child_raw); | 
 |   expected.insert(grand_child1_raw); | 
 |   expected.insert(grand_child2_raw); | 
 |  | 
 |   actual.clear(); | 
 |   GatherDrawnLayers(render_surface_layer_list_impl(), &actual); | 
 |   EXPECT_EQ(expected, actual); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, DrawPropertyScales) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   LayerTreeSettings settings = host()->settings(); | 
 |   settings.layer_transforms_should_scale_layer_contents = true; | 
 |   FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider, | 
 |                                   &shared_bitmap_manager, &task_graph_runner); | 
 |  | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   LayerImpl* root_layer = root.get(); | 
 |   scoped_ptr<LayerImpl> child1 = LayerImpl::Create(host_impl.active_tree(), 2); | 
 |   LayerImpl* child1_layer = child1.get(); | 
 |   scoped_ptr<LayerImpl> child2 = LayerImpl::Create(host_impl.active_tree(), 3); | 
 |   LayerImpl* child2_layer = child2.get(); | 
 |  | 
 |   root->AddChild(std::move(child1)); | 
 |   root->AddChild(std::move(child2)); | 
 |   root->SetForceRenderSurface(true); | 
 |   root->SetDrawsContent(true); | 
 |  | 
 |   gfx::Transform identity_matrix, scale_transform_child1, | 
 |       scale_transform_child2; | 
 |   scale_transform_child1.Scale(2, 3); | 
 |   scale_transform_child2.Scale(4, 5); | 
 |  | 
 |   SetLayerPropertiesForTesting(root_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1, 1), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child1_layer, scale_transform_child1, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 1), | 
 |                                true, false, false); | 
 |  | 
 |   child1_layer->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 4)); | 
 |   child1_layer->SetDrawsContent(true); | 
 |  | 
 |   scoped_ptr<LayerImpl> replica_layer = | 
 |       LayerImpl::Create(host_impl.active_tree(), 5); | 
 |   replica_layer->SetMaskLayer(LayerImpl::Create(host_impl.active_tree(), 6)); | 
 |   child1_layer->SetReplicaLayer(std::move(replica_layer)); | 
 |   child1_layer->SetHasRenderSurface(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root_layer); | 
 |  | 
 |   TransformOperations scale; | 
 |   scale.AppendScale(5.f, 8.f, 3.f); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     scoped_refptr<AnimationTimeline> timeline = | 
 |         AnimationTimeline::Create(AnimationIdProvider::NextTimelineId()); | 
 |     host_impl.animation_host()->AddAnimationTimeline(timeline); | 
 |  | 
 |     AddAnimatedTransformToLayerWithPlayer(child2_layer->id(), timeline, 1.0, | 
 |                                           TransformOperations(), scale); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(child2_layer, 1.0, TransformOperations(), | 
 |                                 scale); | 
 |   } | 
 |  | 
 |   SetLayerPropertiesForTesting(child2_layer, scale_transform_child2, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(1, 1), | 
 |                                true, false, false); | 
 |   child2_layer->SetDrawsContent(true); | 
 |  | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root_layer); | 
 |  | 
 |   EXPECT_FLOAT_EQ(1.f, root_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(3.f, child1_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(3.f, child1_layer->mask_layer()->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(5.f, child2_layer->GetIdealContentsScale()); | 
 |  | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, root_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, child1_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->replica_layer() | 
 |                       ->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       8.f, child2_layer->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   // Changing page-scale would affect ideal_contents_scale and | 
 |   // maximum_animation_contents_scale. | 
 |  | 
 |   float page_scale_factor = 3.f; | 
 |   float device_scale_factor = 1.0f; | 
 |   std::vector<LayerImpl*> render_surface_layer_list; | 
 |   gfx::Size device_viewport_size = | 
 |       gfx::Size(root_layer->bounds().width() * device_scale_factor, | 
 |                 root_layer->bounds().height() * device_scale_factor); | 
 |   root_layer->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root_layer, device_viewport_size, &render_surface_layer_list, | 
 |       root_layer->layer_tree_impl()->current_render_surface_list_id()); | 
 |  | 
 |   inputs.page_scale_factor = page_scale_factor; | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   inputs.page_scale_layer = root_layer; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   EXPECT_FLOAT_EQ(3.f, root_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(9.f, child1_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(9.f, child1_layer->mask_layer()->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ( | 
 |       9.f, | 
 |       child1_layer->replica_layer()->mask_layer()->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(15.f, child2_layer->GetIdealContentsScale()); | 
 |  | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, root_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, child1_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->replica_layer() | 
 |                       ->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       24.f, child2_layer->draw_properties().maximum_animation_contents_scale); | 
 |  | 
 |   // Changing device-scale would affect ideal_contents_scale and | 
 |   // maximum_animation_contents_scale. | 
 |  | 
 |   device_scale_factor = 4.0f; | 
 |   inputs.device_scale_factor = device_scale_factor; | 
 |   inputs.can_adjust_raster_scales = true; | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   EXPECT_FLOAT_EQ(12.f, root_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(36.f, child1_layer->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(36.f, child1_layer->mask_layer()->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ( | 
 |       36.f, | 
 |       child1_layer->replica_layer()->mask_layer()->GetIdealContentsScale()); | 
 |   EXPECT_FLOAT_EQ(60.f, child2_layer->GetIdealContentsScale()); | 
 |  | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, root_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       0.f, child1_layer->draw_properties().maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ(0.f, | 
 |                   child1_layer->replica_layer() | 
 |                       ->mask_layer() | 
 |                       ->draw_properties() | 
 |                       .maximum_animation_contents_scale); | 
 |   EXPECT_FLOAT_EQ( | 
 |       96.f, child2_layer->draw_properties().maximum_animation_contents_scale); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleContentRectInChildRenderSurface) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), | 
 |                                gfx::Transform(), | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(768 / 2, 3000), | 
 |                                true, | 
 |                                false); | 
 |   root->SetIsDrawable(true); | 
 |  | 
 |   scoped_refptr<Layer> clip = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(clip.get(), | 
 |                                gfx::Transform(), | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(768 / 2, 10000), | 
 |                                true, | 
 |                                false); | 
 |   clip->SetMasksToBounds(true); | 
 |  | 
 |   scoped_refptr<Layer> content = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(content.get(), | 
 |                                gfx::Transform(), | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                gfx::Size(768 / 2, 10000), | 
 |                                true, | 
 |                                false); | 
 |   content->SetIsDrawable(true); | 
 |   content->SetForceRenderSurface(true); | 
 |  | 
 |   root->AddChild(clip); | 
 |   clip->AddChild(content); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Size device_viewport_size(768, 582); | 
 |   LayerTreeHostCommon::CalcDrawPropsMainInputs inputs(host()->root_layer(), | 
 |                                                       device_viewport_size); | 
 |   inputs.device_scale_factor = 2.f; | 
 |   inputs.page_scale_factor = 1.f; | 
 |   inputs.page_scale_layer = NULL; | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   // Layers in the root render surface have their visible content rect clipped | 
 |   // by the viewport. | 
 |   EXPECT_EQ(gfx::Rect(768 / 2, 582 / 2), | 
 |             root->visible_rect_from_property_trees()); | 
 |  | 
 |   // Layers drawing to a child render surface should still have their visible | 
 |   // content rect clipped by the viewport. | 
 |   EXPECT_EQ(gfx::Rect(768 / 2, 582 / 2), | 
 |             content->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, BoundsDeltaAffectVisibleContentRect) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |  | 
 |   // Set two layers: the root layer clips it's child, | 
 |   // the child draws its content. | 
 |  | 
 |   gfx::Size root_size = gfx::Size(300, 500); | 
 |  | 
 |   // Sublayer should be bigger than the root enlarged by bounds_delta. | 
 |   gfx::Size sublayer_size = gfx::Size(300, 1000); | 
 |  | 
 |   // Device viewport accomidated the root and the top controls. | 
 |   gfx::Size device_viewport_size = gfx::Size(300, 600); | 
 |   gfx::Transform identity_matrix; | 
 |  | 
 |   host_impl.SetViewportSize(device_viewport_size); | 
 |   host_impl.active_tree()->SetRootLayer( | 
 |       LayerImpl::Create(host_impl.active_tree(), 1)); | 
 |  | 
 |   LayerImpl* root = host_impl.active_tree()->root_layer(); | 
 |   SetLayerPropertiesForTesting(root, | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                root_size, | 
 |                                false, | 
 |                                false, | 
 |                                true); | 
 |   root->SetMasksToBounds(true); | 
 |  | 
 |   root->AddChild(LayerImpl::Create(host_impl.active_tree(), 2)); | 
 |  | 
 |   LayerImpl* sublayer = root->child_at(0); | 
 |   SetLayerPropertiesForTesting(sublayer, | 
 |                                identity_matrix, | 
 |                                gfx::Point3F(), | 
 |                                gfx::PointF(), | 
 |                                sublayer_size, | 
 |                                false, | 
 |                                false, | 
 |                                false); | 
 |   sublayer->SetDrawsContent(true); | 
 |  | 
 |   LayerImplList layer_impl_list; | 
 |   root->layer_tree_impl()->IncrementRenderSurfaceListIdForTesting(); | 
 |   LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs( | 
 |       root, device_viewport_size, &layer_impl_list, | 
 |       root->layer_tree_impl()->current_render_surface_list_id()); | 
 |  | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |   EXPECT_EQ(gfx::Rect(root_size), sublayer->visible_layer_rect()); | 
 |  | 
 |   root->SetBoundsDelta(gfx::Vector2dF(0.0, 50.0)); | 
 |   LayerTreeHostCommon::CalculateDrawProperties(&inputs); | 
 |  | 
 |   gfx::Rect affected_by_delta(0, 0, root_size.width(), | 
 |                               root_size.height() + 50); | 
 |   EXPECT_EQ(affected_by_delta, sublayer->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, NodesAffectedByBoundsDeltaGetUpdated) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> inner_viewport_container_layer = | 
 |       Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> inner_viewport_scroll_layer = | 
 |       Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> outer_viewport_container_layer = | 
 |       Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> outer_viewport_scroll_layer = | 
 |       Layer::Create(layer_settings()); | 
 |  | 
 |   root->AddChild(inner_viewport_container_layer); | 
 |   inner_viewport_container_layer->AddChild(inner_viewport_scroll_layer); | 
 |   inner_viewport_scroll_layer->AddChild(outer_viewport_container_layer); | 
 |   outer_viewport_container_layer->AddChild(outer_viewport_scroll_layer); | 
 |  | 
 |   inner_viewport_scroll_layer->SetScrollClipLayerId( | 
 |       inner_viewport_container_layer->id()); | 
 |   outer_viewport_scroll_layer->SetScrollClipLayerId( | 
 |       outer_viewport_container_layer->id()); | 
 |  | 
 |   inner_viewport_scroll_layer->SetIsContainerForFixedPositionLayers(true); | 
 |   outer_viewport_scroll_layer->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |   host()->RegisterViewportLayers(nullptr, root, inner_viewport_scroll_layer, | 
 |                                  outer_viewport_scroll_layer); | 
 |  | 
 |   scoped_refptr<Layer> fixed_to_inner = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> fixed_to_outer = Layer::Create(layer_settings()); | 
 |  | 
 |   inner_viewport_scroll_layer->AddChild(fixed_to_inner); | 
 |   outer_viewport_scroll_layer->AddChild(fixed_to_outer); | 
 |  | 
 |   LayerPositionConstraint fixed_to_right; | 
 |   fixed_to_right.set_is_fixed_position(true); | 
 |   fixed_to_right.set_is_fixed_to_right_edge(true); | 
 |  | 
 |   fixed_to_inner->SetPositionConstraint(fixed_to_right); | 
 |   fixed_to_outer->SetPositionConstraint(fixed_to_right); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   TransformTree& transform_tree = host()->property_trees()->transform_tree; | 
 |   EXPECT_TRUE(transform_tree.HasNodesAffectedByInnerViewportBoundsDelta()); | 
 |   EXPECT_TRUE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta()); | 
 |  | 
 |   LayerPositionConstraint fixed_to_left; | 
 |   fixed_to_left.set_is_fixed_position(true); | 
 |   fixed_to_inner->SetPositionConstraint(fixed_to_left); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_FALSE(transform_tree.HasNodesAffectedByInnerViewportBoundsDelta()); | 
 |   EXPECT_TRUE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta()); | 
 |  | 
 |   fixed_to_outer->SetPositionConstraint(fixed_to_left); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_FALSE(transform_tree.HasNodesAffectedByInnerViewportBoundsDelta()); | 
 |   EXPECT_FALSE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleContentRectForAnimatedLayer) { | 
 |   const gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> animated = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(animated); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(animated.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |   root->SetForceRenderSurface(true); | 
 |   animated->SetOpacity(0.f); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddOpacityTransitionToLayerWithPlayer(animated->id(), timeline(), 10.0, 0.f, | 
 |                                           1.f, false); | 
 |   } else { | 
 |     AddOpacityTransitionToController(animated->layer_animation_controller(), | 
 |                                      10.0, 0.f, 1.f, false); | 
 |   } | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   EXPECT_FALSE(animated->visible_rect_from_property_trees().IsEmpty()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        VisibleContentRectForAnimatedLayerWithSingularTransform) { | 
 |   const gfx::Transform identity_matrix; | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> clip = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> animated = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> surface = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> descendant_of_animation = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(clip); | 
 |   clip->AddChild(animated); | 
 |   animated->AddChild(surface); | 
 |   surface->AddChild(descendant_of_animation); | 
 |  | 
 |   clip->SetMasksToBounds(true); | 
 |   surface->SetForceRenderSurface(true); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform uninvertible_matrix; | 
 |   uninvertible_matrix.Scale3d(6.f, 6.f, 0.f); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(clip.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(animated.get(), uninvertible_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(120, 120), true, false); | 
 |   SetLayerPropertiesForTesting(surface.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(descendant_of_animation.get(), identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(200, 200), true, false); | 
 |  | 
 |   TransformOperations start_transform_operations; | 
 |   start_transform_operations.AppendMatrix(uninvertible_matrix); | 
 |   TransformOperations end_transform_operations; | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedTransformToLayerWithPlayer(animated->id(), timeline(), 10.0, | 
 |                                           start_transform_operations, | 
 |                                           end_transform_operations); | 
 |   } else { | 
 |     AddAnimatedTransformToLayer(animated.get(), 10.0, | 
 |                                 start_transform_operations, | 
 |                                 end_transform_operations); | 
 |   } | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   // The animated layer has a singular transform and maps to a non-empty rect in | 
 |   // clipped target space, so is treated as fully visible. | 
 |   EXPECT_EQ(gfx::Rect(120, 120), animated->visible_rect_from_property_trees()); | 
 |  | 
 |   // The singular transform on |animated| is flattened when inherited by | 
 |   // |surface|, and this happens to make it invertible. | 
 |   EXPECT_EQ(gfx::Rect(2, 2), surface->visible_rect_from_property_trees()); | 
 |   EXPECT_EQ(gfx::Rect(2, 2), | 
 |             descendant_of_animation->visible_rect_from_property_trees()); | 
 |  | 
 |   gfx::Transform zero_matrix; | 
 |   zero_matrix.Scale3d(0.f, 0.f, 0.f); | 
 |   SetLayerPropertiesForTesting(animated.get(), zero_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(120, 120), true, false); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   // The animated layer maps to the empty rect in clipped target space, so is | 
 |   // treated as having an empty visible rect. | 
 |   EXPECT_EQ(gfx::Rect(), animated->visible_rect_from_property_trees()); | 
 |  | 
 |   // This time, flattening does not make |animated|'s transform invertible. This | 
 |   // means the clip cannot be projected into |surface|'s space, so we treat | 
 |   // |surface| and layers that draw into it as having empty visible rect. | 
 |   EXPECT_EQ(gfx::Rect(), surface->visible_rect_from_property_trees()); | 
 |   EXPECT_EQ(gfx::Rect(), | 
 |             descendant_of_animation->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | // Verify that having an animated filter (but no current filter, as these | 
 | // are mutually exclusive) correctly creates a render surface. | 
 | TEST_F(LayerTreeHostCommonTest, AnimatedFilterCreatesRenderSurface) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grandchild = Layer::Create(layer_settings()); | 
 |   root->AddChild(child); | 
 |   child->AddChild(grandchild); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false); | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimatedFilterToLayerWithPlayer(child->id(), timeline(), 10.0, 0.1f, | 
 |                                        0.2f); | 
 |   } else { | 
 |     AddAnimatedFilterToLayer(child.get(), 10.0, 0.1f, 0.2f); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_TRUE(child->has_render_surface()); | 
 |   EXPECT_FALSE(grandchild->has_render_surface()); | 
 |  | 
 |   EXPECT_TRUE(root->filters().IsEmpty()); | 
 |   EXPECT_TRUE(child->filters().IsEmpty()); | 
 |   EXPECT_TRUE(grandchild->filters().IsEmpty()); | 
 |  | 
 |   EXPECT_FALSE(root->FilterIsAnimating()); | 
 |   EXPECT_TRUE(child->FilterIsAnimating()); | 
 |   EXPECT_FALSE(grandchild->FilterIsAnimating()); | 
 | } | 
 |  | 
 | // Verify that having a filter animation with a delayed start time creates a | 
 | // render surface. | 
 | TEST_F(LayerTreeHostCommonTest, DelayedFilterAnimationCreatesRenderSurface) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grandchild = Layer::Create(layer_settings()); | 
 |   root->AddChild(child); | 
 |   child->AddChild(grandchild); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false); | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   scoped_ptr<KeyframedFilterAnimationCurve> curve( | 
 |       KeyframedFilterAnimationCurve::Create()); | 
 |   FilterOperations start_filters; | 
 |   start_filters.Append(FilterOperation::CreateBrightnessFilter(0.1f)); | 
 |   FilterOperations end_filters; | 
 |   end_filters.Append(FilterOperation::CreateBrightnessFilter(0.3f)); | 
 |   curve->AddKeyframe( | 
 |       FilterKeyframe::Create(base::TimeDelta(), start_filters, nullptr)); | 
 |   curve->AddKeyframe(FilterKeyframe::Create( | 
 |       base::TimeDelta::FromMilliseconds(100), end_filters, nullptr)); | 
 |   scoped_ptr<Animation> animation = | 
 |       Animation::Create(std::move(curve), 0, 1, Animation::FILTER); | 
 |   animation->set_fill_mode(Animation::FILL_MODE_NONE); | 
 |   animation->set_time_offset(base::TimeDelta::FromMilliseconds(-1000)); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimationToLayerWithPlayer(child->id(), timeline(), | 
 |                                   std::move(animation)); | 
 |   } else { | 
 |     child->layer_animation_controller()->AddAnimation(std::move(animation)); | 
 |   } | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 |  | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_TRUE(child->has_render_surface()); | 
 |   EXPECT_FALSE(grandchild->has_render_surface()); | 
 |  | 
 |   EXPECT_TRUE(root->filters().IsEmpty()); | 
 |   EXPECT_TRUE(child->filters().IsEmpty()); | 
 |   EXPECT_TRUE(grandchild->filters().IsEmpty()); | 
 |  | 
 |   EXPECT_FALSE(root->FilterIsAnimating()); | 
 |   EXPECT_FALSE(root->HasPotentiallyRunningFilterAnimation()); | 
 |   EXPECT_FALSE(child->FilterIsAnimating()); | 
 |   EXPECT_TRUE(child->HasPotentiallyRunningFilterAnimation()); | 
 |   EXPECT_FALSE(grandchild->FilterIsAnimating()); | 
 |   EXPECT_FALSE(grandchild->HasPotentiallyRunningFilterAnimation()); | 
 | } | 
 |  | 
 | // Ensures that the property tree code accounts for offsets between fixed | 
 | // position layers and their respective containers. | 
 | TEST_F(LayerTreeHostCommonTest, PropertyTreesAccountForFixedParentOffset) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grandchild = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(child); | 
 |   child->AddChild(grandchild); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1000, 1000), gfx::Size(50, 50), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(-1000, -1000), | 
 |                                gfx::Size(50, 50), true, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   grandchild->SetPositionConstraint(constraint); | 
 |  | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), | 
 |             grandchild->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | // Ensures that the property tree code accounts for offsets between fixed | 
 | // position containers and their transform tree parents, when a fixed position | 
 | // layer's container is its layer tree parent, but this parent doesn't have its | 
 | // own transform tree node. | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        PropertyTreesAccountForFixedParentOffsetWhenContainerIsParent) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grandchild = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(child); | 
 |   child->AddChild(grandchild); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(1000, 1000), gfx::Size(50, 50), true, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(-1000, -1000), | 
 |                                gfx::Size(50, 50), true, false); | 
 |  | 
 |   root->SetMasksToBounds(true); | 
 |   child->SetIsContainerForFixedPositionLayers(true); | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   grandchild->SetPositionConstraint(constraint); | 
 |  | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(0, 0, 50, 50), | 
 |             grandchild->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, CombineClipsUsingContentTarget) { | 
 |   // In the following layer tree, the layer |box|'s render target is |surface|. | 
 |   // |surface| also creates a transform node. We want to combine clips for |box| | 
 |   // in the space of its target (i.e., |surface|), not its target's target. This | 
 |   // test ensures that happens. | 
 |  | 
 |   gfx::Transform rotate; | 
 |   rotate.Rotate(5); | 
 |   gfx::Transform identity; | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(2500, 1500), true, | 
 |                                false); | 
 |  | 
 |   scoped_refptr<Layer> frame_clip = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(frame_clip.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(2500, 1500), true, | 
 |                                false); | 
 |   frame_clip->SetMasksToBounds(true); | 
 |  | 
 |   scoped_refptr<Layer> rotated = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(rotated.get(), rotate, | 
 |                                gfx::Point3F(1250, 250, 0), gfx::PointF(), | 
 |                                gfx::Size(2500, 500), true, false); | 
 |  | 
 |   scoped_refptr<Layer> surface = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(surface.get(), rotate, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(2500, 500), true, | 
 |                                false); | 
 |   surface->SetOpacity(0.5); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> container = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(container.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(300, 300), true, false); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> box = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(box.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |  | 
 |   root->AddChild(frame_clip); | 
 |   frame_clip->AddChild(rotated); | 
 |   rotated->AddChild(surface); | 
 |   surface->AddChild(container); | 
 |   surface->AddChild(box); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root.get()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, OnlyApplyFixedPositioningOnce) { | 
 |   gfx::Transform identity; | 
 |   gfx::Transform translate_z; | 
 |   translate_z.Translate3d(0, 0, 10); | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(800, 800), true, false); | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   scoped_refptr<Layer> frame_clip = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(frame_clip.get(), translate_z, gfx::Point3F(), | 
 |                                gfx::PointF(500, 100), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   frame_clip->SetMasksToBounds(true); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> fixed = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(fixed.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1000, 1000), true, | 
 |                                false); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |  | 
 |   root->AddChild(frame_clip); | 
 |   frame_clip->AddChild(fixed); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   gfx::Rect expected(0, 0, 100, 100); | 
 |   EXPECT_EQ(expected, fixed->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        PropertyTreesAccountForScrollCompensationAdjustment) { | 
 |   gfx::Transform identity; | 
 |   gfx::Transform translate_z; | 
 |   translate_z.Translate3d(0, 0, 10); | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(800, 800), true, false); | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   scoped_refptr<Layer> frame_clip = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(frame_clip.get(), translate_z, gfx::Point3F(), | 
 |                                gfx::PointF(500, 100), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   frame_clip->SetMasksToBounds(true); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> scroller = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(scroller.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1000, 1000), true, | 
 |                                false); | 
 |  | 
 |   scroller->SetScrollCompensationAdjustment(gfx::Vector2dF(0.3f, 0.7f)); | 
 |   scroller->SetScrollOffset(gfx::ScrollOffset(0.3, 0.7)); | 
 |   scroller->SetScrollClipLayerId(frame_clip->id()); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> fixed = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(fixed.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> fixed_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(fixed_child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |  | 
 |   fixed_child->SetPositionConstraint(constraint); | 
 |  | 
 |   root->AddChild(frame_clip); | 
 |   frame_clip->AddChild(scroller); | 
 |   scroller->AddChild(fixed); | 
 |   fixed->AddChild(fixed_child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   gfx::Rect expected(0, 0, 50, 50); | 
 |   EXPECT_EQ(expected, fixed->visible_rect_from_property_trees()); | 
 |  | 
 |   expected = gfx::Rect(0, 0, 10, 10); | 
 |   EXPECT_EQ(expected, fixed_child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, FixedClipsShouldBeAssociatedWithTheRightNode) { | 
 |   gfx::Transform identity; | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(800, 800), true, false); | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   scoped_refptr<Layer> frame_clip = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(frame_clip.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(500, 100), gfx::Size(100, 100), true, | 
 |                                false); | 
 |   frame_clip->SetMasksToBounds(true); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> scroller = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(scroller.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(1000, 1000), true, | 
 |                                false); | 
 |  | 
 |   scroller->SetScrollOffset(gfx::ScrollOffset(100, 100)); | 
 |   scroller->SetScrollClipLayerId(frame_clip->id()); | 
 |  | 
 |   scoped_refptr<LayerWithForcedDrawsContent> fixed = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(fixed.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(100, 100), gfx::Size(50, 50), true, | 
 |                                false); | 
 |  | 
 |   LayerPositionConstraint constraint; | 
 |   constraint.set_is_fixed_position(true); | 
 |   fixed->SetPositionConstraint(constraint); | 
 |   fixed->SetForceRenderSurface(true); | 
 |   fixed->SetMasksToBounds(true); | 
 |  | 
 |   root->AddChild(frame_clip); | 
 |   frame_clip->AddChild(scroller); | 
 |   scroller->AddChild(fixed); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   gfx::Rect expected(0, 0, 50, 50); | 
 |   EXPECT_EQ(expected, fixed->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ChangingAxisAlignmentTriggersRebuild) { | 
 |   gfx::Transform identity; | 
 |   gfx::Transform translate; | 
 |   gfx::Transform rotate; | 
 |  | 
 |   translate.Translate(10, 10); | 
 |   rotate.Rotate(45); | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(800, 800), true, false); | 
 |   root->SetIsContainerForFixedPositionLayers(true); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_FALSE(host()->property_trees()->needs_rebuild); | 
 |  | 
 |   root->SetTransform(translate); | 
 |   EXPECT_FALSE(host()->property_trees()->needs_rebuild); | 
 |  | 
 |   root->SetTransform(rotate); | 
 |   EXPECT_TRUE(host()->property_trees()->needs_rebuild); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ChangeTransformOrigin) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   root->AddChild(child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform scale_matrix; | 
 |   scale_matrix.Scale(2.f, 2.f); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), scale_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), child->visible_rect_from_property_trees()); | 
 |  | 
 |   child->SetTransformOrigin(gfx::Point3F(10.f, 10.f, 10.f)); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(5, 5, 5, 5), child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, UpdateScrollChildPosition) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> scroll_parent = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> scroll_child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |  | 
 |   root->AddChild(scroll_child); | 
 |   root->AddChild(scroll_parent); | 
 |   scroll_child->SetScrollParent(scroll_parent.get()); | 
 |   scroll_parent->SetScrollClipLayerId(root->id()); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   gfx::Transform identity_transform; | 
 |   gfx::Transform scale; | 
 |   scale.Scale(2.f, 2.f); | 
 |   SetLayerPropertiesForTesting(root.get(), identity_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false); | 
 |   SetLayerPropertiesForTesting(scroll_child.get(), scale, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(40, 40), true, false); | 
 |   SetLayerPropertiesForTesting(scroll_parent.get(), identity_transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(25, 25), | 
 |             scroll_child->visible_rect_from_property_trees()); | 
 |  | 
 |   scroll_child->SetPosition(gfx::PointF(0, -10.f)); | 
 |   scroll_parent->SetScrollOffset(gfx::ScrollOffset(0.f, 10.f)); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 5, 25, 25), | 
 |             scroll_child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | static void CopyOutputCallback(scoped_ptr<CopyOutputResult> result) { | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, NumCopyRequestsInTargetSubtree) { | 
 |   // If the layer has a node in effect_tree, the return value of | 
 |   // num_copy_requests_in_target_subtree()  must be equal to the actual number | 
 |   // of copy requests in the sub-layer_tree; Otherwise, the number is expected | 
 |   // to be the value of its nearest ancestor that owns an effect node and | 
 |   // greater than or equal to the actual number of copy requests in the | 
 |   // sub-layer_tree. | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child1 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child2 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grandchild = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> greatgrandchild = Layer::Create(layer_settings()); | 
 |  | 
 |   root->AddChild(child1); | 
 |   root->AddChild(child2); | 
 |   child1->AddChild(grandchild); | 
 |   grandchild->AddChild(greatgrandchild); | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   child1->RequestCopyOfOutput( | 
 |       CopyOutputRequest::CreateBitmapRequest(base::Bind(&CopyOutputCallback))); | 
 |   greatgrandchild->RequestCopyOfOutput( | 
 |       CopyOutputRequest::CreateBitmapRequest(base::Bind(&CopyOutputCallback))); | 
 |   child2->SetOpacity(0.f); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |  | 
 |   EXPECT_EQ(root->num_copy_requests_in_target_subtree(), 2); | 
 |   EXPECT_EQ(child1->num_copy_requests_in_target_subtree(), 2); | 
 |   EXPECT_EQ(child2->num_copy_requests_in_target_subtree(), 0); | 
 |   EXPECT_EQ(grandchild->num_copy_requests_in_target_subtree(), 2); | 
 |   EXPECT_EQ(greatgrandchild->num_copy_requests_in_target_subtree(), 1); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SkippingSubtreeMain) { | 
 |   gfx::Transform identity; | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   FakeContentLayerClient client; | 
 |   client.set_bounds(root->bounds()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> grandchild = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   scoped_refptr<FakePictureLayer> greatgrandchild( | 
 |       FakePictureLayer::Create(layer_settings(), &client)); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   SetLayerPropertiesForTesting(greatgrandchild.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |  | 
 |   root->AddChild(child); | 
 |   child->AddChild(grandchild); | 
 |   grandchild->AddChild(greatgrandchild); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   // Check the non-skipped case. | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), grandchild->visible_rect_from_property_trees()); | 
 |  | 
 |   // Now we will reset the visible rect from property trees for the grandchild, | 
 |   // and we will configure |child| in several ways that should force the subtree | 
 |   // to be skipped. The visible content rect for |grandchild| should, therefore, | 
 |   // remain empty. | 
 |   grandchild->set_visible_rect_from_property_trees(gfx::Rect()); | 
 |   gfx::Transform singular; | 
 |   singular.matrix().set(0, 0, 0); | 
 |  | 
 |   child->SetTransform(singular); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), grandchild->visible_rect_from_property_trees()); | 
 |   child->SetTransform(identity); | 
 |  | 
 |   child->SetHideLayerAndSubtree(true); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), grandchild->visible_rect_from_property_trees()); | 
 |   child->SetHideLayerAndSubtree(false); | 
 |  | 
 |   gfx::Transform zero_z_scale; | 
 |   zero_z_scale.Scale3d(1, 1, 0); | 
 |   child->SetTransform(zero_z_scale); | 
 |  | 
 |   // Add a transform animation with a start delay. Now, even though |child| has | 
 |   // a singular transform, the subtree should still get processed. | 
 |   int animation_id = 0; | 
 |   scoped_ptr<Animation> animation = Animation::Create( | 
 |       scoped_ptr<AnimationCurve>(new FakeTransformTransition(1.0)), | 
 |       animation_id, 1, Animation::TRANSFORM); | 
 |   animation->set_fill_mode(Animation::FILL_MODE_NONE); | 
 |   animation->set_time_offset(base::TimeDelta::FromMilliseconds(-1000)); | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimationToLayerWithPlayer(child->id(), timeline(), | 
 |                                   std::move(animation)); | 
 |   } else { | 
 |     child->AddAnimation(std::move(animation)); | 
 |   } | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), grandchild->visible_rect_from_property_trees()); | 
 |   grandchild->set_visible_rect_from_property_trees(gfx::Rect()); | 
 |  | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     RemoveAnimationFromLayerWithExistingPlayer(child->id(), timeline(), | 
 |                                                animation_id); | 
 |   } else { | 
 |     child->RemoveAnimation(animation_id); | 
 |   } | 
 |   child->SetTransform(identity); | 
 |   child->SetOpacity(0.f); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), grandchild->visible_rect_from_property_trees()); | 
 |  | 
 |   // Now, even though child has zero opacity, we will configure |grandchild| and | 
 |   // |greatgrandchild| in several ways that should force the subtree to be | 
 |   // processed anyhow. | 
 |   grandchild->RequestCopyOfOutput( | 
 |       CopyOutputRequest::CreateBitmapRequest(base::Bind(&CopyOutputCallback))); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), grandchild->visible_rect_from_property_trees()); | 
 |   greatgrandchild->set_visible_rect_from_property_trees(gfx::Rect()); | 
 |  | 
 |   // Add an opacity animation with a start delay. | 
 |   animation_id = 1; | 
 |   animation = Animation::Create( | 
 |       scoped_ptr<AnimationCurve>(new FakeFloatTransition(1.0, 0.f, 1.f)), | 
 |       animation_id, 1, Animation::OPACITY); | 
 |   animation->set_fill_mode(Animation::FILL_MODE_NONE); | 
 |   animation->set_time_offset(base::TimeDelta::FromMilliseconds(-1000)); | 
 |   if (layer_settings().use_compositor_animation_timelines) { | 
 |     AddAnimationToLayerWithExistingPlayer(child->id(), timeline(), | 
 |                                           std::move(animation)); | 
 |   } else { | 
 |     child->AddAnimation(std::move(animation)); | 
 |   } | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), grandchild->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SkippingSubtreeImpl) { | 
 |   FakeImplTaskRunnerProvider task_runner_provider; | 
 |   TestSharedBitmapManager shared_bitmap_manager; | 
 |   TestTaskGraphRunner task_graph_runner; | 
 |   FakeLayerTreeHostImpl host_impl(&task_runner_provider, &shared_bitmap_manager, | 
 |                                   &task_graph_runner); | 
 |  | 
 |   gfx::Transform identity; | 
 |   scoped_ptr<LayerImpl> root = LayerImpl::Create(host_impl.active_tree(), 1); | 
 |   scoped_ptr<LayerImpl> child = LayerImpl::Create(host_impl.active_tree(), 2); | 
 |   scoped_ptr<LayerImpl> grandchild = | 
 |       LayerImpl::Create(host_impl.active_tree(), 3); | 
 |  | 
 |   scoped_ptr<FakePictureLayerImpl> greatgrandchild( | 
 |       FakePictureLayerImpl::Create(host_impl.active_tree(), 4)); | 
 |  | 
 |   child->SetDrawsContent(true); | 
 |   grandchild->SetDrawsContent(true); | 
 |   greatgrandchild->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(grandchild.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |  | 
 |   LayerImpl* child_ptr = child.get(); | 
 |   LayerImpl* grandchild_ptr = grandchild.get(); | 
 |  | 
 |   child->AddChild(std::move(grandchild)); | 
 |   root->AddChild(std::move(child)); | 
 |  | 
 |   // Check the non-skipped case. | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), | 
 |             grandchild_ptr->visible_rect_from_property_trees()); | 
 |  | 
 |   // Now we will reset the visible rect from property trees for the grandchild, | 
 |   // and we will configure |child| in several ways that should force the subtree | 
 |   // to be skipped. The visible content rect for |grandchild| should, therefore, | 
 |   // remain empty. | 
 |   grandchild_ptr->set_visible_rect_from_property_trees(gfx::Rect()); | 
 |   gfx::Transform singular; | 
 |   singular.matrix().set(0, 0, 0); | 
 |  | 
 |   child_ptr->SetTransform(singular); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), | 
 |             grandchild_ptr->visible_rect_from_property_trees()); | 
 |   child_ptr->SetTransform(identity); | 
 |  | 
 |   child_ptr->SetHideLayerAndSubtree(true); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), | 
 |             grandchild_ptr->visible_rect_from_property_trees()); | 
 |   child_ptr->SetHideLayerAndSubtree(false); | 
 |  | 
 |   child_ptr->SetOpacity(0.f); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), | 
 |             grandchild_ptr->visible_rect_from_property_trees()); | 
 |  | 
 |   // Now, even though child has zero opacity, we will configure |grandchild| and | 
 |   // |greatgrandchild| in several ways that should force the subtree to be | 
 |   // processed anyhow. | 
 |   std::vector<scoped_ptr<CopyOutputRequest>> requests; | 
 |   requests.push_back(CopyOutputRequest::CreateEmptyRequest()); | 
 |  | 
 |   grandchild_ptr->PassCopyRequests(&requests); | 
 |   root.get()->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), | 
 |             grandchild_ptr->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SkippingLayer) { | 
 |   gfx::Transform identity; | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   FakeContentLayerClient client; | 
 |   client.set_bounds(root->bounds()); | 
 |   scoped_refptr<LayerWithForcedDrawsContent> child = | 
 |       make_scoped_refptr(new LayerWithForcedDrawsContent(layer_settings())); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false); | 
 |   root->AddChild(child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(10, 10), child->visible_rect_from_property_trees()); | 
 |   child->set_visible_rect_from_property_trees(gfx::Rect()); | 
 |  | 
 |   child->SetHideLayerAndSubtree(true); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), child->visible_rect_from_property_trees()); | 
 |   child->SetHideLayerAndSubtree(false); | 
 |  | 
 |   child->SetBounds(gfx::Size()); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), child->visible_rect_from_property_trees()); | 
 |   child->SetBounds(gfx::Size(10, 10)); | 
 |  | 
 |   gfx::Transform rotate; | 
 |   child->SetDoubleSided(false); | 
 |   rotate.RotateAboutXAxis(180.f); | 
 |   child->SetTransform(rotate); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), child->visible_rect_from_property_trees()); | 
 |   child->SetDoubleSided(true); | 
 |   child->SetTransform(identity); | 
 |  | 
 |   child->SetOpacity(0.f); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(gfx::Rect(0, 0), child->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, LayerTreeRebuildTest) { | 
 |   // Ensure that the treewalk in LayerTreeHostCommom:: | 
 |   // PreCalculateMetaInformation happens when its required. | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |  | 
 |   root->AddChild(parent); | 
 |   parent->AddChild(child); | 
 |  | 
 |   child->SetClipParent(root.get()); | 
 |  | 
 |   gfx::Transform identity; | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(parent.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   child->RequestCopyOfOutput( | 
 |       CopyOutputRequest::CreateRequest(base::Bind(&EmptyCopyOutputCallback))); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(parent->num_unclipped_descendants(), 1u); | 
 |  | 
 |   EXPECT_GT(root->num_copy_requests_in_target_subtree(), 0); | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_GT(root->num_copy_requests_in_target_subtree(), 0); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, InputHandlersRecursiveUpdateTest) { | 
 |   // Ensure that the treewalk in LayertreeHostCommon:: | 
 |   // PreCalculateMetaInformation updates input handlers correctly. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(root); | 
 |  | 
 |   gfx::Transform identity; | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(child, identity, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, false); | 
 |  | 
 |   EXPECT_EQ(root->layer_or_descendant_has_input_handler(), false); | 
 |  | 
 |   child->SetHaveWheelEventHandlers(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(root->layer_or_descendant_has_input_handler(), true); | 
 |  | 
 |   child->SetHaveWheelEventHandlers(false); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(root->layer_or_descendant_has_input_handler(), false); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ResetPropertyTreeIndices) { | 
 |   gfx::Transform identity; | 
 |   gfx::Transform translate_z; | 
 |   translate_z.Translate3d(0, 0, 10); | 
 |  | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(800, 800), true, false); | 
 |  | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |   SetLayerPropertiesForTesting(child.get(), translate_z, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |  | 
 |   root->AddChild(child); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_NE(-1, child->transform_tree_index()); | 
 |  | 
 |   child->RemoveFromParent(); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root.get()); | 
 |   EXPECT_EQ(-1, child->transform_tree_index()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ResetLayerDrawPropertiestest) { | 
 |   scoped_refptr<Layer> root = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child = Layer::Create(layer_settings()); | 
 |  | 
 |   root->AddChild(child); | 
 |   gfx::Transform identity; | 
 |  | 
 |   SetLayerPropertiesForTesting(root.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |   SetLayerPropertiesForTesting(child.get(), identity, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false); | 
 |  | 
 |   host()->SetRootLayer(root); | 
 |  | 
 |   EXPECT_FALSE(root->layer_or_descendant_is_drawn()); | 
 |   EXPECT_FALSE(root->visited()); | 
 |   EXPECT_FALSE(root->sorted_for_recursion()); | 
 |   EXPECT_FALSE(child->layer_or_descendant_is_drawn()); | 
 |   EXPECT_FALSE(child->visited()); | 
 |   EXPECT_FALSE(child->sorted_for_recursion()); | 
 |  | 
 |   root->set_layer_or_descendant_is_drawn(true); | 
 |   root->set_visited(true); | 
 |   root->set_sorted_for_recursion(true); | 
 |   child->set_layer_or_descendant_is_drawn(true); | 
 |   child->set_visited(true); | 
 |   child->set_sorted_for_recursion(true); | 
 |  | 
 |   LayerTreeHostCommon::PreCalculateMetaInformationForTesting(root.get()); | 
 |  | 
 |   EXPECT_FALSE(root->layer_or_descendant_is_drawn()); | 
 |   EXPECT_FALSE(root->visited()); | 
 |   EXPECT_FALSE(root->sorted_for_recursion()); | 
 |   EXPECT_FALSE(child->layer_or_descendant_is_drawn()); | 
 |   EXPECT_FALSE(child->visited()); | 
 |   EXPECT_FALSE(child->sorted_for_recursion()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceClipsSubtree) { | 
 |   // Ensure that a Clip Node is added when a render surface applies clip. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* significant_transform = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* layer_clips_subtree = AddChild<LayerImpl>(significant_transform); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(layer_clips_subtree); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   // This transform should be a significant one so that a transform node is | 
 |   // formed for it. | 
 |   gfx::Transform transform1; | 
 |   transform1.RotateAboutYAxis(45); | 
 |   transform1.RotateAboutXAxis(30); | 
 |   // This transform should be a 3d transform as we want the render surface | 
 |   // to flatten the transform | 
 |   gfx::Transform transform2; | 
 |   transform2.Translate3d(10, 10, 10); | 
 |  | 
 |   layer_clips_subtree->SetMasksToBounds(true); | 
 |   test_layer->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false); | 
 |   SetLayerPropertiesForTesting(significant_transform, transform1, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false); | 
 |   SetLayerPropertiesForTesting(layer_clips_subtree, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false); | 
 |   SetLayerPropertiesForTesting(render_surface, transform2, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false); | 
 |   SetLayerPropertiesForTesting(test_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false); | 
 |  | 
 |   root->SetForceRenderSurface(true); | 
 |   significant_transform->SetForceRenderSurface(false); | 
 |   layer_clips_subtree->SetForceRenderSurface(true); | 
 |   render_surface->SetForceRenderSurface(true); | 
 |   test_layer->SetForceRenderSurface(false); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   TransformTree transform_tree = | 
 |       root->layer_tree_impl()->property_trees()->transform_tree; | 
 |   TransformNode* transform_node = | 
 |       transform_tree.Node(significant_transform->transform_tree_index()); | 
 |   EXPECT_EQ(transform_node->owner_id, significant_transform->id()); | 
 |  | 
 |   EXPECT_TRUE(root->has_render_surface()); | 
 |   EXPECT_FALSE(significant_transform->has_render_surface()); | 
 |   EXPECT_TRUE(layer_clips_subtree->has_render_surface()); | 
 |   EXPECT_TRUE(render_surface->has_render_surface()); | 
 |   EXPECT_FALSE(test_layer->has_render_surface()); | 
 |  | 
 |   ClipTree clip_tree = root->layer_tree_impl()->property_trees()->clip_tree; | 
 |   ClipNode* clip_node = clip_tree.Node(render_surface->clip_tree_index()); | 
 |   EXPECT_FALSE(clip_node->data.applies_local_clip); | 
 |   EXPECT_EQ(gfx::Rect(22, 21), test_layer->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TransformOfParentClipNodeAncestorOfTarget) { | 
 |   // Ensure that when parent clip node's transform is an ancestor of current | 
 |   // clip node's target, clip is 'projected' from parent space to current | 
 |   // target space and visible rects are calculated correctly. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_layer = AddChild<LayerImpl>(root); | 
 |   LayerImpl* target_layer = AddChild<LayerImpl>(clip_layer); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(target_layer); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   gfx::Transform transform; | 
 |   transform.RotateAboutYAxis(45); | 
 |   clip_layer->SetMasksToBounds(true); | 
 |   target_layer->SetMasksToBounds(true); | 
 |   test_layer->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_layer, transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(target_layer, transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(test_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(30, 30), test_layer->visible_rect_from_property_trees()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        RenderSurfaceWithUnclippedDescendantsClipsSubtree) { | 
 |   // Ensure clip rect is calculated correctly when render surface has unclipped | 
 |   // descendants. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* between_clip_parent_and_child = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* render_surface = | 
 |       AddChild<LayerImpl>(between_clip_parent_and_child); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   gfx::Transform transform; | 
 |   transform.Translate(2.0, 2.0); | 
 |  | 
 |   test_layer->SetDrawsContent(true); | 
 |   render_surface->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(render_surface); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(between_clip_parent_and_child, transform, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(30, 30), | 
 |                                true, false, false); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(test_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(30, 30), test_layer->clip_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        RenderSurfaceWithUnclippedDescendantsButDoesntApplyOwnClip) { | 
 |   // Ensure that the visible layer rect of a descendant of a surface with | 
 |   // unclipped descendants is computed correctly, when the surface doesn't apply | 
 |   // a clip. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface); | 
 |   LayerImpl* child = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |  | 
 |   clip_child->SetDrawsContent(true); | 
 |   child->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 10), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 15), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(10, 10), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(40, 40), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(40, 40), child->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        RenderSurfaceClipsSubtreeAndHasUnclippedDescendants) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* test_layer1 = AddChild<LayerImpl>(render_surface); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(test_layer1); | 
 |   LayerImpl* test_layer2 = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   root->SetMasksToBounds(true); | 
 |   render_surface->SetMasksToBounds(true); | 
 |   render_surface->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |   test_layer1->SetDrawsContent(true); | 
 |   test_layer2->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(test_layer1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(test_layer2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), render_surface->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), test_layer1->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30), clip_child->visible_layer_rect()); | 
 |   EXPECT_EQ(gfx::Rect(30, 30), test_layer2->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, UnclippedClipParent) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   clip_parent->SetDrawsContent(true); | 
 |   render_surface->SetMasksToBounds(true); | 
 |   render_surface->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |  | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   // The clip child should inherit its clip parent's clipping state, not its | 
 |   // tree parent's clipping state. | 
 |   EXPECT_FALSE(clip_parent->is_clipped()); | 
 |   EXPECT_TRUE(render_surface->is_clipped()); | 
 |   EXPECT_FALSE(clip_child->is_clipped()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, RenderSurfaceContentRectWithMultipleSurfaces) { | 
 |   // Tests the value of render surface content rect when we have multiple types | 
 |   // of surfaces : unclipped surfaces, surfaces with unclipped surfaces and | 
 |   // clipped surfaces. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* unclipped_surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(unclipped_surface); | 
 |   LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* unclipped_desc_surface2 = | 
 |       AddChild<LayerImpl>(unclipped_desc_surface); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface2); | 
 |   LayerImpl* clipped_surface = AddChild<LayerImpl>(clip_child); | 
 |  | 
 |   unclipped_surface->SetDrawsContent(true); | 
 |   unclipped_desc_surface->SetDrawsContent(true); | 
 |   unclipped_desc_surface2->SetDrawsContent(true); | 
 |   clipped_surface->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(80, 80), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(unclipped_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(50, 50), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface2, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), gfx::Size(60, 60), | 
 |                                true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clipped_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(70, 70), true, false, | 
 |                                true); | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   unclipped_surface->SetMasksToBounds(true); | 
 |   unclipped_desc_surface->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), | 
 |             unclipped_surface->render_surface()->content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(100, 100), | 
 |             unclipped_desc_surface->render_surface()->content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), | 
 |             unclipped_desc_surface2->render_surface()->content_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), | 
 |             clipped_surface->render_surface()->content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipBetweenClipChildTargetAndClipParentTarget) { | 
 |   // Tests the value of render surface content rect when we have a layer that | 
 |   // clips between the clip parent's target and clip child's target. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_layer = AddChild<LayerImpl>(surface); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(clip_layer); | 
 |   LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface); | 
 |  | 
 |   clip_child->SetDrawsContent(true); | 
 |   unclipped_desc_surface->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform translate; | 
 |   translate.Translate(10, 10); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface, translate, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   surface->SetMasksToBounds(true); | 
 |   clip_layer->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(10, 10), | 
 |             unclipped_desc_surface->render_surface()->content_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, VisibleRectForDescendantOfScaledSurface) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* surface = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* clip_layer = AddChild<LayerImpl>(surface); | 
 |   LayerImpl* clip_parent = AddChild<LayerImpl>(clip_layer); | 
 |   LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface); | 
 |  | 
 |   clip_child->SetDrawsContent(true); | 
 |   unclipped_desc_surface->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   gfx::Transform identity_matrix; | 
 |   gfx::Transform scale; | 
 |   scale.Scale(2, 2); | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(surface, scale, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   SetLayerPropertiesForTesting(unclipped_desc_surface, identity_matrix, | 
 |                                gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(100, 100), true, false, true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(100, 100), true, false, | 
 |                                false); | 
 |   surface->SetMasksToBounds(true); | 
 |   clip_layer->SetMasksToBounds(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(20, 20), clip_child->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, LayerWithInputHandlerAndZeroOpacity) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(root); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   gfx::Transform translation; | 
 |   translation.Translate(10, 10); | 
 |   SetLayerPropertiesForTesting(test_layer, translation, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(20, 20), true, false, | 
 |                                false); | 
 |  | 
 |   render_surface->SetMasksToBounds(true); | 
 |   test_layer->SetDrawsContent(true); | 
 |   test_layer->SetOpacity(0); | 
 |   test_layer->SetHaveWheelEventHandlers(true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(translation, test_layer->ScreenSpaceTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ClipChildVisibleRect) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* clip_parent = AddChildToRoot<LayerImpl>(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   clip_parent->SetMasksToBounds(true); | 
 |   render_surface->SetMasksToBounds(true); | 
 |   render_surface->SetDrawsContent(true); | 
 |   clip_child->SetDrawsContent(true); | 
 |   clip_child->SetClipParent(clip_parent); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   clip_parent->SetClipChildren(clip_children.release()); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_parent, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10.f, 10.f), gfx::Size(40, 40), true, | 
 |                                false, false); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_EQ(gfx::Rect(40, 40), clip_child->visible_layer_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, | 
 |        LayerClipRectLargerThanClippingRenderSurfaceRect) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface = AddChild<LayerImpl>(root); | 
 |   LayerImpl* test_layer = AddChild<LayerImpl>(render_surface); | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(test_layer, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(50, 50), true, false, | 
 |                                false); | 
 |   root->SetMasksToBounds(true); | 
 |   root->SetDrawsContent(true); | 
 |   render_surface->SetMasksToBounds(true); | 
 |   render_surface->SetDrawsContent(true); | 
 |   test_layer->SetMasksToBounds(true); | 
 |   test_layer->SetDrawsContent(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(30, 30), root->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), render_surface->clip_rect()); | 
 |   EXPECT_EQ(gfx::Rect(50, 50), test_layer->clip_rect()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SubtreeIsHiddenTest) { | 
 |   // Tests that subtree is hidden is updated. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* hidden = AddChild<LayerImpl>(root); | 
 |   LayerImpl* test = AddChild<LayerImpl>(hidden); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(hidden, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10, 10), gfx::Size(30, 30), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(test, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |  | 
 |   hidden->SetHideLayerAndSubtree(true); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRUE(test->IsHidden()); | 
 |  | 
 |   hidden->SetHideLayerAndSubtree(false); | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_FALSE(test->IsHidden()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, TwoUnclippedRenderSurfaces) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(root); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1); | 
 |   LayerImpl* clip_child = AddChild<LayerImpl>(render_surface2); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   clip_child->SetClipParent(root); | 
 |   scoped_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>); | 
 |   clip_children->insert(clip_child); | 
 |   root->SetClipChildren(clip_children.release()); | 
 |   root->SetMasksToBounds(true); | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(10, 10), gfx::Size(30, 30), true, | 
 |                                false, true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(clip_child, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                false); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_EQ(gfx::Rect(-10, -10, 30, 30), render_surface2->clip_rect()); | 
 |   // A clip node is created for every render surface and for layers that have | 
 |   // local clip. So, here it should be craeted for every layer. | 
 |   EXPECT_EQ(root->layer_tree_impl()->property_trees()->clip_tree.size(), 5u); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, MaskLayerScreenSpaceTransform) { | 
 |   // Tests that the mask layer gets its owning layer's screen space transform. | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* child = AddChild<LayerImpl>(root); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   gfx::Transform transform; | 
 |   transform.Translate(10, 10); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(child, transform, gfx::Point3F(), gfx::PointF(), | 
 |                                gfx::Size(30, 30), true, false, false); | 
 |   root->SetDrawsContent(true); | 
 |   child->SetDrawsContent(false); | 
 |   child->SetMaskLayer(LayerImpl::Create(root->layer_tree_impl(), 100)); | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(transform, | 
 |                                   child->mask_layer()->ScreenSpaceTransform()); | 
 |   transform.Translate(10, 10); | 
 |   child->SetTransform(transform); | 
 |   child->SetDrawsContent(true); | 
 |   root->layer_tree_impl()->property_trees()->needs_rebuild = true; | 
 |   ExecuteCalculateDrawProperties(root); | 
 |   EXPECT_TRANSFORMATION_MATRIX_EQ(transform, | 
 |                                   child->mask_layer()->ScreenSpaceTransform()); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, LargeTransformTest) { | 
 |   LayerImpl* root = root_layer(); | 
 |   LayerImpl* render_surface1 = AddChild<LayerImpl>(root); | 
 |   LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1); | 
 |  | 
 |   const gfx::Transform identity_matrix; | 
 |   render_surface1->SetDrawsContent(true); | 
 |   render_surface2->SetDrawsContent(true); | 
 |   render_surface2->SetMasksToBounds(true); | 
 |  | 
 |   gfx::Transform large_transform; | 
 |   large_transform.Scale(99999999999999999999.f, 99999999999999999999.f); | 
 |   large_transform.Scale(9999999999999999999.f, 9999999999999999999.f); | 
 |   EXPECT_TRUE(std::isinf(large_transform.matrix().get(0, 0))); | 
 |   EXPECT_TRUE(std::isinf(large_transform.matrix().get(1, 1))); | 
 |  | 
 |   SetLayerPropertiesForTesting(root, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface1, large_transform, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |   SetLayerPropertiesForTesting(render_surface2, identity_matrix, gfx::Point3F(), | 
 |                                gfx::PointF(), gfx::Size(30, 30), true, false, | 
 |                                true); | 
 |  | 
 |   ExecuteCalculateDrawProperties(root); | 
 |  | 
 |   bool is_inf_or_nan = | 
 |       std::isinf( | 
 |           render_surface2->render_surface()->draw_transform().matrix().get( | 
 |               0, 0)) || | 
 |       std::isnan( | 
 |           render_surface2->render_surface()->draw_transform().matrix().get(0, | 
 |                                                                            0)); | 
 |   EXPECT_TRUE(is_inf_or_nan); | 
 |   is_inf_or_nan = | 
 |       std::isinf( | 
 |           render_surface2->render_surface()->draw_transform().matrix().get( | 
 |               1, 1)) || | 
 |       std::isnan( | 
 |           render_surface2->render_surface()->draw_transform().matrix().get(1, | 
 |                                                                            1)); | 
 |   EXPECT_TRUE(is_inf_or_nan); | 
 |   EXPECT_EQ(gfx::RectF(), | 
 |             render_surface2->render_surface()->DrawableContentRect()); | 
 |  | 
 |   std::vector<LayerImpl*>* rsll = render_surface_layer_list_impl(); | 
 |   bool root_in_rsll = | 
 |       std::find(rsll->begin(), rsll->end(), root) != rsll->end(); | 
 |   EXPECT_TRUE(root_in_rsll); | 
 |   bool render_surface2_in_rsll = | 
 |       std::find(rsll->begin(), rsll->end(), render_surface2) != rsll->end(); | 
 |   EXPECT_FALSE(render_surface2_in_rsll); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SerializeScrollUpdateInfo) { | 
 |   LayerTreeHostCommon::ScrollUpdateInfo scroll; | 
 |   scroll.layer_id = 2; | 
 |   scroll.scroll_delta = gfx::Vector2d(5, 10); | 
 |  | 
 |   proto::ScrollUpdateInfo proto; | 
 |   scroll.ToProtobuf(&proto); | 
 |   LayerTreeHostCommon::ScrollUpdateInfo new_scroll; | 
 |   new_scroll.FromProtobuf(proto); | 
 |  | 
 |   EXPECT_EQ(scroll, new_scroll); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, SerializeScrollAndScale) { | 
 |   ScrollAndScaleSet scroll_and_scale_set; | 
 |  | 
 |   LayerTreeHostCommon::ScrollUpdateInfo scroll1; | 
 |   scroll1.layer_id = 1; | 
 |   scroll1.scroll_delta = gfx::Vector2d(5, 10); | 
 |   LayerTreeHostCommon::ScrollUpdateInfo scroll2; | 
 |   scroll2.layer_id = 2; | 
 |   scroll2.scroll_delta = gfx::Vector2d(1, 5); | 
 |   scroll_and_scale_set.scrolls.push_back(scroll1); | 
 |   scroll_and_scale_set.scrolls.push_back(scroll2); | 
 |  | 
 |   scroll_and_scale_set.page_scale_delta = 0.3f; | 
 |   scroll_and_scale_set.elastic_overscroll_delta = gfx::Vector2dF(0.5f, 0.6f); | 
 |   scroll_and_scale_set.top_controls_delta = 0.9f; | 
 |  | 
 |   proto::ScrollAndScaleSet proto; | 
 |   scroll_and_scale_set.ToProtobuf(&proto); | 
 |   ScrollAndScaleSet new_scroll_and_scale_set; | 
 |   new_scroll_and_scale_set.FromProtobuf(proto); | 
 |  | 
 |   EXPECT_TRUE(scroll_and_scale_set.EqualsForTesting(new_scroll_and_scale_set)); | 
 | } | 
 |  | 
 | TEST_F(LayerTreeHostCommonTest, ScrollTreeBuilderTest) { | 
 |   // Test the behavior of scroll tree builder | 
 |   // Topology: | 
 |   // +root1(1) | 
 |   // +--parent2(2)[should_scroll_on_main_thread & scrollable] | 
 |   // +----child6(6)[should_scroll_on_main_thread] | 
 |   // +------grand_child10(10)[should_scroll_on_main_thread] | 
 |   // +--parent3(3) | 
 |   // +----child7(7)[scrollable] | 
 |   // +----child8(8)[scroll_parent=7] | 
 |   // +------grand_child11(11)[scrollable] | 
 |   // +--parent4(4) | 
 |   // +----child9(9) | 
 |   // +------grand_child12(12) | 
 |   // +--parent5(5)[contains_non_fast_scrollable_region] | 
 |   // | 
 |   // Expected scroll tree topology: | 
 |   // +property_tree_root---owner:-1 | 
 |   // +--root---owner:1, id:1 | 
 |   // +----node---owner:2, id:2 | 
 |   // +------node---owner:6, id:3 | 
 |   // +----node---owner:7, id:4 | 
 |   // +------node---owner:11, id:5 | 
 |   // +----node---owner:5, id:6 | 
 |   // | 
 |   // Extra check: | 
 |   //   scroll_tree_index() of: | 
 |   //     grand_child10:3 | 
 |   //     parent3:1 | 
 |   //     child8:4 | 
 |   //     parent4:1 | 
 |   //     child9:1 | 
 |   //     grand_child12:1 | 
 |   scoped_refptr<Layer> root1 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent2 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent3 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent4 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> parent5 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child6 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child7 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child8 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> child9 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child10 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child11 = Layer::Create(layer_settings()); | 
 |   scoped_refptr<Layer> grand_child12 = Layer::Create(layer_settings()); | 
 |  | 
 |   root1->AddChild(parent2); | 
 |   root1->AddChild(parent3); | 
 |   root1->AddChild(parent4); | 
 |   root1->AddChild(parent5); | 
 |   parent2->AddChild(child6); | 
 |   parent3->AddChild(child7); | 
 |   parent3->AddChild(child8); | 
 |   parent4->AddChild(child9); | 
 |   child6->AddChild(grand_child10); | 
 |   child8->AddChild(grand_child11); | 
 |   child9->AddChild(grand_child12); | 
 |   host()->SetRootLayer(root1); | 
 |  | 
 |   parent2->AddMainThreadScrollingReasons( | 
 |       MainThreadScrollingReason::kHasBackgroundAttachmentFixedObjects); | 
 |   parent2->SetScrollClipLayerId(root1->id()); | 
 |   child6->AddMainThreadScrollingReasons( | 
 |       MainThreadScrollingReason::kHasBackgroundAttachmentFixedObjects); | 
 |   grand_child10->AddMainThreadScrollingReasons( | 
 |       MainThreadScrollingReason::kHasBackgroundAttachmentFixedObjects); | 
 |  | 
 |   child7->SetScrollClipLayerId(root1->id()); | 
 |   child8->SetScrollParent(child7.get()); | 
 |   grand_child11->SetScrollClipLayerId(root1->id()); | 
 |  | 
 |   parent5->SetNonFastScrollableRegion(gfx::Rect(0, 0, 50, 50)); | 
 |  | 
 |   ExecuteCalculateDrawPropertiesWithPropertyTrees(root1.get()); | 
 |  | 
 |   const int kInvalidPropertyTreeNodeId = -1; | 
 |   const int kRootPropertyTreeNodeId = 0; | 
 |  | 
 |   // Property tree root | 
 |   ScrollTree scroll_tree = host()->property_trees()->scroll_tree; | 
 |   ScrollTree expected_scroll_tree; | 
 |   ScrollNode* property_tree_root = expected_scroll_tree.Node(0); | 
 |   property_tree_root->id = kRootPropertyTreeNodeId; | 
 |   property_tree_root->parent_id = kInvalidPropertyTreeNodeId; | 
 |   property_tree_root->owner_id = kInvalidPropertyTreeNodeId; | 
 |   property_tree_root->data.scrollable = false; | 
 |   property_tree_root->data.should_scroll_on_main_thread = false; | 
 |   property_tree_root->data.contains_non_fast_scrollable_region = false; | 
 |   property_tree_root->data.transform_id = kRootPropertyTreeNodeId; | 
 |  | 
 |   // The node owned by root1 | 
 |   ScrollNode scroll_root1; | 
 |   scroll_root1.id = 1; | 
 |   scroll_root1.owner_id = root1->id(); | 
 |   scroll_root1.data.transform_id = root1->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_root1, 0); | 
 |  | 
 |   // The node owned by parent2 | 
 |   ScrollNode scroll_parent2; | 
 |   scroll_parent2.id = 2; | 
 |   scroll_parent2.owner_id = parent2->id(); | 
 |   scroll_parent2.data.scrollable = true; | 
 |   scroll_parent2.data.should_scroll_on_main_thread = true; | 
 |   scroll_parent2.data.transform_id = parent2->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_parent2, 1); | 
 |  | 
 |   // The node owned by child6 | 
 |   ScrollNode scroll_child6; | 
 |   scroll_child6.id = 3; | 
 |   scroll_child6.owner_id = child6->id(); | 
 |   scroll_child6.data.should_scroll_on_main_thread = true; | 
 |   scroll_child6.data.transform_id = child6->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_child6, 2); | 
 |  | 
 |   // The node owned by child7, child7 also owns a transform node | 
 |   ScrollNode scroll_child7; | 
 |   scroll_child7.id = 4; | 
 |   scroll_child7.owner_id = child7->id(); | 
 |   scroll_child7.data.scrollable = true; | 
 |   scroll_child7.data.transform_id = child7->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_child7, 1); | 
 |  | 
 |   // The node owned by grand_child11, grand_child11 also owns a transform node | 
 |   ScrollNode scroll_grand_child11; | 
 |   scroll_grand_child11.id = 5; | 
 |   scroll_grand_child11.owner_id = grand_child11->id(); | 
 |   scroll_grand_child11.data.scrollable = true; | 
 |   scroll_grand_child11.data.transform_id = | 
 |       grand_child11->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_grand_child11, 4); | 
 |  | 
 |   // The node owned by parent5 | 
 |   ScrollNode scroll_parent5; | 
 |   scroll_parent5.id = 8; | 
 |   scroll_parent5.owner_id = parent5->id(); | 
 |   scroll_parent5.data.contains_non_fast_scrollable_region = true; | 
 |   scroll_parent5.data.transform_id = parent5->transform_tree_index(); | 
 |   expected_scroll_tree.Insert(scroll_parent5, 1); | 
 |  | 
 |   expected_scroll_tree.set_needs_update(false); | 
 |  | 
 |   EXPECT_EQ(expected_scroll_tree, scroll_tree); | 
 |  | 
 |   // Check other layers' scroll_tree_index | 
 |   EXPECT_EQ(scroll_child6.id, grand_child10->scroll_tree_index()); | 
 |   EXPECT_EQ(scroll_root1.id, parent3->scroll_tree_index()); | 
 |   EXPECT_EQ(scroll_child7.id, child8->scroll_tree_index()); | 
 |   EXPECT_EQ(scroll_root1.id, parent4->scroll_tree_index()); | 
 |   EXPECT_EQ(scroll_root1.id, child9->scroll_tree_index()); | 
 |   EXPECT_EQ(scroll_root1.id, grand_child12->scroll_tree_index()); | 
 | } | 
 |  | 
 | }  // namespace | 
 | }  // namespace cc |