blob: e13870d128b4c42392d4ecaaf0a9ba3f8fb104b9 [file] [log] [blame]
// 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 <memory>
#include <set>
#include <tuple>
#include <vector>
#include "base/memory/ptr_util.h"
#include "base/stl_util.h"
#include "cc/animation/animation_host.h"
#include "cc/animation/animation_id_provider.h"
#include "cc/animation/keyframed_animation_curve.h"
#include "cc/animation/single_keyframe_effect_animation.h"
#include "cc/animation/transform_operations.h"
#include "cc/base/math_util.h"
#include "cc/input/main_thread_scrolling_reason.h"
#include "cc/layers/content_layer_client.h"
#include "cc/layers/effect_tree_layer_list_iterator.h"
#include "cc/layers/layer.h"
#include "cc/layers/layer_client.h"
#include "cc/layers/layer_impl.h"
#include "cc/layers/render_surface_impl.h"
#include "cc/layers/texture_layer_impl.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_frame_sink.h"
#include "cc/test/fake_layer_tree_host.h"
#include "cc/test/fake_layer_tree_host_impl.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_test_common.h"
#include "cc/test/test_task_graph_runner.h"
#include "cc/trees/clip_node.h"
#include "cc/trees/draw_property_utils.h"
#include "cc/trees/effect_node.h"
#include "cc/trees/property_tree_builder.h"
#include "cc/trees/scroll_node.h"
#include "cc/trees/single_thread_proxy.h"
#include "cc/trees/task_runner_provider.h"
#include "cc/trees/transform_node.h"
#include "components/viz/common/frame_sinks/copy_output_request.h"
#include "components/viz/common/frame_sinks/copy_output_result.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/quad_f.h"
#include "ui/gfx/geometry/rect_conversions.h"
#include "ui/gfx/geometry/size_conversions.h"
#include "ui/gfx/geometry/vector2d_conversions.h"
#include "ui/gfx/transform.h"
namespace cc {
namespace {
bool LayerSubtreeHasCopyRequest(Layer* layer) {
LayerTreeHost* host = layer->layer_tree_host();
int index = layer->effect_tree_index();
auto* node = host->property_trees()->effect_tree.Node(index);
return node->subtree_has_copy_request;
}
class VerifyTreeCalcsLayerTreeSettings : public LayerTreeSettings {
public:
VerifyTreeCalcsLayerTreeSettings() = default;
};
class LayerTreeHostCommonTestBase : public LayerTestCommon::LayerImplTest {
public:
LayerTreeHostCommonTestBase()
: LayerTestCommon::LayerImplTest(VerifyTreeCalcsLayerTreeSettings()) {}
explicit LayerTreeHostCommonTestBase(const LayerTreeSettings& settings)
: LayerTestCommon::LayerImplTest(settings) {}
static void SetScrollOffsetDelta(LayerImpl* layer_impl,
const gfx::Vector2dF& delta) {
if (layer_impl->layer_tree_impl()
->property_trees()
->scroll_tree.SetScrollOffsetDeltaForTesting(
layer_impl->element_id(), delta))
layer_impl->layer_tree_impl()->DidUpdateScrollOffset(
layer_impl->element_id());
}
static float GetMaximumAnimationScale(LayerImpl* layer_impl) {
return layer_impl->layer_tree_impl()
->property_trees()
->GetAnimationScales(layer_impl->transform_tree_index(),
layer_impl->layer_tree_impl())
.maximum_animation_scale;
}
static float GetStartingAnimationScale(LayerImpl* layer_impl) {
return layer_impl->layer_tree_impl()
->property_trees()
->GetAnimationScales(layer_impl->transform_tree_index(),
layer_impl->layer_tree_impl())
.starting_animation_scale;
}
void ExecuteCalculateDrawProperties(Layer* root_layer,
float device_scale_factor,
float page_scale_factor,
Layer* page_scale_layer,
Layer* inner_viewport_scroll_layer,
Layer* outer_viewport_scroll_layer) {
EXPECT_TRUE(page_scale_layer || (page_scale_factor == 1.f));
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_host()->SetViewportSizeAndScale(
device_viewport_size, device_scale_factor,
viz::LocalSurfaceIdAllocation());
// We are probably not testing what is intended if the root_layer bounds are
// empty.
DCHECK(!root_layer->bounds().IsEmpty());
LayerTreeHostCommon::CalcDrawPropsMainInputsForTesting inputs(
root_layer, device_viewport_size);
inputs.device_scale_factor = device_scale_factor;
inputs.page_scale_factor = page_scale_factor;
inputs.page_scale_layer = page_scale_layer;
inputs.inner_viewport_scroll_layer = inner_viewport_scroll_layer;
inputs.outer_viewport_scroll_layer = outer_viewport_scroll_layer;
if (page_scale_layer) {
PropertyTrees* property_trees =
root_layer->layer_tree_host()->property_trees();
inputs.page_scale_transform_node = property_trees->transform_tree.Node(
page_scale_layer->transform_tree_index());
}
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
}
void ExecuteCalculateDrawProperties(LayerImpl* root_layer,
float device_scale_factor,
float page_scale_factor,
LayerImpl* page_scale_layer,
LayerImpl* inner_viewport_scroll_layer,
LayerImpl* outer_viewport_scroll_layer) {
if (device_scale_factor !=
root_layer->layer_tree_impl()->device_scale_factor())
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
root_layer->layer_tree_impl()->SetDeviceScaleFactor(device_scale_factor);
EXPECT_TRUE(page_scale_layer || (page_scale_factor == 1.f));
gfx::Size device_viewport_size =
gfx::Size(root_layer->bounds().width() * device_scale_factor,
root_layer->bounds().height() * device_scale_factor);
render_surface_list_impl_.reset(new RenderSurfaceList);
// We are probably not testing what is intended if the root_layer bounds are
// empty.
DCHECK(!root_layer->bounds().IsEmpty());
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, device_viewport_size, render_surface_list_impl_.get());
inputs.device_scale_factor = device_scale_factor;
inputs.page_scale_factor = page_scale_factor;
inputs.page_scale_layer = page_scale_layer;
inputs.inner_viewport_scroll_layer = inner_viewport_scroll_layer;
inputs.outer_viewport_scroll_layer = outer_viewport_scroll_layer;
inputs.can_adjust_raster_scales = true;
if (page_scale_layer) {
PropertyTrees* property_trees =
root_layer->layer_tree_impl()->property_trees();
inputs.page_scale_transform_node = property_trees->transform_tree.Node(
page_scale_layer->transform_tree_index());
}
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
}
template <class LayerType>
void ExecuteCalculateDrawProperties(LayerType* root_layer) {
LayerType* page_scale_application_layer = nullptr;
LayerType* inner_viewport_scroll_layer = nullptr;
LayerType* outer_viewport_scroll_layer = nullptr;
ExecuteCalculateDrawProperties(
root_layer, 1.f, 1.f, page_scale_application_layer,
inner_viewport_scroll_layer, outer_viewport_scroll_layer);
}
template <class LayerType>
void ExecuteCalculateDrawProperties(LayerType* root_layer,
float device_scale_factor) {
LayerType* page_scale_application_layer = nullptr;
LayerType* inner_viewport_scroll_layer = nullptr;
LayerType* outer_viewport_scroll_layer = nullptr;
ExecuteCalculateDrawProperties(
root_layer, device_scale_factor, 1.f, page_scale_application_layer,
inner_viewport_scroll_layer, outer_viewport_scroll_layer);
}
const LayerList* GetUpdateLayerList() { return &update_layer_list_; }
void ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(Layer* root_layer) {
DCHECK(root_layer->layer_tree_host());
const Layer* page_scale_layer =
root_layer->layer_tree_host()->page_scale_layer();
Layer* inner_viewport_scroll_layer =
root_layer->layer_tree_host()->inner_viewport_scroll_layer();
Layer* outer_viewport_scroll_layer =
root_layer->layer_tree_host()->outer_viewport_scroll_layer();
const ElementId overscroll_elasticity_element_id =
root_layer->layer_tree_host()->overscroll_elasticity_element_id();
gfx::Vector2dF elastic_overscroll =
root_layer->layer_tree_host()->elastic_overscroll();
float page_scale_factor = 1.f;
float device_scale_factor = 1.f;
gfx::Size device_viewport_size =
gfx::Size(root_layer->bounds().width() * device_scale_factor,
root_layer->bounds().height() * device_scale_factor);
PropertyTrees* property_trees =
root_layer->layer_tree_host()->property_trees();
update_layer_list_.clear();
PropertyTreeBuilder::BuildPropertyTrees(
root_layer, page_scale_layer, inner_viewport_scroll_layer,
outer_viewport_scroll_layer, overscroll_elasticity_element_id,
elastic_overscroll, page_scale_factor, device_scale_factor,
gfx::Rect(device_viewport_size), gfx::Transform(), property_trees);
draw_property_utils::UpdatePropertyTrees(root_layer->layer_tree_host(),
property_trees);
draw_property_utils::FindLayersThatNeedUpdates(
root_layer->layer_tree_host(), property_trees, &update_layer_list_);
}
void ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(
LayerImpl* root_layer) {
DCHECK(root_layer->layer_tree_impl());
bool can_adjust_raster_scales = true;
const LayerImpl* page_scale_layer = nullptr;
LayerImpl* inner_viewport_scroll_layer =
root_layer->layer_tree_impl()->InnerViewportScrollLayer();
LayerImpl* outer_viewport_scroll_layer =
root_layer->layer_tree_impl()->OuterViewportScrollLayer();
const ElementId overscroll_elasticity_element_id =
root_layer->layer_tree_impl()->OverscrollElasticityElementId();
gfx::Vector2dF elastic_overscroll =
root_layer->layer_tree_impl()->elastic_overscroll()->Current(
root_layer->layer_tree_impl()->IsActiveTree());
float page_scale_factor = 1.f;
float device_scale_factor = 1.f;
gfx::Size device_viewport_size =
gfx::Size(root_layer->bounds().width() * device_scale_factor,
root_layer->bounds().height() * device_scale_factor);
update_layer_impl_list_.reset(new LayerImplList);
root_layer->layer_tree_impl()->BuildLayerListForTesting();
PropertyTrees* property_trees =
root_layer->layer_tree_impl()->property_trees();
PropertyTreeBuilder::BuildPropertyTrees(
root_layer, page_scale_layer, inner_viewport_scroll_layer,
outer_viewport_scroll_layer, overscroll_elasticity_element_id,
elastic_overscroll, page_scale_factor, device_scale_factor,
gfx::Rect(device_viewport_size), gfx::Transform(), property_trees);
draw_property_utils::UpdatePropertyTreesAndRenderSurfaces(
root_layer, property_trees, can_adjust_raster_scales);
draw_property_utils::FindLayersThatNeedUpdates(
root_layer->layer_tree_impl(), property_trees,
update_layer_impl_list_.get());
draw_property_utils::ComputeDrawPropertiesOfVisibleLayers(
update_layer_impl_list(), property_trees);
}
void ExecuteCalculateDrawPropertiesWithoutAdjustingRasterScales(
LayerImpl* root_layer) {
gfx::Size device_viewport_size =
gfx::Size(root_layer->bounds().width(), root_layer->bounds().height());
render_surface_list_impl_.reset(new RenderSurfaceList);
DCHECK(!root_layer->bounds().IsEmpty());
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, device_viewport_size, render_surface_list_impl_.get());
inputs.can_adjust_raster_scales = false;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
}
bool UpdateLayerImplListContains(int id) const {
for (auto* layer : *update_layer_impl_list_) {
if (layer->id() == id)
return true;
}
return false;
}
bool UpdateLayerListContains(int id) const {
for (const auto& layer : update_layer_list_) {
if (layer->id() == id)
return true;
}
return false;
}
const RenderSurfaceList* render_surface_list_impl() const {
return render_surface_list_impl_.get();
}
const LayerImplList* update_layer_impl_list() const {
return update_layer_impl_list_.get();
}
const LayerList& update_layer_list() const { return update_layer_list_; }
bool VerifyLayerInList(scoped_refptr<Layer> layer,
const LayerList* layer_list) {
return base::ContainsValue(*layer_list, layer);
}
private:
std::unique_ptr<RenderSurfaceList> render_surface_list_impl_;
LayerList update_layer_list_;
std::unique_ptr<LayerImplList> update_layer_impl_list_;
};
class LayerTreeHostCommonTest : public LayerTreeHostCommonTestBase,
public testing::Test {};
class LayerTreeSettingsScaleContent : public VerifyTreeCalcsLayerTreeSettings {
public:
LayerTreeSettingsScaleContent() {
layer_transforms_should_scale_layer_contents = true;
}
};
class LayerTreeHostCommonScalingTest : public LayerTreeHostCommonTestBase,
public testing::Test {
public:
LayerTreeHostCommonScalingTest()
: LayerTreeHostCommonTestBase(LayerTreeSettingsScaleContent()) {}
};
class LayerTreeHostCommonDrawRectsTest : public LayerTreeHostCommonTest {
public:
LayerTreeHostCommonDrawRectsTest() : LayerTreeHostCommonTest() {}
LayerImpl* TestVisibleRectAndDrawableContentRect(
const gfx::Rect& target_rect,
const gfx::Transform& layer_transform,
const gfx::Rect& layer_rect) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* target = AddChild<LayerImpl>(root);
LayerImpl* drawing_layer = AddChild<LayerImpl>(target);
root->SetDrawsContent(true);
target->SetDrawsContent(true);
target->SetMasksToBounds(true);
drawing_layer->SetDrawsContent(true);
root->SetBounds(gfx::Size(500, 500));
root->test_properties()->force_render_surface = true;
target->test_properties()->position = gfx::PointF(target_rect.origin());
target->SetBounds(target_rect.size());
target->test_properties()->force_render_surface = true;
drawing_layer->test_properties()->transform = layer_transform;
drawing_layer->test_properties()->position =
gfx::PointF(layer_rect.origin());
drawing_layer->SetBounds(layer_rect.size());
drawing_layer->test_properties()->should_flatten_transform = false;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
return drawing_layer;
}
};
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.
LayerImpl* parent = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(parent);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
parent->SetBounds(gfx::Size(100, 100));
ExecuteCalculateDrawProperties(parent);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(), child->DrawTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
child->ScreenSpaceTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
grand_child->DrawTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
grand_child->ScreenSpaceTransform());
}
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_for_testing();
LayerImpl* child = AddChild<LayerImpl>(parent);
child->SetDrawsContent(true);
parent->SetBounds(gfx::Size(100, 100));
child->test_properties()->position = gfx::PointF(10, 10);
child->SetBounds(gfx::Size(100, 100));
child->test_properties()->opacity = 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->transform_id, transform_tree_size);
}
TEST_F(LayerTreeHostCommonTest, TransformsForSingleLayer) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* layer = AddChild<LayerImpl>(root);
TransformTree& transform_tree =
host_impl()->active_tree()->property_trees()->transform_tree;
EffectTree& effect_tree =
host_impl()->active_tree()->property_trees()->effect_tree;
root->SetBounds(gfx::Size(1, 2));
// Case 1: Setting the bounds of the layer should not affect either the draw
// transform or the screenspace transform.
layer->SetBounds(gfx::Size(10, 12));
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::ScreenSpaceTransform(layer, transform_tree));
// Case 2: The anchor point by itself (without a layer transform) should have
// no effect on the transforms.
layer->test_properties()->transform_origin = gfx::Point3F(2.5f, 3.0f, 0.f);
layer->SetBounds(gfx::Size(10, 12));
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::ScreenSpaceTransform(layer, transform_tree));
// Case 3: 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);
layer->test_properties()->position = gfx::PointF(0.f, 1.2f);
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
position_transform,
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
position_transform,
draw_property_utils::ScreenSpaceTransform(layer, transform_tree));
// Case 4: 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);
layer->test_properties()->transform = layer_transform;
layer->test_properties()->transform_origin = gfx::Point3F();
layer->test_properties()->position = gfx::PointF();
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
layer_transform,
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
layer_transform,
draw_property_utils::ScreenSpaceTransform(layer, transform_tree));
// Case 5: 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);
layer->test_properties()->transform_origin = gfx::Point3F(5.f, 0.f, 0.f);
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_result,
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_result,
draw_property_utils::ScreenSpaceTransform(layer, transform_tree));
// Case 6: 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);
layer->test_properties()->position = gfx::PointF(0.f, 1.2f);
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_result,
draw_property_utils::DrawTransform(layer, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_result,
draw_property_utils::ScreenSpaceTransform(layer, transform_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());
float page_scale = 0.888f;
const float kDeviceScale = 1.666f;
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> sublayer_scoped_ptr(
LayerImpl::Create(host_impl.active_tree(), 1));
LayerImpl* sublayer = sublayer_scoped_ptr.get();
sublayer->SetDrawsContent(true);
sublayer->SetBounds(gfx::Size(500, 500));
std::unique_ptr<LayerImpl> scroll_layer_scoped_ptr(
LayerImpl::Create(host_impl.active_tree(), 2));
LayerImpl* scroll_layer = scroll_layer_scoped_ptr.get();
scroll_layer->SetBounds(gfx::Size(10, 20));
scroll_layer->SetElementId(LayerIdToElementIdForTesting(scroll_layer->id()));
scroll_layer->SetScrollable(
gfx::Size(scroll_layer->bounds().width() + kMaxScrollOffset.x(),
scroll_layer->bounds().height() + kMaxScrollOffset.y()));
scroll_layer->test_properties()->AddChild(std::move(sublayer_scoped_ptr));
std::unique_ptr<LayerImpl> root(
LayerImpl::Create(host_impl.active_tree(), 3));
root->SetBounds(gfx::Size(3, 4));
root->test_properties()->AddChild(std::move(scroll_layer_scoped_ptr));
LayerImpl* root_layer = root.get();
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.active_tree()->BuildPropertyTreesForTesting();
auto& scroll_tree = host_impl.active_tree()->property_trees()->scroll_tree;
scroll_tree.UpdateScrollOffsetBaseForTesting(scroll_layer->element_id(),
kScrollOffset);
SetScrollOffsetDelta(scroll_layer, kScrollDelta);
ExecuteCalculateDrawProperties(root_layer, kDeviceScale, page_scale,
scroll_layer->test_properties()->parent,
nullptr, nullptr);
gfx::Transform expected_transform;
gfx::PointF sub_layer_screen_position = kScrollLayerPosition - kScrollDelta;
expected_transform.Translate(
std::round(sub_layer_screen_position.x() * page_scale * kDeviceScale),
std::round(sub_layer_screen_position.y() * page_scale * kDeviceScale));
expected_transform.Scale(page_scale * kDeviceScale,
page_scale * 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);
scroll_layer->test_properties()->transform = arbitrary_translate;
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_layer, kDeviceScale, page_scale,
scroll_layer->test_properties()->parent,
nullptr, nullptr);
expected_transform.MakeIdentity();
expected_transform.Translate(
std::round(kTranslateX * page_scale * kDeviceScale +
sub_layer_screen_position.x() * page_scale * kDeviceScale),
std::round(kTranslateY * page_scale * kDeviceScale +
sub_layer_screen_position.y() * page_scale * kDeviceScale));
expected_transform.Scale(page_scale * kDeviceScale,
page_scale * kDeviceScale);
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_transform,
sublayer->DrawTransform());
// Test that page scale is updated even when we don't rebuild property trees.
page_scale = 1.888f;
LayerTreeImpl::ViewportLayerIds viewport_ids;
viewport_ids.page_scale = scroll_layer->test_properties()->parent->id();
root_layer->layer_tree_impl()->SetViewportLayersFromIds(viewport_ids);
root_layer->layer_tree_impl()->SetPageScaleOnActiveTree(page_scale);
EXPECT_FALSE(root_layer->layer_tree_impl()->property_trees()->needs_rebuild);
ExecuteCalculateDrawProperties(root_layer, kDeviceScale, page_scale,
scroll_layer->test_properties()->parent,
nullptr, nullptr);
expected_transform.MakeIdentity();
expected_transform.Translate(
std::round(kTranslateX * page_scale * kDeviceScale +
sub_layer_screen_position.x() * page_scale * kDeviceScale),
std::round(kTranslateY * page_scale * kDeviceScale +
sub_layer_screen_position.y() * page_scale * kDeviceScale));
expected_transform.Scale(page_scale * kDeviceScale,
page_scale * kDeviceScale);
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_transform,
sublayer->DrawTransform());
}
TEST_F(LayerTreeHostCommonTest, TransformsForSimpleHierarchy) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(parent);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
// One-time setup of root layer
root->SetBounds(gfx::Size(1, 2));
TransformTree& transform_tree =
host_impl()->active_tree()->property_trees()->transform_tree;
EffectTree& effect_tree =
host_impl()->active_tree()->property_trees()->effect_tree;
// Case 1: parent's anchor point should not affect child or grand_child.
parent->test_properties()->transform_origin = gfx::Point3F(2.5f, 3.0f, 0.f);
parent->SetBounds(gfx::Size(10, 12));
child->SetBounds(gfx::Size(16, 18));
grand_child->SetBounds(gfx::Size(76, 78));
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::DrawTransform(child, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::ScreenSpaceTransform(child, transform_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(), draw_property_utils::DrawTransform(
grand_child, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(),
draw_property_utils::ScreenSpaceTransform(grand_child, transform_tree));
// Case 2: parent's position affects child and grand_child.
gfx::Transform parent_position_transform;
parent_position_transform.Translate(0.f, 1.2f);
parent->test_properties()->position = gfx::PointF(0.f, 1.2f);
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_position_transform,
draw_property_utils::DrawTransform(child, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_position_transform,
draw_property_utils::ScreenSpaceTransform(child, transform_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_position_transform, draw_property_utils::DrawTransform(
grand_child, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_position_transform,
draw_property_utils::ScreenSpaceTransform(grand_child, transform_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);
parent->test_properties()->transform = parent_layer_transform;
parent->test_properties()->position = gfx::PointF();
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_composite_transform,
draw_property_utils::DrawTransform(child, transform_tree, effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_composite_transform,
draw_property_utils::ScreenSpaceTransform(child, transform_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_composite_transform,
draw_property_utils::DrawTransform(grand_child, transform_tree,
effect_tree));
EXPECT_TRANSFORMATION_MATRIX_EQ(
parent_composite_transform,
draw_property_utils::ScreenSpaceTransform(grand_child, transform_tree));
}
TEST_F(LayerTreeHostCommonTest, TransformsForSingleRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChildToRoot<LayerImpl>();
LayerImpl* child = AddChild<LayerImpl>(parent);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
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);
root->SetBounds(gfx::Size(1, 2));
parent->test_properties()->transform = parent_layer_transform;
parent->test_properties()->transform_origin = gfx::Point3F(2.5f, 30.f, 0.f);
parent->SetBounds(gfx::Size(100, 120));
child->SetBounds(gfx::Size(16, 18));
child->test_properties()->force_render_surface = true;
grand_child->SetBounds(gfx::Size(8, 10));
grand_child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
// Render surface should have been created now.
ASSERT_TRUE(GetRenderSurface(child));
ASSERT_EQ(GetRenderSurface(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()->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()->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
// - 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_for_testing();
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);
// In combination with descendant draws content, opacity != 1 forces the layer
// to have a new render surface.
render_surface1->test_properties()->opacity = 0.5f;
render_surface2->test_properties()->opacity = 0.33f;
// 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.
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 A =
translation_to_anchor * 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);
root->SetBounds(gfx::Size(1, 2));
parent->test_properties()->transform_origin = gfx::Point3F(2.5f, 0.f, 0.f);
parent->test_properties()->transform = layer_transform;
parent->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
render_surface1->test_properties()->transform = layer_transform;
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
render_surface2->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
render_surface2->test_properties()->transform = layer_transform;
render_surface2->SetBounds(gfx::Size(10, 10));
render_surface2->test_properties()->force_render_surface = true;
child_of_root->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
child_of_root->test_properties()->transform = layer_transform;
child_of_root->SetBounds(gfx::Size(10, 10));
child_of_rs1->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
child_of_rs1->test_properties()->transform = layer_transform;
child_of_rs1->SetBounds(gfx::Size(10, 10));
child_of_rs2->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
child_of_rs2->test_properties()->transform = layer_transform;
child_of_rs2->SetBounds(gfx::Size(10, 10));
grand_child_of_root->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
grand_child_of_root->test_properties()->transform = layer_transform;
grand_child_of_root->SetBounds(gfx::Size(10, 10));
grand_child_of_rs1->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
grand_child_of_rs1->test_properties()->transform = layer_transform;
grand_child_of_rs1->SetBounds(gfx::Size(10, 10));
grand_child_of_rs2->test_properties()->transform_origin =
gfx::Point3F(2.5f, 0.f, 0.f);
grand_child_of_rs2->test_properties()->transform = layer_transform;
grand_child_of_rs2->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawProperties(root);
// Only layers that are associated with render surfaces should have an actual
// RenderSurface() value.
ASSERT_TRUE(GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(child_of_root), GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(grand_child_of_root), GetRenderSurface(root));
ASSERT_NE(GetRenderSurface(render_surface1), GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(child_of_rs1), GetRenderSurface(render_surface1));
ASSERT_EQ(GetRenderSurface(grand_child_of_rs1),
GetRenderSurface(render_surface1));
ASSERT_NE(GetRenderSurface(render_surface2), GetRenderSurface(root));
ASSERT_NE(GetRenderSurface(render_surface2),
GetRenderSurface(render_surface1));
ASSERT_EQ(GetRenderSurface(child_of_rs2), GetRenderSurface(render_surface2));
ASSERT_EQ(GetRenderSurface(grand_child_of_rs2),
GetRenderSurface(render_surface2));
// Verify all render target accessors
EXPECT_EQ(GetRenderSurface(root), parent->render_target());
EXPECT_EQ(GetRenderSurface(root), child_of_root->render_target());
EXPECT_EQ(GetRenderSurface(root), grand_child_of_root->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1),
render_surface1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1), child_of_rs1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1),
grand_child_of_rs1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface2),
render_surface2->render_target());
EXPECT_EQ(GetRenderSurface(render_surface2), child_of_rs2->render_target());
EXPECT_EQ(GetRenderSurface(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, GetRenderSurface(render_surface1)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(
A * A * S1, GetRenderSurface(render_surface1)->screen_space_transform());
// Draw transform of render surface 2 is described with respect to render
// surface 1.
EXPECT_TRANSFORMATION_MATRIX_EQ(
SS1 * A * S2, GetRenderSurface(render_surface2)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(
A * A * A * S2,
GetRenderSurface(render_surface2)->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_for_testing();
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);
root->SetBounds(gfx::Size(100, 100));
child->test_properties()->transform = rotation_about_y_axis;
child->SetBounds(gfx::Size(10, 10));
child->test_properties()->force_render_surface = true;
grand_child->test_properties()->transform = rotation_about_y_axis;
grand_child->SetBounds(gfx::Size(10, 10));
great_grand_child->SetBounds(gfx::Size(10, 10));
// No layers in this test should preserve 3d.
ASSERT_TRUE(root->test_properties()->should_flatten_transform);
ASSERT_TRUE(child->test_properties()->should_flatten_transform);
ASSERT_TRUE(grand_child->test_properties()->should_flatten_transform);
ASSERT_TRUE(great_grand_child->test_properties()->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(GetRenderSurface(child));
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_child_draw_transform,
GetRenderSurface(child)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_child_screen_space_transform,
GetRenderSurface(child)->screen_space_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(), 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) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
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);
root->SetBounds(gfx::Size(200, 200));
child->test_properties()->transform = child_transform;
child->SetBounds(gfx::Size(10, 10));
grand_child->test_properties()->transform = grand_child_transform;
grand_child->SetBounds(gfx::Size(100, 100));
grand_child->test_properties()->should_flatten_transform = false;
grand_child->SetDrawsContent(true);
float device_scale_factor = 1.f;
float page_scale_factor = 1.f;
LayerImpl* page_scale_layer = nullptr;
LayerImpl* inner_viewport_scroll_layer = nullptr;
LayerImpl* outer_viewport_scroll_layer = nullptr;
ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor,
page_scale_layer, inner_viewport_scroll_layer,
outer_viewport_scroll_layer);
// Mapping grand_child's bounds to screen space produces an empty rect, but
// only because it is turned sideways. The entire rect is contained inside
// the clip, and is only empty so long as the numerical precision of the
// transform is effectively perfect. Currently we do the calculation the
// other way around, and the Projection of the screen space clip into layer
// space includes the entire bounds.
EXPECT_EQ(gfx::Rect(grand_child->bounds()),
grand_child->visible_layer_rect());
}
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_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
grand_child->SetDrawsContent(true);
root->SetBounds(gfx::Size(100, 100));
// The child height is zero, but has non-zero width that should be accounted
// for while computing draw transforms.
child->SetBounds(gfx::Size(10, 0));
child->test_properties()->force_render_surface = true;
grand_child->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(child));
// This is the real test, the rest are sanity checks.
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(child)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(), child->DrawTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
grand_child->DrawTransform());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceWithSublayerScale) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(render_surface);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
gfx::Transform translate;
translate.Translate3d(5, 5, 5);
root->SetBounds(gfx::Size(100, 100));
render_surface->test_properties()->transform = translate;
render_surface->SetBounds(gfx::Size(100, 100));
render_surface->test_properties()->force_render_surface = true;
child->test_properties()->transform = translate;
child->SetBounds(gfx::Size(100, 100));
grand_child->test_properties()->transform = translate;
grand_child->SetBounds(gfx::Size(100, 100));
grand_child->SetDrawsContent(true);
// render_surface will have a sublayer scale because of device scale factor.
float device_scale_factor = 2.0f;
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), translate, &render_surface_list_impl);
inputs.device_scale_factor = device_scale_factor;
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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.
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
root->SetDrawsContent(true);
root->SetBounds(gfx::Size(100, 100));
child->SetDrawsContent(true);
child->SetBounds(gfx::Size(100, 100));
child->SetMasksToBounds(true);
gfx::Transform translate;
translate.Translate(50, 50);
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), translate, &render_surface_list_impl);
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_TRANSFORMATION_MATRIX_EQ(
translate, root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
translate, child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(translate, child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(50, 50, 100, 100), child->clip_rect());
}
gfx::Transform scale;
scale.Scale(2, 2);
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), scale, &render_surface_list_impl);
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_TRANSFORMATION_MATRIX_EQ(
scale, root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
scale, child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(scale, child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(0, 0, 200, 200), child->clip_rect());
}
gfx::Transform rotate;
rotate.Rotate(2);
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), rotate, &render_surface_list_impl);
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_TRANSFORMATION_MATRIX_EQ(
rotate, root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
rotate, child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(rotate, child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(-4, 0, 104, 104), child->clip_rect());
}
gfx::Transform composite;
composite.ConcatTransform(translate);
composite.ConcatTransform(scale);
composite.ConcatTransform(rotate);
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), composite, &render_surface_list_impl);
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_TRANSFORMATION_MATRIX_EQ(
composite, root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
composite, child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(composite, child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(89, 103, 208, 208), child->clip_rect());
}
// Verify it composes correctly with device scale.
float device_scale_factor = 1.5f;
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), translate, &render_surface_list_impl);
inputs.device_scale_factor = device_scale_factor;
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
gfx::Transform device_scaled_translate = translate;
device_scaled_translate.Scale(device_scale_factor, device_scale_factor);
EXPECT_TRANSFORMATION_MATRIX_EQ(
device_scaled_translate,
root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
device_scaled_translate,
child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(device_scaled_translate,
child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(50, 50, 150, 150), child->clip_rect());
}
// Verify it composes correctly with page scale.
float page_scale_factor = 2.f;
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), translate, &render_surface_list_impl);
inputs.page_scale_factor = page_scale_factor;
inputs.page_scale_layer = root;
inputs.page_scale_transform_node =
inputs.property_trees->transform_tree.Node(
inputs.page_scale_layer->transform_tree_index());
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
gfx::Transform page_scaled_translate = translate;
page_scaled_translate.Scale(page_scale_factor, page_scale_factor);
EXPECT_TRANSFORMATION_MATRIX_EQ(
page_scaled_translate, root->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(
page_scaled_translate, child->draw_properties().target_space_transform);
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(page_scaled_translate,
child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(50, 50, 200, 200), child->clip_rect());
}
// Verify that it composes correctly with transforms directly on root layer.
root->test_properties()->transform = composite;
{
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), composite, &render_surface_list_impl);
inputs.property_trees->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(root)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(compositeSquared,
child->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(254, 316, 428, 428), child->clip_rect());
}
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceForNonAxisAlignedClipping) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* rotated_and_transparent = AddChildToRoot<LayerImpl>();
LayerImpl* clips_subtree = AddChild<LayerImpl>(rotated_and_transparent);
LayerImpl* draws_content = AddChild<LayerImpl>(clips_subtree);
root->SetBounds(gfx::Size(10, 10));
rotated_and_transparent->SetBounds(gfx::Size(10, 10));
rotated_and_transparent->test_properties()->opacity = 0.5f;
gfx::Transform rotate;
rotate.Rotate(2);
rotated_and_transparent->test_properties()->transform = rotate;
clips_subtree->SetBounds(gfx::Size(10, 10));
clips_subtree->SetMasksToBounds(true);
draws_content->SetBounds(gfx::Size(10, 10));
draws_content->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EffectTree& effect_tree =
root->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = effect_tree.Node(clips_subtree->effect_tree_index());
EXPECT_TRUE(node->has_render_surface);
}
TEST_F(LayerTreeHostCommonTest, EffectNodesForNonAxisAlignedClips) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* rotate_and_clip = AddChildToRoot<LayerImpl>();
LayerImpl* only_clip = AddChild<LayerImpl>(rotate_and_clip);
LayerImpl* rotate_and_clip2 = AddChild<LayerImpl>(only_clip);
gfx::Transform rotate;
rotate.Rotate(2);
root->SetBounds(gfx::Size(10, 10));
rotate_and_clip->SetBounds(gfx::Size(10, 10));
rotate_and_clip->test_properties()->transform = rotate;
rotate_and_clip->SetMasksToBounds(true);
only_clip->SetBounds(gfx::Size(10, 10));
only_clip->SetMasksToBounds(true);
rotate_and_clip2->SetBounds(gfx::Size(10, 10));
rotate_and_clip2->test_properties()->transform = rotate;
rotate_and_clip2->SetMasksToBounds(true);
ExecuteCalculateDrawProperties(root);
// non-axis aligned clip should create an effect node
EXPECT_NE(root->effect_tree_index(), rotate_and_clip->effect_tree_index());
// Since only_clip's clip is in the same non-axis aligned space as
// rotate_and_clip's clip, no new effect node should be created.
EXPECT_EQ(rotate_and_clip->effect_tree_index(),
only_clip->effect_tree_index());
// rotate_and_clip2's clip and only_clip's clip are in different non-axis
// aligned spaces. So, new effect node should be created.
EXPECT_NE(rotate_and_clip2->effect_tree_index(),
only_clip->effect_tree_index());
}
TEST_F(LayerTreeHostCommonTest,
RenderSurfaceListForRenderSurfaceWithClippedLayer) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>();
LayerImpl* child = AddChild<LayerImpl>(render_surface1);
root->SetBounds(gfx::Size(10, 10));
root->SetMasksToBounds(true);
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
child->SetDrawsContent(true);
child->test_properties()->position = gfx::PointF(30.f, 30.f);
child->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawProperties(root);
// The child layer's content is entirely outside the root'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(GetRenderSurface(root));
EXPECT_EQ(1U, render_surface_list_impl()->size());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceListForTransparentChild) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(render_surface1);
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
render_surface1->test_properties()->opacity = 0.f;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
// Since the layer is transparent, render_surface1->GetRenderSurface() should
// not have gotten added anywhere. Also, the drawable content rect should not
// have been extended by the children.
ASSERT_TRUE(GetRenderSurface(root));
EXPECT_EQ(0, GetRenderSurface(root)->num_contributors());
EXPECT_EQ(1U, render_surface_list.size());
EXPECT_EQ(static_cast<viz::RenderPassId>(root->id()),
render_surface_list.at(0)->id());
EXPECT_EQ(gfx::Rect(), root->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonTest,
RenderSurfaceListForTransparentChildWithBackdropFilter) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(render_surface1);
root->SetBounds(gfx::Size(10, 10));
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
render_surface1->test_properties()->opacity = 0.f;
render_surface1->SetDrawsContent(true);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(1.5f));
render_surface1->test_properties()->backdrop_filters = filters;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
root->layer_tree_impl()->SetElementIdsForTesting();
{
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_EQ(2U, render_surface_list.size());
}
// The layer is fully transparent, but has a backdrop filter, so it
// shouldn't be skipped and should be drawn.
ASSERT_TRUE(GetRenderSurface(root));
EXPECT_EQ(1, GetRenderSurface(root)->num_contributors());
EXPECT_EQ(gfx::RectF(0, 0, 10, 10),
GetRenderSurface(root)->DrawableContentRect());
EffectTree& effect_tree =
root->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = effect_tree.Node(render_surface1->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
// When root is transparent, the layer should not be drawn.
root->layer_tree_impl()->SetOpacityMutated(root->element_id(), 0.f);
root->layer_tree_impl()->SetOpacityMutated(render_surface1->element_id(),
1.f);
render_surface1->set_visible_layer_rect(gfx::Rect());
{
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
}
node = effect_tree.Node(render_surface1->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
EXPECT_EQ(gfx::Rect(), render_surface1->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceListForFilter) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChild<LayerImpl>(root);
LayerImpl* child1 = AddChild<LayerImpl>(parent);
LayerImpl* child2 = AddChild<LayerImpl>(parent);
gfx::Transform scale_matrix;
scale_matrix.Scale(2.0f, 2.0f);
root->SetBounds(gfx::Size(100, 100));
parent->test_properties()->transform = scale_matrix;
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(10.0f));
parent->test_properties()->filters = filters;
parent->test_properties()->force_render_surface = true;
child1->SetBounds(gfx::Size(25, 25));
child1->SetDrawsContent(true);
child1->test_properties()->force_render_surface = true;
child2->test_properties()->position = gfx::PointF(25, 25);
child2->SetBounds(gfx::Size(25, 25));
child2->SetDrawsContent(true);
child2->test_properties()->force_render_surface = true;
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
ASSERT_TRUE(GetRenderSurface(parent));
EXPECT_EQ(2, GetRenderSurface(parent)->num_contributors());
EXPECT_EQ(4U, render_surface_list.size());
// The rectangle enclosing child1 and child2 (0,0 50x50), expanded for the
// blur (-30,-30 110x110), and then scaled by the scale matrix
// (-60,-60 220x220).
EXPECT_EQ(gfx::RectF(-60, -60, 220, 220),
GetRenderSurface(parent)->DrawableContentRect());
}
TEST_F(LayerTreeHostCommonTest, DrawableContentRectForReferenceFilter) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(25, 25));
child->SetDrawsContent(true);
child->test_properties()->force_render_surface = true;
FilterOperations filters;
filters.Append(FilterOperation::CreateReferenceFilter(
sk_make_sp<OffsetPaintFilter>(50, 50, nullptr)));
child->test_properties()->filters = filters;
ExecuteCalculateDrawProperties(root);
// The render surface's size should be unaffected by the offset image filter;
// it need only have a drawable content rect large enough to contain the
// contents (at the new offset).
ASSERT_TRUE(GetRenderSurface(child));
EXPECT_EQ(gfx::RectF(50, 50, 25, 25),
GetRenderSurface(child)->DrawableContentRect());
}
TEST_F(LayerTreeHostCommonTest, DrawableContentRectForReferenceFilterHighDpi) {
const float device_scale_factor = 2.0f;
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(25, 25));
child->SetDrawsContent(true);
child->test_properties()->force_render_surface = true;
FilterOperations filters;
filters.Append(FilterOperation::CreateReferenceFilter(
sk_make_sp<OffsetPaintFilter>(50, 50, nullptr)));
child->test_properties()->filters = filters;
ExecuteCalculateDrawProperties(root, device_scale_factor);
// The render surface's size should be unaffected by the offset image filter;
// it need only have a drawable content rect large enough to contain the
// contents (at the new offset). All coordinates should be scaled by 2,
// corresponding to the device scale factor.
ASSERT_TRUE(GetRenderSurface(child));
EXPECT_EQ(gfx::RectF(100, 100, 50, 50),
GetRenderSurface(child)->DrawableContentRect());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceForBlendMode) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
root->SetBounds(gfx::Size(10, 10));
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
child->test_properties()->blend_mode = SkBlendMode::kMultiply;
child->test_properties()->opacity = 0.5f;
child->test_properties()->force_render_surface = true;
ExecuteCalculateDrawProperties(root);
// Since the child layer has a blend mode other than normal, it should get
// its own render surface.
ASSERT_TRUE(GetRenderSurface(child));
EXPECT_EQ(1.0f, child->draw_opacity());
EXPECT_EQ(0.5f, GetRenderSurface(child)->draw_opacity());
EXPECT_EQ(SkBlendMode::kMultiply, GetRenderSurface(child)->BlendMode());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceDrawOpacity) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* surface1 = AddChildToRoot<LayerImpl>();
LayerImpl* not_surface = AddChild<LayerImpl>(surface1);
LayerImpl* surface2 = AddChild<LayerImpl>(not_surface);
root->SetBounds(gfx::Size(10, 10));
surface1->SetBounds(gfx::Size(10, 10));
surface1->SetDrawsContent(true);
surface1->test_properties()->opacity = 0.5f;
surface1->test_properties()->force_render_surface = true;
not_surface->SetBounds(gfx::Size(10, 10));
not_surface->test_properties()->opacity = 0.5f;
surface2->SetBounds(gfx::Size(10, 10));
surface2->SetDrawsContent(true);
surface2->test_properties()->opacity = 0.5f;
surface2->test_properties()->force_render_surface = true;
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(surface1));
ASSERT_EQ(GetRenderSurface(not_surface), GetRenderSurface(surface1));
ASSERT_TRUE(GetRenderSurface(surface2));
EXPECT_EQ(0.5f, GetRenderSurface(surface1)->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, GetRenderSurface(surface2)->draw_opacity());
}
TEST_F(LayerTreeHostCommonTest, ForceRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>();
LayerImpl* child = AddChild<LayerImpl>(render_surface1);
root->SetBounds(gfx::Size(10, 10));
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
{
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
// The root layer always creates a render surface
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(render_surface1), GetRenderSurface(root));
}
{
render_surface1->test_properties()->force_render_surface = false;
render_surface1->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(render_surface1), GetRenderSurface(root));
}
}
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_for_testing();
LayerImpl* parent = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(parent);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
gfx::Transform rotation_about_y_axis;
rotation_about_y_axis.RotateAboutYAxis(30.0);
root->SetBounds(gfx::Size(100, 100));
parent->test_properties()->transform = rotation_about_y_axis;
parent->SetBounds(gfx::Size(10, 10));
parent->test_properties()->force_render_surface = true;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
grand_child->SetBounds(gfx::Size(10, 10));
grand_child->SetDrawsContent(true);
grand_child->test_properties()->should_flatten_transform = false;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(parent));
EXPECT_EQ(GetRenderSurface(child), GetRenderSurface(parent));
EXPECT_EQ(GetRenderSurface(grand_child), GetRenderSurface(parent));
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(), child->DrawTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
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.
gfx::Transform flattened_rotation_about_y = rotation_about_y_axis;
flattened_rotation_about_y.FlattenTo2d();
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_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* root = root_layer_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* grand_child = AddChild<LayerImpl>(child);
LayerImpl* great_grand_child = AddChild<LayerImpl>(grand_child);
// leaf_node1 ensures that root and child are kept on the render_surface_list,
// even though grand_child and great_grand_child should be clipped.
LayerImpl* leaf_node1 = AddChild<LayerImpl>(child);
LayerImpl* leaf_node2 = AddChild<LayerImpl>(great_grand_child);
root->SetBounds(gfx::Size(500, 500));
child->SetBounds(gfx::Size(20, 20));
child->SetMasksToBounds(true);
child->test_properties()->force_render_surface = true;
grand_child->test_properties()->position = gfx::PointF(45.f, 45.f);
grand_child->SetBounds(gfx::Size(10, 10));
great_grand_child->SetBounds(gfx::Size(10, 10));
leaf_node1->SetBounds(gfx::Size(500, 500));
leaf_node1->SetDrawsContent(true);
leaf_node1->SetBounds(gfx::Size(20, 20));
leaf_node2->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
ASSERT_EQ(2U, render_surface_list_impl()->size());
EXPECT_EQ(static_cast<uint64_t>(root->id()),
render_surface_list_impl()->at(0)->id());
EXPECT_EQ(static_cast<uint64_t>(child->id()),
render_surface_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:
// - root 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 root.
// - grand_child is a render surface, and the only visible content in child.
// It is positioned outside of the clip rect from root.
// In this configuration, grand_child should be outside the clipped
// content rect of the child, making grand_child not appear in the
// render_surface_list.
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* grand_child = AddChild<LayerImpl>(child);
LayerImpl* leaf_node = AddChild<LayerImpl>(grand_child);
root->SetMasksToBounds(true);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(20, 20));
child->test_properties()->force_render_surface = true;
grand_child->test_properties()->position = gfx::PointF(200.f, 200.f);
grand_child->SetBounds(gfx::Size(10, 10));
grand_child->test_properties()->force_render_surface = true;
leaf_node->SetBounds(gfx::Size(10, 10));
leaf_node->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
// We should cull child and grand_child from the
// render_surface_list.
ASSERT_EQ(1U, render_surface_list_impl()->size());
EXPECT_EQ(static_cast<uint64_t>(root->id()),
render_surface_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.
LayerImpl* root = root_layer_for_testing();
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->SetBounds(gfx::Size(100, 100));
parent->SetBounds(gfx::Size(100, 100));
parent->SetDrawsContent(true);
child1->SetBounds(gfx::Size(100, 100));
child1->SetDrawsContent(true);
child2->SetBounds(gfx::Size(100, 100));
child2->SetDrawsContent(true);
child2->test_properties()->force_render_surface = true;
grand_child->SetBounds(gfx::Size(100, 100));
grand_child->SetDrawsContent(true);
leaf_node1->SetBounds(gfx::Size(100, 100));
leaf_node1->SetDrawsContent(true);
leaf_node2->SetBounds(gfx::Size(100, 100));
leaf_node2->SetDrawsContent(true);
// Case 1: nothing is clipped except the root render surface.
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(root));
ASSERT_TRUE(GetRenderSurface(child2));
EXPECT_FALSE(root->is_clipped());
EXPECT_TRUE(GetRenderSurface(root)->is_clipped());
EXPECT_FALSE(parent->is_clipped());
EXPECT_FALSE(child1->is_clipped());
EXPECT_FALSE(child2->is_clipped());
EXPECT_FALSE(GetRenderSurface(child2)->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(GetRenderSurface(root));
ASSERT_TRUE(GetRenderSurface(child2));
EXPECT_FALSE(root->is_clipped());
EXPECT_TRUE(GetRenderSurface(root)->is_clipped());
EXPECT_TRUE(parent->is_clipped());
EXPECT_TRUE(child1->is_clipped());
EXPECT_FALSE(child2->is_clipped());
EXPECT_TRUE(GetRenderSurface(child2)->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(GetRenderSurface(root));
ASSERT_TRUE(GetRenderSurface(child2));
EXPECT_FALSE(root->is_clipped());
EXPECT_TRUE(GetRenderSurface(root)->is_clipped());
EXPECT_FALSE(parent->is_clipped());
EXPECT_FALSE(child1->is_clipped());
EXPECT_TRUE(child2->is_clipped());
EXPECT_FALSE(GetRenderSurface(child2)->is_clipped());
EXPECT_FALSE(grand_child->is_clipped());
EXPECT_FALSE(leaf_node1->is_clipped());
EXPECT_TRUE(leaf_node2->is_clipped());
}
TEST_F(LayerTreeHostCommonTest, UpdateClipRectCorrectly) {
// Tests that when as long as layer is clipped, it's clip rect is set to
// correct value.
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(parent);
root->SetBounds(gfx::Size(100, 100));
root->SetDrawsContent(true);
parent->SetBounds(gfx::Size(100, 100));
parent->SetDrawsContent(true);
child->SetBounds(gfx::Size(100, 100));
child->SetDrawsContent(true);
child->SetMasksToBounds(true);
ExecuteCalculateDrawProperties(root);
EXPECT_FALSE(root->is_clipped());
EXPECT_FALSE(parent->is_clipped());
EXPECT_TRUE(child->is_clipped());
EXPECT_EQ(gfx::Rect(100, 100), child->clip_rect());
parent->SetMasksToBounds(true);
child->test_properties()->position = gfx::PointF(100.f, 100.f);
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_FALSE(root->is_clipped());
EXPECT_TRUE(parent->is_clipped());
EXPECT_TRUE(child->is_clipped());
EXPECT_EQ(gfx::Rect(), child->clip_rect());
}
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.
LayerImpl* parent = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(parent);
LayerImpl* grand_child1 = AddChild<LayerImpl>(child);
LayerImpl* grand_child2 = AddChild<LayerImpl>(child);
LayerImpl* grand_child3 = AddChild<LayerImpl>(child);
LayerImpl* grand_child4 = AddChild<LayerImpl>(child);
parent->SetBounds(gfx::Size(500, 500));
child->SetMasksToBounds(true);
child->SetBounds(gfx::Size(20, 20));
child->test_properties()->force_render_surface = true;
grand_child1->test_properties()->position = gfx::PointF(5.f, 5.f);
grand_child1->SetBounds(gfx::Size(10, 10));
grand_child1->SetDrawsContent(true);
grand_child2->test_properties()->position = gfx::PointF(15.f, 15.f);
grand_child2->SetBounds(gfx::Size(10, 10));
grand_child2->SetDrawsContent(true);
grand_child3->test_properties()->position = gfx::PointF(15.f, 15.f);
grand_child3->SetMasksToBounds(true);
grand_child3->SetBounds(gfx::Size(10, 10));
grand_child3->SetDrawsContent(true);
grand_child4->test_properties()->position = gfx::PointF(45.f, 45.f);
grand_child4->SetBounds(gfx::Size(10, 10));
grand_child4->SetDrawsContent(true);
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_for_testing();
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);
LayerImpl* leaf_node2 = AddChild<LayerImpl>(grand_child2);
LayerImpl* leaf_node3 = AddChild<LayerImpl>(grand_child3);
LayerImpl* leaf_node4 = AddChild<LayerImpl>(grand_child4);
parent->SetBounds(gfx::Size(500, 500));
child->SetBounds(gfx::Size(20, 20));
child->SetMasksToBounds(true);
child->test_properties()->force_render_surface = true;
grand_child1->test_properties()->position = gfx::PointF(5.f, 5.f);
grand_child1->SetBounds(gfx::Size(10, 10));
grand_child1->test_properties()->force_render_surface = true;
grand_child2->test_properties()->position = gfx::PointF(15.f, 15.f);
grand_child2->SetBounds(gfx::Size(10, 10));
grand_child2->test_properties()->force_render_surface = true;
grand_child3->test_properties()->position = gfx::PointF(15.f, 15.f);
grand_child3->SetBounds(gfx::Size(10, 10));
grand_child3->SetMasksToBounds(true);
grand_child3->test_properties()->force_render_surface = true;
grand_child4->test_properties()->position = gfx::PointF(45.f, 45.f);
grand_child4->SetBounds(gfx::Size(10, 10));
grand_child4->SetMasksToBounds(true);
grand_child4->test_properties()->force_render_surface = true;
leaf_node1->SetBounds(gfx::Size(10, 10));
leaf_node1->SetDrawsContent(true);
leaf_node2->SetBounds(gfx::Size(10, 10));
leaf_node2->SetDrawsContent(true);
leaf_node3->SetBounds(gfx::Size(10, 10));
leaf_node3->SetDrawsContent(true);
leaf_node4->SetBounds(gfx::Size(10, 10));
leaf_node4->SetDrawsContent(true);
ExecuteCalculateDrawProperties(parent);
ASSERT_TRUE(GetRenderSurface(grand_child1));
ASSERT_TRUE(GetRenderSurface(grand_child2));
ASSERT_TRUE(GetRenderSurface(grand_child3));
// Surfaces are clipped by their parent, but un-affected by the owning layer's
// MasksToBounds.
EXPECT_EQ(gfx::Rect(0, 0, 20, 20),
GetRenderSurface(grand_child1)->clip_rect());
EXPECT_EQ(gfx::Rect(0, 0, 20, 20),
GetRenderSurface(grand_child2)->clip_rect());
EXPECT_EQ(gfx::Rect(0, 0, 20, 20),
GetRenderSurface(grand_child3)->clip_rect());
}
TEST_F(LayerTreeHostCommonTest, AnimationsForRenderSurfaceHierarchy) {
LayerImpl* root = root_layer_for_testing();
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);
root->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);
root->test_properties()->transform = layer_transform;
root->SetBounds(gfx::Size(10, 10));
root->test_properties()->transform_origin = gfx::Point3F(0.25f, 0.f, 0.f);
render_surface1->test_properties()->transform = layer_transform;
render_surface1->test_properties()->position = gfx::PointF(2.5f, 0.f);
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
render_surface1->test_properties()->force_render_surface = true;
render_surface2->test_properties()->transform = layer_transform;
render_surface2->test_properties()->position = gfx::PointF(2.5f, 0.f);
render_surface2->SetBounds(gfx::Size(10, 10));
render_surface2->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
render_surface2->test_properties()->force_render_surface = true;
child_of_root->test_properties()->transform = layer_transform;
child_of_root->test_properties()->position = gfx::PointF(2.5f, 0.f);
child_of_root->SetBounds(gfx::Size(10, 10));
child_of_root->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
child_of_rs1->test_properties()->transform = layer_transform;
child_of_rs1->test_properties()->position = gfx::PointF(2.5f, 0.f);
child_of_rs1->SetBounds(gfx::Size(10, 10));
child_of_rs1->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
child_of_rs2->test_properties()->transform = layer_transform;
child_of_rs2->test_properties()->position = gfx::PointF(2.5f, 0.f);
child_of_rs2->SetBounds(gfx::Size(10, 10));
child_of_rs2->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
grand_child_of_root->test_properties()->transform = layer_transform;
grand_child_of_root->test_properties()->position = gfx::PointF(2.5f, 0.f);
grand_child_of_root->SetBounds(gfx::Size(10, 10));
grand_child_of_root->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
grand_child_of_rs1->test_properties()->transform = layer_transform;
grand_child_of_rs1->test_properties()->position = gfx::PointF(2.5f, 0.f);
grand_child_of_rs1->SetBounds(gfx::Size(10, 10));
grand_child_of_rs1->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
grand_child_of_rs2->test_properties()->transform = layer_transform;
grand_child_of_rs2->test_properties()->position = gfx::PointF(2.5f, 0.f);
grand_child_of_rs2->SetBounds(gfx::Size(10, 10));
grand_child_of_rs2->test_properties()->transform_origin =
gfx::Point3F(0.25f, 0.f, 0.f);
root->layer_tree_impl()->BuildLayerListAndPropertyTreesForTesting();
// Put an animated opacity on the render surface.
AddOpacityTransitionToElementWithAnimation(
render_surface1->element_id(), timeline_impl(), 10.0, 1.f, 0.f, false);
// Also put an animated opacity on a layer without descendants.
AddOpacityTransitionToElementWithAnimation(grand_child_of_root->element_id(),
timeline_impl(), 10.0, 1.f, 0.f,
false);
// Put a transform animation on the render surface.
AddAnimatedTransformToElementWithAnimation(render_surface2->element_id(),
timeline_impl(), 10.0, 30, 0);
// Also put transform animations on grand_child_of_root, and
// grand_child_of_rs2
AddAnimatedTransformToElementWithAnimation(grand_child_of_root->element_id(),
timeline_impl(), 10.0, 30, 0);
AddAnimatedTransformToElementWithAnimation(grand_child_of_rs2->element_id(),
timeline_impl(), 10.0, 30, 0);
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
// Only layers that are associated with render surfaces should have an actual
// RenderSurface() value.
ASSERT_TRUE(GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(child_of_root), GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(grand_child_of_root), GetRenderSurface(root));
ASSERT_NE(GetRenderSurface(render_surface1), GetRenderSurface(root));
ASSERT_EQ(GetRenderSurface(child_of_rs1), GetRenderSurface(render_surface1));
ASSERT_EQ(GetRenderSurface(grand_child_of_rs1),
GetRenderSurface(render_surface1));
ASSERT_NE(GetRenderSurface(render_surface2), GetRenderSurface(root));
ASSERT_NE(GetRenderSurface(render_surface2),
GetRenderSurface(render_surface1));
ASSERT_EQ(GetRenderSurface(child_of_rs2), GetRenderSurface(render_surface2));
ASSERT_EQ(GetRenderSurface(grand_child_of_rs2),
GetRenderSurface(render_surface2));
// Verify all render target accessors
EXPECT_EQ(GetRenderSurface(root), root->render_target());
EXPECT_EQ(GetRenderSurface(root), child_of_root->render_target());
EXPECT_EQ(GetRenderSurface(root), grand_child_of_root->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1),
render_surface1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1), child_of_rs1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface1),
grand_child_of_rs1->render_target());
EXPECT_EQ(GetRenderSurface(render_surface2),
render_surface2->render_target());
EXPECT_EQ(GetRenderSurface(render_surface2), child_of_rs2->render_target());
EXPECT_EQ(GetRenderSurface(render_surface2),
grand_child_of_rs2->render_target());
// Verify screen_space_transform_is_animating values
EXPECT_FALSE(root->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, root->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* root = root_layer_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* grand_child = AddChild<LayerImpl>(child);
gfx::Transform large_transform;
large_transform.Scale(SkDoubleToMScalar(1e37), SkDoubleToMScalar(1e37));
root->SetBounds(gfx::Size(10, 10));
child->test_properties()->transform = large_transform;
child->SetBounds(gfx::Size(10, 10));
grand_child->test_properties()->transform = large_transform;
grand_child->SetBounds(gfx::Size(10, 10));
grand_child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(), grand_child->visible_layer_rect());
}
static bool TransformIsAnimating(LayerImpl* layer) {
MutatorHost* host = layer->layer_tree_impl()->mutator_host();
return host->IsAnimatingTransformProperty(
layer->element_id(), layer->GetElementTypeForAnimation());
}
static bool HasPotentiallyRunningTransformAnimation(LayerImpl* layer) {
MutatorHost* host = layer->layer_tree_impl()->mutator_host();
return host->HasPotentiallyRunningTransformAnimation(
layer->element_id(), layer->GetElementTypeForAnimation());
}
TEST_F(LayerTreeHostCommonTest,
ScreenSpaceTransformIsAnimatingWithDelayedAnimation) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
LayerImpl* great_grand_child = AddChild<LayerImpl>(grand_child);
root->SetDrawsContent(true);
child->SetDrawsContent(true);
grand_child->SetDrawsContent(true);
great_grand_child->SetDrawsContent(true);
root->SetBounds(gfx::Size(10, 10));
child->SetBounds(gfx::Size(10, 10));
grand_child->SetBounds(gfx::Size(10, 10));
great_grand_child->SetBounds(gfx::Size(10, 10));
SetElementIdsForTesting();
// Add a transform animation with a start delay to |grand_child|.
std::unique_ptr<KeyframeModel> keyframe_model = KeyframeModel::Create(
std::unique_ptr<AnimationCurve>(new FakeTransformTransition(1.0)), 0, 1,
TargetProperty::TRANSFORM);
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
AddKeyframeModelToElementWithAnimation(
grand_child->element_id(), timeline_impl(), std::move(keyframe_model));
ExecuteCalculateDrawProperties(root);
EXPECT_FALSE(root->screen_space_transform_is_animating());
EXPECT_FALSE(child->screen_space_transform_is_animating());
EXPECT_FALSE(TransformIsAnimating(grand_child));
EXPECT_TRUE(HasPotentiallyRunningTransformAnimation(grand_child));
EXPECT_TRUE(grand_child->screen_space_transform_is_animating());
EXPECT_TRUE(great_grand_child->screen_space_transform_is_animating());
}
TEST_F(LayerTreeHostCommonDrawRectsTest, DrawRectsForIdentityTransform) {
// Test visible layer rect and drawable content rect are calculated correctly
// 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_visible_layer_rect = gfx::Rect(30, 30);
gfx::Rect expected_drawable_content_rect = gfx::Rect(10, 10, 30, 30);
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// Case 2: Layer is outside the surface rect.
layer_content_rect = gfx::Rect(120, 120, 30, 30);
expected_visible_layer_rect = gfx::Rect();
expected_drawable_content_rect = gfx::Rect();
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// Case 3: Layer is partially overlapping the surface rect.
layer_content_rect = gfx::Rect(80, 80, 30, 30);
expected_visible_layer_rect = gfx::Rect(20, 20);
expected_drawable_content_rect = gfx::Rect(80, 80, 20, 20);
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonDrawRectsTest, DrawRectsFor2DRotations) {
// Test visible layer rect and drawable content rect are calculated correctly
// for rotations about z-axis (i.e. 2D rotations).
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_visible_layer_rect = gfx::Rect(30, 30);
gfx::Rect expected_drawable_content_rect = gfx::Rect(28, 50, 44, 43);
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// 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);
expected_visible_layer_rect = gfx::Rect();
expected_drawable_content_rect = gfx::Rect();
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// 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_visible_layer_rect = gfx::Rect(30, 30);
expected_drawable_content_rect = gfx::Rect(22, 43);
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// 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);
// Right half of layer bounds.
expected_visible_layer_rect = gfx::Rect(15, 0, 15, 30);
expected_drawable_content_rect = gfx::Rect(22, 22);
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonDrawRectsTest, DrawRectsFor3dOrthographicTransform) {
// Test visible layer rect and drawable content rect are calculated 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.
// 100 is the un-rotated layer width; divided by sqrt(2) is the rotated width.
layer_to_surface_transform.MakeIdentity();
layer_to_surface_transform.RotateAboutYAxis(45.0);
gfx::Rect expected_visible_layer_rect = gfx::Rect(100, 100);
gfx::Rect expected_drawable_content_rect = gfx::Rect(71, 100);
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// 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.
// 50 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.
// Tight half of the layer.
expected_visible_layer_rect = gfx::Rect(50, 0, 50, 100);
expected_drawable_content_rect = gfx::Rect(36, 100);
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonDrawRectsTest, DrawRectsFor3dPerspectiveTransform) {
// Test visible layer rect and drawable content rect are calculated 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);
// Layer position is (-50, -50), visible rect in layer space is layer bounds
// offset by layer position.
gfx::Rect expected_visible_layer_rect = gfx::Rect(50, 50, 150, 150);
gfx::Rect expected_drawable_content_rect = gfx::Rect(38, 38);
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
// 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 -25 units in projected surface units (so that
// only half of it is visible), then we would need to translate by (-36 / 9) *
// -25 == -100 in the layer's units.
layer_to_surface_transform.Translate3d(-100.0, 0.0, 0.0);
// Visible layer rect is moved by 100, and drawable content rect is in target
// space and is moved by 25.
expected_visible_layer_rect = gfx::Rect(150, 50, 50, 150);
expected_drawable_content_rect = gfx::Rect(13, 38);
drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonDrawRectsTest,
DrawRectsFor3dOrthographicIsNotClippedBehindSurface) {
// 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);
// Layer is rotated about Y Axis, and its width is 100/sqrt(2) in surface
// space.
gfx::Rect expected_visible_layer_rect = gfx::Rect(100, 100);
gfx::Rect expected_drawable_content_rect = gfx::Rect(14, 0, 72, 100);
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonDrawRectsTest,
DrawRectsFor3dPerspectiveWhenClippedByW) {
// Test visible layer rect and drawable content rect are calculated 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. The drawable
// content rect would be entire surface rect because layer is rotated at the
// camera position.
gfx::Rect target_surface_rect = gfx::Rect(0, 0, 200, 200);
gfx::Rect layer_content_rect = gfx::Rect(0, 0, 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(10.0, 0.0, 1.0);
layer_to_surface_transform.RotateAboutYAxis(-45.0);
layer_to_surface_transform.Translate(-10, -1);
// 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);
gfx::Rect expected_visible_layer_rect = gfx::Rect(0, 1, 10, 1);
gfx::Rect expected_drawable_content_rect = target_surface_rect;
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
static bool ProjectionClips(const gfx::Transform& map_transform,
const gfx::RectF& mapped_rect) {
gfx::Transform inverse(Inverse(map_transform));
bool clipped = false;
if (!clipped)
MathUtil::ProjectPoint(inverse, mapped_rect.top_right(), &clipped);
if (!clipped)
MathUtil::ProjectPoint(inverse, mapped_rect.origin(), &clipped);
if (!clipped)
MathUtil::ProjectPoint(inverse, mapped_rect.bottom_right(), &clipped);
if (!clipped)
MathUtil::ProjectPoint(inverse, mapped_rect.bottom_left(), &clipped);
return clipped;
}
TEST_F(LayerTreeHostCommonDrawRectsTest, DrawRectsForPerspectiveUnprojection) {
// 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(0, 0, 150, 150);
gfx::Rect layer_content_rect = gfx::Rect(0, 0, 20, 20);
gfx::Transform layer_to_surface_transform;
layer_to_surface_transform.MakeIdentity();
layer_to_surface_transform.Translate(10, 10);
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);
layer_to_surface_transform.Translate(-10, -10);
// Sanity check that un-projection does indeed cause w < 0, otherwise this
// code is not testing the intended scenario.
gfx::RectF clipped_rect = MathUtil::MapClippedRect(
layer_to_surface_transform, gfx::RectF(layer_content_rect));
ASSERT_TRUE(ProjectionClips(layer_to_surface_transform, clipped_rect));
// 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_visible_layer_rect = layer_content_rect;
gfx::Rect expected_drawable_content_rect = target_surface_rect;
LayerImpl* drawing_layer = TestVisibleRectAndDrawableContentRect(
target_surface_rect, layer_to_surface_transform, layer_content_rect);
EXPECT_EQ(expected_visible_layer_rect, drawing_layer->visible_layer_rect());
EXPECT_EQ(expected_drawable_content_rect,
drawing_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonTest, DrawableAndVisibleContentRectsForSimpleLayers) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child1_layer = AddChildToRoot<LayerImpl>();
LayerImpl* child2_layer = AddChildToRoot<LayerImpl>();
LayerImpl* child3_layer = AddChildToRoot<LayerImpl>();
root->SetBounds(gfx::Size(100, 100));
child1_layer->SetBounds(gfx::Size(50, 50));
child1_layer->SetDrawsContent(true);
child2_layer->test_properties()->position = gfx::PointF(75.f, 75.f);
child2_layer->SetBounds(gfx::Size(50, 50));
child2_layer->SetDrawsContent(true);
child3_layer->test_properties()->position = gfx::PointF(125.f, 125.f);
child3_layer->SetBounds(gfx::Size(50, 50));
child3_layer->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* grand_child1 = AddChild<LayerImpl>(child);
LayerImpl* grand_child2 = AddChild<LayerImpl>(child);
LayerImpl* grand_child3 = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(100, 100));
child->SetMasksToBounds(true);
grand_child1->test_properties()->position = gfx::PointF(5.f, 5.f);
grand_child1->SetBounds(gfx::Size(50, 50));
grand_child1->SetDrawsContent(true);
grand_child2->test_properties()->position = gfx::PointF(75.f, 75.f);
grand_child2->SetBounds(gfx::Size(50, 50));
grand_child2->SetDrawsContent(true);
grand_child3->test_properties()->position = gfx::PointF(125.f, 125.f);
grand_child3->SetBounds(gfx::Size(50, 50));
grand_child3->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
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);
root->SetBounds(gfx::Size(100, 100));
child->test_properties()->transform = child_scale_matrix;
child->SetBounds(gfx::Size(10, 10));
child->SetMasksToBounds(true);
grand_child->test_properties()->transform = grand_child_scale_matrix;
grand_child->SetBounds(gfx::Size(100, 100));
grand_child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
// The visible rect is expanded to integer coordinates.
EXPECT_EQ(gfx::Rect(41, 41), grand_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectWithClippingAndFilters) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip = AddChild<LayerImpl>(root);
LayerImpl* filter = AddChild<LayerImpl>(clip);
LayerImpl* filter_child = AddChild<LayerImpl>(filter);
root->SetBounds(gfx::Size(100, 100));
clip->SetBounds(gfx::Size(10, 10));
filter->test_properties()->force_render_surface = true;
filter_child->SetBounds(gfx::Size(2000, 2000));
filter_child->test_properties()->position = gfx::PointF(-50, -50);
filter_child->SetDrawsContent(true);
clip->SetMasksToBounds(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(50, 50, 10, 10), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), GetRenderSurface(filter)->content_rect());
FilterOperations blur_filter;
blur_filter.Append(FilterOperation::CreateBlurFilter(4.0f));
filter->test_properties()->filters = blur_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(38, 38, 34, 34), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(-12, -12, 34, 34),
GetRenderSurface(filter)->content_rect());
gfx::Transform vertical_flip;
vertical_flip.Scale(1, -1);
sk_sp<PaintFilter> flip_filter = sk_make_sp<MatrixPaintFilter>(
vertical_flip.matrix(), kLow_SkFilterQuality, nullptr);
FilterOperations reflection_filter;
reflection_filter.Append(
FilterOperation::CreateReferenceFilter(sk_make_sp<XfermodePaintFilter>(
SkBlendMode::kSrcOver, std::move(flip_filter), nullptr)));
filter->test_properties()->filters = reflection_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(49, 39, 12, 21), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(-1, -11, 12, 21),
GetRenderSurface(filter)->content_rect());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectWithScalingClippingAndFilters) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* scale = AddChild<LayerImpl>(root);
LayerImpl* clip = AddChild<LayerImpl>(scale);
LayerImpl* filter = AddChild<LayerImpl>(clip);
LayerImpl* filter_child = AddChild<LayerImpl>(filter);
root->SetBounds(gfx::Size(100, 100));
clip->SetBounds(gfx::Size(10, 10));
filter->test_properties()->force_render_surface = true;
filter_child->SetBounds(gfx::Size(2000, 2000));
filter_child->test_properties()->position = gfx::PointF(-50, -50);
filter_child->SetDrawsContent(true);
clip->SetMasksToBounds(true);
gfx::Transform scale_transform;
scale_transform.Scale(3, 3);
scale->test_properties()->transform = scale_transform;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(50, 50, 10, 10), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(30, 30), GetRenderSurface(filter)->content_rect());
FilterOperations blur_filter;
blur_filter.Append(FilterOperation::CreateBlurFilter(4.0f));
filter->test_properties()->filters = blur_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(38, 38, 34, 34), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(-36, -36, 102, 102),
GetRenderSurface(filter)->content_rect());
gfx::Transform vertical_flip;
vertical_flip.Scale(1, -1);
sk_sp<PaintFilter> flip_filter = sk_make_sp<MatrixPaintFilter>(
vertical_flip.matrix(), kLow_SkFilterQuality, nullptr);
FilterOperations reflection_filter;
reflection_filter.Append(
FilterOperation::CreateReferenceFilter(sk_make_sp<XfermodePaintFilter>(
SkBlendMode::kSrcOver, std::move(flip_filter), nullptr)));
filter->test_properties()->filters = reflection_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(49, 39, 12, 21), filter_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(-1, -31, 32, 61),
GetRenderSurface(filter)->content_rect());
}
TEST_F(LayerTreeHostCommonTest, ClipRectWithClipParentAndFilters) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip = AddChild<LayerImpl>(root);
LayerImpl* filter = AddChild<LayerImpl>(clip);
LayerImpl* filter_child_1 = AddChild<LayerImpl>(filter);
LayerImpl* filter_child_2 = AddChild<LayerImpl>(filter);
root->SetBounds(gfx::Size(100, 100));
clip->SetBounds(gfx::Size(10, 10));
filter->test_properties()->force_render_surface = true;
filter_child_1->SetBounds(gfx::Size(20, 20));
filter_child_1->SetDrawsContent(true);
filter_child_2->SetBounds(gfx::Size(20, 20));
filter_child_2->SetDrawsContent(true);
root->SetMasksToBounds(true);
clip->SetMasksToBounds(true);
root->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
root->test_properties()->clip_children->insert(filter_child_1);
filter_child_1->test_properties()->clip_parent = root;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(filter_child_1->is_clipped());
EXPECT_TRUE(filter_child_2->is_clipped());
EXPECT_EQ(gfx::Rect(100, 100), filter_child_1->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), filter_child_2->clip_rect());
FilterOperations blur_filter;
blur_filter.Append(FilterOperation::CreateBlurFilter(4.0f));
filter->test_properties()->filters = blur_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(filter_child_1->is_clipped());
EXPECT_TRUE(filter_child_2->is_clipped());
EXPECT_EQ(gfx::Rect(100, 100), filter_child_1->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), filter_child_2->clip_rect());
}
TEST_F(LayerTreeHostCommonTest, ClipRectWithClippedDescendantOfFilter) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* filter = AddChild<LayerImpl>(root);
LayerImpl* clip = AddChild<LayerImpl>(filter);
LayerImpl* filter_grand_child = AddChild<LayerImpl>(clip);
root->SetBounds(gfx::Size(100, 100));
filter->test_properties()->force_render_surface = true;
clip->SetBounds(gfx::Size(10, 10));
clip->SetMasksToBounds(true);
filter_grand_child->SetBounds(gfx::Size(20, 20));
filter_grand_child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(filter_grand_child->is_clipped());
EXPECT_EQ(gfx::Rect(10, 10), filter_grand_child->clip_rect());
FilterOperations blur_filter;
blur_filter.Append(FilterOperation::CreateBlurFilter(4.0f));
filter->test_properties()->filters = blur_filter;
host_impl()->active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(filter_grand_child->is_clipped());
EXPECT_EQ(gfx::Rect(10, 10), filter_grand_child->clip_rect());
}
TEST_F(LayerTreeHostCommonTest,
DrawableAndVisibleContentRectsForLayersInUnclippedRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(render_surface);
LayerImpl* child2 = AddChild<LayerImpl>(render_surface);
LayerImpl* child3 = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(100, 100));
render_surface->SetBounds(gfx::Size(3, 4));
render_surface->test_properties()->force_render_surface = true;
child1->test_properties()->position = gfx::PointF(5.f, 5.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child2->test_properties()->position = gfx::PointF(75.f, 75.f);
child2->SetBounds(gfx::Size(50, 50));
child2->SetDrawsContent(true);
child3->test_properties()->position = gfx::PointF(125.f, 125.f);
child3->SetBounds(gfx::Size(50, 50));
child3->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(render_surface));
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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),
GetRenderSurface(render_surface)->DrawableContentRect());
// All layers that draw content into the unclipped surface are also unclipped.
// Only the viewport clip should apply
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_EQ(gfx::Rect(0, 0, 0, 0), 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,
VisibleContentRectsForClippedSurfaceWithEmptyClip) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child1 = AddChild<LayerImpl>(root);
LayerImpl* child2 = AddChild<LayerImpl>(root);
LayerImpl* child3 = AddChild<LayerImpl>(root);
root->SetBounds(gfx::Size(100, 100));
child1->test_properties()->position = gfx::PointF(5.f, 5.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child2->test_properties()->position = gfx::PointF(75.f, 75.f);
child2->SetBounds(gfx::Size(50, 50));
child2->SetDrawsContent(true);
child3->test_properties()->position = gfx::PointF(125.f, 125.f);
child3->SetBounds(gfx::Size(50, 50));
child3->SetDrawsContent(true);
RenderSurfaceList render_surface_list_impl;
// Now set the root render surface an empty clip.
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, gfx::Size(), &render_surface_list_impl);
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
ASSERT_TRUE(GetRenderSurface(root));
EXPECT_FALSE(root->is_clipped());
gfx::Rect empty;
EXPECT_EQ(empty, GetRenderSurface(root)->clip_rect());
EXPECT_TRUE(GetRenderSurface(root)->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_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
root->SetBounds(gfx::Size(100, 100));
child->test_properties()->position = gfx::PointF(5.f, 5.f);
child->SetBounds(gfx::Size(50, 50));
child->SetDrawsContent(true);
// Case 1: a truly degenerate matrix
gfx::Transform uninvertible_matrix(0.0, 0.0, 0.0, 0.0, 0.0, 0.0);
ASSERT_FALSE(uninvertible_matrix.IsInvertible());
child->test_properties()->transform = uninvertible_matrix;
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());
child->test_properties()->transform = uninvertible_matrix;
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());
child->test_properties()->transform = uninvertible_matrix;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(child->visible_layer_rect().IsEmpty());
EXPECT_TRUE(child->drawable_content_rect().IsEmpty());
}
TEST_F(LayerTreeHostCommonTest,
VisibleContentRectForLayerWithUninvertibleDrawTransform) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* grand_child = AddChild<LayerImpl>(child);
gfx::Transform perspective;
perspective.ApplyPerspectiveDepth(SkDoubleToMScalar(1e-12));
gfx::Transform rotation;
rotation.RotateAboutYAxis(45.0);
root->SetBounds(gfx::Size(100, 100));
child->test_properties()->transform = perspective;
child->test_properties()->position = gfx::PointF(10.f, 10.f);
child->SetBounds(gfx::Size(100, 100));
child->SetDrawsContent(true);
child->test_properties()->sorting_context_id = 1;
child->test_properties()->should_flatten_transform = false;
grand_child->test_properties()->transform = rotation;
grand_child->SetBounds(gfx::Size(100, 100));
grand_child->SetDrawsContent(true);
child->test_properties()->sorting_context_id = 1;
grand_child->test_properties()->should_flatten_transform = false;
grand_child->test_properties()->sorting_context_id = 1;
ExecuteCalculateDrawProperties(root);
// Though all layers have invertible transforms, matrix multiplication using
// floating-point math makes the draw transform uninvertible.
EXPECT_FALSE(root->layer_tree_impl()
->property_trees()
->transform_tree.Node(grand_child->transform_tree_index())
->ancestors_are_invertible);
// CalcDrawProps skips a subtree when a layer's screen space transform is
// uninvertible
EXPECT_EQ(gfx::Rect(), grand_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, OcclusionBySiblingOfTarget) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
std::unique_ptr<LayerTreeFrameSink> layer_tree_frame_sink =
FakeLayerTreeFrameSink::Create3d();
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.active_tree(), 2);
std::unique_ptr<TextureLayerImpl> surface =
TextureLayerImpl::Create(host_impl.active_tree(), 3);
std::unique_ptr<TextureLayerImpl> surface_child =
TextureLayerImpl::Create(host_impl.active_tree(), 4);
std::unique_ptr<TextureLayerImpl> surface_sibling =
TextureLayerImpl::Create(host_impl.active_tree(), 5);
std::unique_ptr<TextureLayerImpl> surface_child_mask =
TextureLayerImpl::Create(host_impl.active_tree(), 6);
surface->SetDrawsContent(true);
surface_child->SetDrawsContent(true);
surface_sibling->SetDrawsContent(true);
surface_child_mask->SetDrawsContent(true);
surface->SetContentsOpaque(true);
surface_child->SetContentsOpaque(true);
surface_sibling->SetContentsOpaque(true);
surface_child_mask->SetContentsOpaque(true);
gfx::Transform translate;
translate.Translate(20.f, 20.f);
root->SetBounds(gfx::Size(1000, 1000));
child->SetBounds(gfx::Size(300, 300));
surface->test_properties()->transform = translate;
surface->SetBounds(gfx::Size(300, 300));
surface->test_properties()->force_render_surface = true;
surface_child->SetBounds(gfx::Size(300, 300));
surface_child->test_properties()->force_render_surface = true;
surface_sibling->SetBounds(gfx::Size(200, 200));
LayerImpl* surface_ptr = surface.get();
LayerImpl* surface_child_ptr = surface_child.get();
LayerImpl* surface_child_mask_ptr = surface_child_mask.get();
host_impl.active_tree()->SetDeviceViewportSize(root->bounds());
surface_child->test_properties()->SetMaskLayer(std::move(surface_child_mask));
surface->test_properties()->AddChild(std::move(surface_child));
child->test_properties()->AddChild(std::move(surface));
child->test_properties()->AddChild(std::move(surface_sibling));
root->test_properties()->AddChild(std::move(child));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.SetVisible(true);
host_impl.InitializeFrameSink(layer_tree_frame_sink.get());
host_impl.active_tree()->BuildLayerListAndPropertyTreesForTesting();
host_impl.active_tree()->UpdateDrawProperties();
EXPECT_TRANSFORMATION_MATRIX_EQ(
GetRenderSurface(surface_ptr)->draw_transform(), translate);
// surface_sibling draws into the root render surface and occludes
// surface_child's contents.
Occlusion actual_occlusion =
GetRenderSurface(surface_child_ptr)->occlusion_in_content_space();
Occlusion expected_occlusion(translate, SimpleEnclosedRegion(gfx::Rect()),
SimpleEnclosedRegion(gfx::Rect(200, 200)));
EXPECT_TRUE(expected_occlusion.IsEqual(actual_occlusion));
// Mask layer should have the same occlusion.
actual_occlusion =
surface_child_mask_ptr->draw_properties().occlusion_in_content_space;
EXPECT_TRUE(expected_occlusion.IsEqual(actual_occlusion));
}
TEST_F(LayerTreeHostCommonTest, TextureLayerSnapping) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
std::unique_ptr<LayerTreeFrameSink> layer_tree_frame_sink =
FakeLayerTreeFrameSink::Create3d();
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<TextureLayerImpl> child =
TextureLayerImpl::Create(host_impl.active_tree(), 2);
LayerImpl* child_ptr = child.get();
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(100, 100));
child->SetDrawsContent(true);
gfx::Transform fractional_translate;
fractional_translate.Translate(10.5f, 20.3f);
child->test_properties()->transform = fractional_translate;
host_impl.active_tree()->SetDeviceViewportSize(root->bounds());
root->test_properties()->AddChild(std::move(child));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.SetVisible(true);
host_impl.InitializeFrameSink(layer_tree_frame_sink.get());
host_impl.active_tree()->BuildLayerListAndPropertyTreesForTesting();
host_impl.active_tree()->UpdateDrawProperties();
EXPECT_NE(child_ptr->ScreenSpaceTransform(), fractional_translate);
fractional_translate.RoundTranslationComponents();
EXPECT_TRANSFORMATION_MATRIX_EQ(child_ptr->ScreenSpaceTransform(),
fractional_translate);
gfx::RectF layer_bounds_in_screen_space =
MathUtil::MapClippedRect(child_ptr->ScreenSpaceTransform(),
gfx::RectF(gfx::SizeF(child_ptr->bounds())));
EXPECT_EQ(layer_bounds_in_screen_space, gfx::RectF(11.f, 20.f, 100.f, 100.f));
}
TEST_F(LayerTreeHostCommonTest,
OcclusionForLayerWithUninvertibleDrawTransform) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
std::unique_ptr<LayerTreeFrameSink> layer_tree_frame_sink =
FakeLayerTreeFrameSink::Create3d();
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.active_tree(), 2);
std::unique_ptr<LayerImpl> grand_child =
LayerImpl::Create(host_impl.active_tree(), 3);
std::unique_ptr<LayerImpl> occluding_child =
LayerImpl::Create(host_impl.active_tree(), 4);
gfx::Transform perspective;
perspective.ApplyPerspectiveDepth(SkDoubleToMScalar(1e-12));
gfx::Transform rotation;
rotation.RotateAboutYAxis(45.0);
root->SetBounds(gfx::Size(1000, 1000));
child->test_properties()->transform = perspective;
child->test_properties()->position = gfx::PointF(10.f, 10.f);
child->SetBounds(gfx::Size(300, 300));
child->test_properties()->should_flatten_transform = false;
child->test_properties()->sorting_context_id = 1;
grand_child->test_properties()->transform = rotation;
grand_child->SetBounds(gfx::Size(200, 200));
grand_child->test_properties()->should_flatten_transform = false;
grand_child->test_properties()->sorting_context_id = 1;
occluding_child->SetBounds(gfx::Size(200, 200));
occluding_child->test_properties()->should_flatten_transform = false;
child->SetDrawsContent(true);
grand_child->SetDrawsContent(true);
occluding_child->SetDrawsContent(true);
occluding_child->SetContentsOpaque(true);
host_impl.active_tree()->SetDeviceViewportSize(root->bounds());
child->test_properties()->AddChild(std::move(grand_child));
root->test_properties()->AddChild(std::move(child));
root->test_properties()->AddChild(std::move(occluding_child));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.SetVisible(true);
host_impl.InitializeFrameSink(layer_tree_frame_sink.get());
host_impl.active_tree()->BuildLayerListAndPropertyTreesForTesting();
host_impl.active_tree()->UpdateDrawProperties();
LayerImpl* grand_child_ptr = host_impl.active_tree()
->root_layer_for_testing()
->test_properties()
->children[0]
->test_properties()
->children[0];
// Though all layers have invertible transforms, matrix multiplication using
// floating-point math makes the draw transform uninvertible.
EXPECT_FALSE(
host_impl.active_tree()
->property_trees()
->transform_tree.Node(grand_child_ptr->transform_tree_index())
->ancestors_are_invertible);
// Since |grand_child| has an uninvertible screen space transform, it is
// skipped so
// that we are not computing its occlusion_in_content_space.
gfx::Rect layer_bounds = gfx::Rect();
EXPECT_EQ(
layer_bounds,
grand_child_ptr->draw_properties()
.occlusion_in_content_space.GetUnoccludedContentRect(layer_bounds));
}
TEST_F(LayerTreeHostCommonTest,
DrawableAndVisibleContentRectsForLayersInClippedRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(render_surface);
LayerImpl* child2 = AddChild<LayerImpl>(render_surface);
LayerImpl* child3 = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(100, 100));
root->SetMasksToBounds(true);
render_surface->SetBounds(gfx::Size(3, 4));
render_surface->test_properties()->force_render_surface = true;
child1->test_properties()->position = gfx::PointF(5.f, 5.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child2->test_properties()->position = gfx::PointF(75.f, 75.f);
child2->SetBounds(gfx::Size(50, 50));
child2->SetDrawsContent(true);
child3->test_properties()->position = gfx::PointF(125.f, 125.f);
child3->SetBounds(gfx::Size(50, 50));
child3->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(render_surface));
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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),
GetRenderSurface(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_for_testing();
LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>();
LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1);
LayerImpl* child1 = AddChild<LayerImpl>(render_surface2);
LayerImpl* child2 = AddChild<LayerImpl>(render_surface2);
LayerImpl* child3 = AddChild<LayerImpl>(render_surface2);
root->SetBounds(gfx::Size(100, 100));
root->SetMasksToBounds(true);
render_surface1->SetBounds(gfx::Size(3, 4));
render_surface1->test_properties()->force_render_surface = true;
render_surface2->SetBounds(gfx::Size(7, 13));
render_surface2->test_properties()->force_render_surface = true;
child1->test_properties()->position = gfx::PointF(5.f, 5.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child2->test_properties()->position = gfx::PointF(75.f, 75.f);
child2->SetBounds(gfx::Size(50, 50));
child2->SetDrawsContent(true);
child3->test_properties()->position = gfx::PointF(125.f, 125.f);
child3->SetBounds(gfx::Size(50, 50));
child3->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(render_surface1));
ASSERT_TRUE(GetRenderSurface(render_surface2));
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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),
GetRenderSurface(render_surface1)->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),
GetRenderSurface(render_surface2)->DrawableContentRect());
// All layers that draw content into render_surface2 think they are unclipped
// by the surface. So, only the viewport clip applies.
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_EQ(gfx::Rect(0, 0, 0, 0), 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_for_testing();
LayerImpl* render_surface1 = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(render_surface1);
LayerImpl* child2 = AddChild<LayerImpl>(child1);
LayerImpl* render_surface2 = AddChild<LayerImpl>(child2);
root->SetBounds(gfx::Size(100, 100));
render_surface1->SetBounds(gfx::Size(100, 100));
render_surface1->test_properties()->force_render_surface = true;
child1->SetBounds(gfx::Size(500, 500));
child1->SetDrawsContent(true);
child2->SetBounds(gfx::Size(700, 700));
child2->SetDrawsContent(true);
render_surface2->SetBounds(gfx::Size(1000, 1000));
render_surface2->test_properties()->force_render_surface = true;
render_surface2->SetDrawsContent(true);
child1->SetMasksToBounds(true);
child2->SetMasksToBounds(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(100, 100), child1->visible_layer_rect());
EXPECT_EQ(gfx::Rect(100, 100), render_surface2->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest,
VisibleRectsWhenClipChildIsBetweenTwoRenderSurfaces) {
LayerImpl* root = root_layer_for_testing();
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);
root->SetBounds(gfx::Size(100, 100));
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->SetMasksToBounds(true);
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
render_surface1->SetBounds(gfx::Size(20, 20));
render_surface1->SetMasksToBounds(true);
render_surface1->SetDrawsContent(true);
render_surface1->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(60, 60));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
render_surface2->SetBounds(gfx::Size(60, 60));
render_surface2->SetDrawsContent(true);
render_surface2->test_properties()->force_render_surface = 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_for_testing();
LayerImpl* clip_parent = AddChildToRoot<LayerImpl>();
LayerImpl* clip_layer = AddChild<LayerImpl>(clip_parent);
LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_layer);
LayerImpl* clip_child = AddChild<LayerImpl>(render_surface1);
LayerImpl* render_surface2 = AddChild<LayerImpl>(clip_child);
root->SetBounds(gfx::Size(100, 100));
clip_parent->test_properties()->position = gfx::PointF(2.f, 2.f);
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
clip_layer->SetBounds(gfx::Size(50, 50));
render_surface1->SetBounds(gfx::Size(20, 20));
render_surface1->SetDrawsContent(true);
render_surface1->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(60, 60));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
render_surface2->SetBounds(gfx::Size(60, 60));
render_surface2->SetDrawsContent(true);
render_surface2->test_properties()->force_render_surface = true;
clip_parent->SetMasksToBounds(true);
clip_layer->SetMasksToBounds(true);
render_surface1->SetMasksToBounds(true);
float device_scale_factor = 1.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_EQ(gfx::Rect(50, 50), GetRenderSurface(render_surface2)->clip_rect());
device_scale_factor = 2.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_EQ(gfx::Rect(100, 100),
GetRenderSurface(render_surface2)->clip_rect());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceContentRectWhenLayerNotDrawn) {
// Test that only drawn layers contribute to render surface content rect.
LayerImpl* root = root_layer_for_testing();
LayerImpl* surface = AddChildToRoot<LayerImpl>();
LayerImpl* test_layer = AddChild<LayerImpl>(surface);
root->SetBounds(gfx::Size(200, 200));
surface->SetBounds(gfx::Size(100, 100));
surface->SetDrawsContent(true);
surface->test_properties()->force_render_surface = true;
test_layer->SetBounds(gfx::Size(150, 150));
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(100, 100), GetRenderSurface(surface)->content_rect());
test_layer->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(150, 150), GetRenderSurface(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_for_testing();
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);
root->SetBounds(gfx::Size(100, 100));
unclipped_surface->SetBounds(gfx::Size(30, 30));
unclipped_surface->SetDrawsContent(true);
unclipped_surface->test_properties()->force_render_surface = true;
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
unclipped_desc_surface->SetBounds(gfx::Size(20, 20));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(60, 60));
clip_child->test_properties()->clip_parent = clip_parent;
clipped_surface->SetBounds(gfx::Size(60, 60));
clipped_surface->SetDrawsContent(true);
clipped_surface->test_properties()->force_render_surface = 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_for_testing();
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);
root->SetBounds(gfx::Size(10, 10));
unclipped_surface->SetBounds(gfx::Size(50, 50));
unclipped_surface->SetDrawsContent(true);
unclipped_surface->test_properties()->force_render_surface = true;
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
unclipped_desc_surface->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(100, 100));
clip_child->test_properties()->clip_parent = clip_parent;
clipped_surface->SetBounds(gfx::Size(50, 50));
clipped_surface->SetDrawsContent(true);
clipped_surface->test_properties()->force_render_surface = true;
clip_parent->SetMasksToBounds(true);
unclipped_desc_surface->SetMasksToBounds(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(10, 10), unclipped_surface->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), 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_for_testing();
LayerImpl* render_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(render_surface);
gfx::Transform child_rotation;
child_rotation.Rotate(45.0);
root->SetBounds(gfx::Size(100, 100));
render_surface->SetBounds(gfx::Size(3, 4));
render_surface->test_properties()->force_render_surface = true;
child1->test_properties()->transform = child_rotation;
child1->test_properties()->position = gfx::PointF(25.f, 25.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child1->test_properties()->transform_origin = gfx::Point3F(25.f, 25.f, 0.f);
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(render_surface));
EXPECT_EQ(gfx::RectF(100.f, 100.f),
GetRenderSurface(root)->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),
GetRenderSurface(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_for_testing();
LayerImpl* render_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(render_surface);
gfx::Transform child_rotation;
child_rotation.Rotate(45.0);
root->SetBounds(gfx::Size(50, 50));
root->SetMasksToBounds(true);
render_surface->SetBounds(gfx::Size(3, 4));
render_surface->test_properties()->force_render_surface = true;
child1->test_properties()->position = gfx::PointF(25.f, 25.f);
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child1->test_properties()->transform = child_rotation;
child1->test_properties()->transform_origin = gfx::Point3F(25.f, 25.f, 0.f);
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(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,
GetRenderSurface(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.
EXPECT_EQ(gfx::Rect(0, 0, 25, 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_for_testing();
FakePictureLayerImpl* render_surface1 =
AddChildToRoot<FakePictureLayerImpl>();
FakePictureLayerImpl* render_surface2 =
AddChild<FakePictureLayerImpl>(render_surface1);
FakePictureLayerImpl* child1 =
AddChild<FakePictureLayerImpl>(render_surface2);
FakePictureLayerImpl* child2 =
AddChild<FakePictureLayerImpl>(render_surface2);
FakePictureLayerImpl* child3 =
AddChild<FakePictureLayerImpl>(render_surface2);
root->SetBounds(gfx::Size(100, 100));
root->SetMasksToBounds(true);
render_surface1->SetBounds(gfx::Size(3, 4));
render_surface1->test_properties()->position = gfx::PointF(5.f, 5.f);
render_surface1->SetDrawsContent(true);
render_surface1->test_properties()->force_render_surface = true;
render_surface2->SetBounds(gfx::Size(7, 13));
render_surface2->test_properties()->position = gfx::PointF(5.f, 5.f);
render_surface2->SetDrawsContent(true);
render_surface2->test_properties()->force_render_surface = true;
child1->SetBounds(gfx::Size(50, 50));
child1->test_properties()->position = gfx::PointF(5.f, 5.f);
child1->SetDrawsContent(true);
child2->SetBounds(gfx::Size(50, 50));
child2->test_properties()->position = gfx::PointF(75.f, 75.f);
child2->SetDrawsContent(true);
child3->SetBounds(gfx::Size(50, 50));
child3->test_properties()->position = gfx::PointF(125.f, 125.f);
child3->SetDrawsContent(true);
float device_scale_factor = 2.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
ASSERT_TRUE(GetRenderSurface(render_surface1));
ASSERT_TRUE(GetRenderSurface(render_surface2));
// drawable_content_rects for all layers and surfaces are scaled by
// device_scale_factor.
EXPECT_EQ(gfx::RectF(200.f, 200.f),
GetRenderSurface(root)->DrawableContentRect());
EXPECT_EQ(gfx::RectF(10.f, 10.f, 190.f, 190.f),
GetRenderSurface(render_surface1)->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),
GetRenderSurface(render_surface2)->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, 15, 15), child2->visible_layer_rect());
EXPECT_EQ(gfx::Rect(0, 0, 0, 0), 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.
LayerImpl* root = root_layer_for_testing();
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);
// Nothing is double-sided
front_facing_child->test_properties()->double_sided = false;
back_facing_child->test_properties()->double_sided = false;
front_facing_surface->test_properties()->double_sided = false;
back_facing_surface->test_properties()->double_sided = false;
front_facing_child_of_front_facing_surface->test_properties()->double_sided =
false;
back_facing_child_of_front_facing_surface->test_properties()->double_sided =
false;
front_facing_child_of_back_facing_surface->test_properties()->double_sided =
false;
back_facing_child_of_back_facing_surface->test_properties()->double_sided =
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);
gfx::Transform backface_matrix;
backface_matrix.Translate(50.0, 50.0);
backface_matrix.RotateAboutYAxis(180.0);
backface_matrix.Translate(-50.0, -50.0);
root->SetBounds(gfx::Size(100, 100));
front_facing_child->SetBounds(gfx::Size(100, 100));
back_facing_child->SetBounds(gfx::Size(100, 100));
front_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_child_of_front_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_child_of_front_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_child_of_back_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_child_of_back_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_surface->test_properties()->force_render_surface = true;
back_facing_surface->test_properties()->force_render_surface = true;
back_facing_child->test_properties()->transform = backface_matrix;
back_facing_surface->test_properties()->transform = backface_matrix;
back_facing_child_of_front_facing_surface->test_properties()->transform =
backface_matrix;
back_facing_child_of_back_facing_surface->test_properties()->transform =
backface_matrix;
// Note: No layers preserve 3d. According to current W3C CSS gfx::Transforms
// spec, these layers should blindly use their own local transforms to
// determine back-face culling.
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
// Verify which render surfaces were created.
EXPECT_EQ(GetRenderSurface(front_facing_child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(back_facing_child), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(front_facing_surface), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(back_facing_surface), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(back_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(front_facing_child_of_front_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(back_facing_child_of_front_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(front_facing_child_of_back_facing_surface),
GetRenderSurface(back_facing_surface));
EXPECT_EQ(GetRenderSurface(back_facing_child_of_back_facing_surface),
GetRenderSurface(back_facing_surface));
EXPECT_EQ(3u, update_layer_impl_list()->size());
EXPECT_TRUE(UpdateLayerImplListContains(front_facing_child->id()));
EXPECT_TRUE(UpdateLayerImplListContains(front_facing_surface->id()));
EXPECT_TRUE(UpdateLayerImplListContains(
front_facing_child_of_front_facing_surface->id()));
}
TEST_F(LayerTreeHostCommonTest, BackFaceCullingWithPreserves3d) {
// Verify the behavior of back-face culling when preserves-3d transform style
// is used.
LayerImpl* root = root_layer_for_testing();
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 mask layer.
LayerImpl* dummy_mask_layer1 = AddMaskLayer<LayerImpl>(front_facing_surface);
LayerImpl* dummy_mask_layer2 = AddMaskLayer<LayerImpl>(back_facing_surface);
// Nothing is double-sided
front_facing_child->test_properties()->double_sided = false;
back_facing_child->test_properties()->double_sided = false;
front_facing_surface->test_properties()->double_sided = false;
back_facing_surface->test_properties()->double_sided = false;
front_facing_child_of_front_facing_surface->test_properties()->double_sided =
false;
back_facing_child_of_front_facing_surface->test_properties()->double_sided =
false;
front_facing_child_of_back_facing_surface->test_properties()->double_sided =
false;
back_facing_child_of_back_facing_surface->test_properties()->double_sided =
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_mask_layer1->SetDrawsContent(true);
dummy_mask_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);
root->SetBounds(gfx::Size(100, 100));
front_facing_child->SetBounds(gfx::Size(100, 100));
back_facing_child->SetBounds(gfx::Size(100, 100));
front_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_child_of_front_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_child_of_front_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_child_of_back_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_child_of_back_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_child->test_properties()->transform = backface_matrix;
back_facing_surface->test_properties()->transform = backface_matrix;
back_facing_child_of_front_facing_surface->test_properties()->transform =
backface_matrix;
back_facing_child_of_back_facing_surface->test_properties()->transform =
backface_matrix;
// 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.
front_facing_surface->test_properties()->should_flatten_transform = false;
front_facing_surface->test_properties()->sorting_context_id = 1;
back_facing_surface->test_properties()->should_flatten_transform = false;
back_facing_surface->test_properties()->sorting_context_id = 1;
front_facing_child_of_front_facing_surface->test_properties()
->sorting_context_id = 1;
back_facing_child_of_front_facing_surface->test_properties()
->sorting_context_id = 1;
front_facing_child_of_back_facing_surface->test_properties()
->sorting_context_id = 1;
back_facing_child_of_back_facing_surface->test_properties()
->sorting_context_id = 1;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
// Verify which render surfaces were created and used.
EXPECT_EQ(GetRenderSurface(front_facing_child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(back_facing_child), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(front_facing_surface), GetRenderSurface(root));
// We expect that a has_render_surface was created but not used.
EXPECT_NE(GetRenderSurface(back_facing_surface), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(back_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(front_facing_child_of_front_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(back_facing_child_of_front_facing_surface),
GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(front_facing_child_of_back_facing_surface),
GetRenderSurface(back_facing_surface));
EXPECT_EQ(GetRenderSurface(back_facing_child_of_back_facing_surface),
GetRenderSurface(back_facing_surface));
EXPECT_EQ(3u, update_layer_impl_list()->size());
EXPECT_TRUE(UpdateLayerImplListContains(front_facing_child->id()));
EXPECT_TRUE(UpdateLayerImplListContains(front_facing_surface->id()));
EXPECT_TRUE(UpdateLayerImplListContains(
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.
//
// Even layers that are animating get culled if their back face is showing and
// they are not double sided.
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChildToRoot<LayerImpl>();
LayerImpl* animating_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child_of_animating_surface =
AddChild<LayerImpl>(animating_surface);
LayerImpl* animating_child = AddChildToRoot<LayerImpl>();
LayerImpl* child2 = AddChildToRoot<LayerImpl>();
// Nothing is double-sided
child->test_properties()->double_sided = false;
child2->test_properties()->double_sided = false;
animating_surface->test_properties()->double_sided = false;
child_of_animating_surface->test_properties()->double_sided = false;
animating_child->test_properties()->double_sided = false;
// Everything draws content.
child->SetDrawsContent(true);
child2->SetDrawsContent(true);
animating_surface->SetDrawsContent(true);
child_of_animating_surface->SetDrawsContent(true);
animating_child->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);
SetElementIdsForTesting();
// Animate the transform on the render surface.
AddAnimatedTransformToElementWithAnimation(animating_surface->element_id(),
timeline_impl(), 10.0, 30, 0);
// This is just an animating layer, not a surface.
AddAnimatedTransformToElementWithAnimation(animating_child->element_id(),
timeline_impl(), 10.0, 30, 0);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(100, 100));
child->test_properties()->transform = backface_matrix;
animating_surface->SetBounds(gfx::Size(100, 100));
animating_surface->test_properties()->transform = backface_matrix;
animating_surface->test_properties()->force_render_surface = true;
child_of_animating_surface->SetBounds(gfx::Size(100, 100));
child_of_animating_surface->test_properties()->transform = backface_matrix;
animating_child->SetBounds(gfx::Size(100, 100));
animating_child->test_properties()->transform = backface_matrix;
child2->SetBounds(gfx::Size(100, 100));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
EXPECT_EQ(GetRenderSurface(child), GetRenderSurface(root));
EXPECT_TRUE(GetRenderSurface(animating_surface));
EXPECT_EQ(GetRenderSurface(child_of_animating_surface),
GetRenderSurface(animating_surface));
EXPECT_EQ(GetRenderSurface(animating_child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(child2), GetRenderSurface(root));
EXPECT_EQ(1u, update_layer_impl_list()->size());
// The back facing layers are culled from the layer list, and have an empty
// visible rect.
EXPECT_TRUE(UpdateLayerImplListContains(child2->id()));
EXPECT_TRUE(child->visible_layer_rect().IsEmpty());
EXPECT_TRUE(animating_surface->visible_layer_rect().IsEmpty());
EXPECT_TRUE(child_of_animating_surface->visible_layer_rect().IsEmpty());
EXPECT_TRUE(animating_child->visible_layer_rect().IsEmpty());
EXPECT_EQ(gfx::Rect(100, 100), child2->visible_layer_rect());
}
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.
LayerImpl* root = root_layer_for_testing();
LayerImpl* front_facing_surface = AddChildToRoot<LayerImpl>();
LayerImpl* back_facing_surface = AddChildToRoot<LayerImpl>();
LayerImpl* child1 = AddChild<LayerImpl>(front_facing_surface);
LayerImpl* child2 = AddChild<LayerImpl>(back_facing_surface);
// RenderSurfaces are not double-sided
front_facing_surface->test_properties()->double_sided = false;
back_facing_surface->test_properties()->double_sided = false;
// Everything draws content.
front_facing_surface->SetDrawsContent(true);
back_facing_surface->SetDrawsContent(true);
child1->SetDrawsContent(true);
child2->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);
root->SetBounds(gfx::Size(100, 100));
front_facing_surface->SetBounds(gfx::Size(100, 100));
front_facing_surface->test_properties()->force_render_surface = true;
back_facing_surface->SetBounds(gfx::Size(100, 100));
back_facing_surface->test_properties()->transform = backface_matrix;
back_facing_surface->test_properties()->force_render_surface = true;
child1->SetBounds(gfx::Size(100, 100));
child2->SetBounds(gfx::Size(100, 100));
front_facing_surface->test_properties()->sorting_context_id = 1;
back_facing_surface->test_properties()->sorting_context_id = 1;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
// Verify which render surfaces were created and used.
EXPECT_TRUE(GetRenderSurface(front_facing_surface));
// We expect the render surface to have been created, but remain unused.
EXPECT_NE(GetRenderSurface(back_facing_surface), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(child1), GetRenderSurface(front_facing_surface));
EXPECT_EQ(GetRenderSurface(child2), GetRenderSurface(back_facing_surface));
EXPECT_EQ(2u, update_layer_impl_list()->size());
EXPECT_TRUE(UpdateLayerImplListContains(front_facing_surface->id()));
EXPECT_TRUE(UpdateLayerImplListContains(child1->id()));
}
TEST_F(LayerTreeHostCommonScalingTest, LayerTransformsInHighDPI) {
// Verify draw and screen space transforms of layers not in a surface.
LayerImpl* root = root_layer_for_testing();
root->SetBounds(gfx::Size(100, 100));
root->SetDrawsContent(true);
LayerImpl* child = AddChildToRoot<LayerImpl>();
child->test_properties()->position = gfx::PointF(2.f, 2.f);
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
LayerImpl* child2 = AddChildToRoot<LayerImpl>();
child2->test_properties()->position = gfx::PointF(2.f, 2.f);
child2->SetBounds(gfx::Size(5, 5));
child2->SetDrawsContent(true);
float device_scale_factor = 2.5f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_FLOAT_EQ(device_scale_factor, root->GetIdealContentsScale());
EXPECT_FLOAT_EQ(device_scale_factor, child->GetIdealContentsScale());
EXPECT_FLOAT_EQ(device_scale_factor, child2->GetIdealContentsScale());
EXPECT_EQ(1u, render_surface_list_impl()->size());
// Verify root transforms
gfx::Transform expected_root_transform;
expected_root_transform.Scale(device_scale_factor, device_scale_factor);
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_root_transform,
root->ScreenSpaceTransform());
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_root_transform,
root->DrawTransform());
// Verify results of transformed root rects
gfx::RectF root_bounds(gfx::SizeF(root->bounds()));
gfx::RectF root_draw_rect =
MathUtil::MapClippedRect(root->DrawTransform(), root_bounds);
gfx::RectF root_screen_space_rect =
MathUtil::MapClippedRect(root->ScreenSpaceTransform(), root_bounds);
gfx::RectF expected_root_draw_rect(gfx::SizeF(root->bounds()));
expected_root_draw_rect.Scale(device_scale_factor);
EXPECT_FLOAT_RECT_EQ(expected_root_draw_rect, root_draw_rect);
EXPECT_FLOAT_RECT_EQ(expected_root_draw_rect, root_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->test_properties()->position.x(),
child->test_properties()->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->test_properties()->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 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_for_testing();
root->SetBounds(gfx::Size(100, 100));
LayerImpl* page_scale = AddChildToRoot<LayerImpl>();
page_scale->SetBounds(gfx::Size(100, 100));
LayerImpl* parent = AddChild<LayerImpl>(page_scale);
parent->SetBounds(gfx::Size(100, 100));
parent->SetDrawsContent(true);
LayerImpl* perspective_surface = AddChild<LayerImpl>(parent);
perspective_surface->test_properties()->position = gfx::PointF(2.f, 2.f);
perspective_surface->SetBounds(gfx::Size(10, 10));
perspective_surface->test_properties()->transform =
perspective_matrix * scale_small_matrix;
perspective_surface->SetDrawsContent(true);
perspective_surface->test_properties()->force_render_surface = true;
LayerImpl* scale_surface = AddChild<LayerImpl>(parent);
scale_surface->test_properties()->position = gfx::PointF(2.f, 2.f);
scale_surface->SetBounds(gfx::Size(10, 10));
scale_surface->test_properties()->transform = scale_small_matrix;
scale_surface->SetDrawsContent(true);
scale_surface->test_properties()->force_render_surface = true;
float device_scale_factor = 2.5f;
float page_scale_factor = 3.f;
LayerTreeImpl::ViewportLayerIds viewport_ids;
viewport_ids.page_scale = page_scale->id();
root->layer_tree_impl()->SetViewportLayersFromIds(viewport_ids);
root->layer_tree_impl()->BuildLayerListAndPropertyTreesForTesting();
root->layer_tree_impl()->SetPageScaleOnActiveTree(page_scale_factor);
ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor,
root, nullptr, nullptr);
EXPECT_FLOAT_EQ(device_scale_factor * page_scale_factor,
parent->GetIdealContentsScale());
EXPECT_FLOAT_EQ(device_scale_factor * page_scale_factor,
perspective_surface->GetIdealContentsScale());
// 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_FLOAT_EQ(max_2d_scale, scale_surface->GetIdealContentsScale());
// 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_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->test_properties()->position.x(),
device_scale_factor * page_scale_factor *
perspective_surface->test_properties()->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,
GetRenderSurface(perspective_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_for_testing();
root->SetBounds(gfx::Size(100, 100));
LayerImpl* parent = AddChildToRoot<LayerImpl>();
parent->SetBounds(gfx::Size(100, 100));
parent->test_properties()->transform = parent_scale_matrix;
parent->SetDrawsContent(true);
LayerImpl* child_scale = AddChild<LayerImpl>(parent);
child_scale->test_properties()->position = gfx::PointF(2.f, 2.f);
child_scale->SetBounds(gfx::Size(10, 10));
child_scale->test_properties()->transform = child_scale_matrix;
child_scale->SetDrawsContent(true);
float device_scale_factor = 2.5f;
float page_scale_factor = 0.01f;
{
ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor,
root, nullptr, nullptr);
// 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_FLOAT_EQ(expected_ideal_scale, parent->GetIdealContentsScale());
expected_ideal_scale = device_scale_factor * page_scale_factor *
initial_parent_scale * initial_child_scale;
EXPECT_LT(expected_ideal_scale, 1.f);
EXPECT_FLOAT_EQ(expected_ideal_scale, child_scale->GetIdealContentsScale());
}
}
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_for_testing();
root->SetBounds(gfx::Size(100, 100));
LayerImpl* parent = AddChildToRoot<LayerImpl>();
parent->SetBounds(gfx::Size(100, 100));
parent->test_properties()->transform = parent_scale_matrix;
parent->SetDrawsContent(true);
LayerImpl* child_scale = AddChild<LayerImpl>(parent);
child_scale->SetBounds(gfx::Size(10, 10));
child_scale->test_properties()->position = gfx::PointF(2.f, 2.f);
child_scale->test_properties()->transform = child_scale_matrix;
child_scale->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_FLOAT_EQ(initial_parent_scale, parent->GetIdealContentsScale());
// Animating layers compute ideal scale in the same way as when
// they are static.
EXPECT_FLOAT_EQ(initial_child_scale * initial_parent_scale,
child_scale->GetIdealContentsScale());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceTransformsInHighDPI) {
LayerImpl* parent = root_layer_for_testing();
parent->SetBounds(gfx::Size(30, 30));
parent->SetDrawsContent(true);
parent->test_properties()->should_flatten_transform = false;
parent->test_properties()->sorting_context_id = 1;
LayerImpl* child = AddChildToRoot<LayerImpl>();
child->SetBounds(gfx::Size(10, 10));
child->test_properties()->position = gfx::PointF(2.f, 2.f);
child->SetDrawsContent(true);
child->test_properties()->force_render_surface = true;
// 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->SetBounds(gfx::Size(10, 10));
duplicate_child_non_owner->SetDrawsContent(true);
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 of force_render_surface).
EXPECT_EQ(2u, render_surface_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->test_properties()->position.x(),
child->test_properties()->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(expected_duplicate_child_draw_transform,
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->test_properties()->position.x(),
device_scale_factor * child->test_properties()->position.y());
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_render_surface_draw_transform,
GetRenderSurface(child)->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,
GetRenderSurface(child)->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,
GetRenderSurface(child)->screen_space_transform());
}
TEST_F(LayerTreeHostCommonTest,
RenderSurfaceTransformsInHighDPIAccurateScaleZeroPosition) {
LayerImpl* parent = root_layer_for_testing();
parent->SetBounds(gfx::Size(33, 31));
parent->SetDrawsContent(true);
LayerImpl* child = AddChildToRoot<LayerImpl>();
child->SetBounds(gfx::Size(13, 11));
child->SetDrawsContent(true);
child->test_properties()->force_render_surface = true;
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 of force_render_surface).
EXPECT_EQ(2u, render_surface_list_impl()->size());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(child)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(gfx::Transform(),
GetRenderSurface(child)->draw_transform());
EXPECT_TRANSFORMATION_MATRIX_EQ(
gfx::Transform(), GetRenderSurface(child)->screen_space_transform());
}
TEST_F(LayerTreeHostCommonTest, LayerSearch) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> child = Layer::Create();
scoped_refptr<Layer> grand_child = Layer::Create();
FakeContentLayerClient client;
scoped_refptr<PictureLayer> mask_layer = PictureLayer::Create(&client);
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(), host()->LayerById(root->id()));
EXPECT_EQ(child.get(), host()->LayerById(child->id()));
EXPECT_EQ(grand_child.get(), host()->LayerById(grand_child->id()));
EXPECT_EQ(mask_layer.get(), host()->LayerById(mask_layer->id()));
EXPECT_FALSE(host()->LayerById(nonexistent_id));
}
TEST_F(LayerTreeHostCommonTest, TransparentChildRenderSurfaceCreation) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(10, 10));
child->test_properties()->opacity = 0.5f;
grand_child->SetBounds(gfx::Size(10, 10));
grand_child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(GetRenderSurface(child), GetRenderSurface(root));
}
TEST_F(LayerTreeHostCommonTest, OpacityAnimatingOnPendingTree) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(host()->GetSettings(), &task_runner_provider,
&task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(100, 100));
root->SetDrawsContent(true);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.pending_tree(), 2);
child->SetBounds(gfx::Size(50, 50));
child->SetDrawsContent(true);
child->test_properties()->opacity = 0.0f;
const int child_id = child->id();
root->test_properties()->AddChild(std::move(child));
LayerImpl* root_layer = root.get();
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
host_impl.pending_tree()->BuildLayerListAndPropertyTreesForTesting();
// Add opacity animation.
scoped_refptr<AnimationTimeline> timeline =
AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl.animation_host()->AddAnimationTimeline(timeline);
ElementId child_element_id =
host_impl.pending_tree()->LayerById(child_id)->element_id();
AddOpacityTransitionToElementWithAnimation(child_element_id, timeline, 10.0,
0.0f, 1.0f, false);
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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_list.size());
ASSERT_EQ(2, GetRenderSurface(root_layer)->num_contributors());
// If the root itself is hidden, the child should not be drawn even if it has
// an animating opacity.
root_layer->test_properties()->opacity = 0.0f;
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
RenderSurfaceList render_surface_list2;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs2(
root_layer, root_layer->bounds(), &render_surface_list2);
inputs2.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs2);
LayerImpl* child_ptr = root_layer->layer_tree_impl()->LayerById(2);
EffectTree& tree =
root_layer->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = tree.Node(child_ptr->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
// A layer should be drawn and it should contribute to drawn surface when
// it has animating opacity even if it has opacity 0.
root_layer->test_properties()->opacity = 1.0f;
child_ptr->test_properties()->opacity = 0.0f;
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
RenderSurfaceList render_surface_list3;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs3(
root_layer, root_layer->bounds(), &render_surface_list3);
inputs3.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs3);
child_ptr = root_layer->layer_tree_impl()->LayerById(2);
tree = root_layer->layer_tree_impl()->property_trees()->effect_tree;
node = tree.Node(child_ptr->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
EXPECT_TRUE(tree.ContributesToDrawnSurface(child_ptr->effect_tree_index()));
// But if the opacity of the layer remains 0 after activation, it should not
// be drawn.
host_impl.ActivateSyncTree();
LayerImpl* active_root = host_impl.active_tree()->LayerById(root_layer->id());
LayerImpl* active_child = host_impl.active_tree()->LayerById(child_ptr->id());
EffectTree& active_effect_tree =
host_impl.active_tree()->property_trees()->effect_tree;
EXPECT_TRUE(active_effect_tree.needs_update());
ExecuteCalculateDrawProperties(active_root);
node = active_effect_tree.Node(active_child->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
EXPECT_FALSE(active_effect_tree.ContributesToDrawnSurface(
active_child->effect_tree_index()));
}
using LCDTextTestParam = std::tuple<bool, bool, bool>;
class LCDTextTest : public LayerTreeHostCommonTestBase,
public testing::TestWithParam<LCDTextTestParam> {
public:
LCDTextTest()
: LayerTreeHostCommonTestBase(LCDTextTestLayerTreeSettings()),
host_impl_(LCDTextTestLayerTreeSettings(),
&task_runner_provider_,
&task_graph_runner_) {}
scoped_refptr<AnimationTimeline> timeline() { return timeline_; }
protected:
LayerTreeSettings LCDTextTestLayerTreeSettings() {
LayerTreeSettings settings = VerifyTreeCalcsLayerTreeSettings();
can_use_lcd_text_ = std::get<0>(GetParam());
layers_always_allowed_lcd_text_ = std::get<1>(GetParam());
settings.can_use_lcd_text = can_use_lcd_text_;
settings.layers_always_allowed_lcd_text = layers_always_allowed_lcd_text_;
return settings;
}
void SetUp() override {
timeline_ =
AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl_.animation_host()->AddAnimationTimeline(timeline_);
std::unique_ptr<LayerImpl> root_ptr =
LayerImpl::Create(host_impl_.active_tree(), 1);
std::unique_ptr<LayerImpl> child_ptr =
LayerImpl::Create(host_impl_.active_tree(), 2);
std::unique_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_->test_properties()->AddChild(std::move(grand_child_ptr));
root_->test_properties()->AddChild(std::move(child_ptr));
host_impl_.active_tree()->SetRootLayerForTesting(std::move(root_ptr));
host_impl_.active_tree()->SetElementIdsForTesting();
root_->SetContentsOpaque(true);
child_->SetContentsOpaque(true);
grand_child_->SetContentsOpaque(true);
root_->SetDrawsContent(true);
child_->SetDrawsContent(true);
grand_child_->SetDrawsContent(true);
root_->SetBounds(gfx::Size(1, 1));
child_->SetBounds(gfx::Size(1, 1));
grand_child_->SetBounds(gfx::Size(1, 1));
child_->test_properties()->force_render_surface = std::get<2>(GetParam());
}
bool can_use_lcd_text_;
bool layers_always_allowed_lcd_text_;
FakeImplTaskRunnerProvider task_runner_provider_;
TestTaskGraphRunner task_graph_runner_;
FakeLayerTreeHostImpl host_impl_;
scoped_refptr<AnimationTimeline> timeline_;
LayerImpl* root_ = nullptr;
LayerImpl* child_ = nullptr;
LayerImpl* grand_child_ = nullptr;
};
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.
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
// Case 2: Integral translation.
gfx::Transform integral_translation;
integral_translation.Translate(1.0, 2.0);
child_->test_properties()->transform = integral_translation;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
// Case 3: Non-integral translation.
gfx::Transform non_integral_translation;
non_integral_translation.Translate(1.5, 2.5);
child_->test_properties()->transform = non_integral_translation;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
// Case 4: Rotation.
gfx::Transform rotation;
rotation.Rotate(10.0);
child_->test_properties()->transform = rotation;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
// Case 5: Scale.
gfx::Transform scale;
scale.Scale(2.0, 2.0);
child_->test_properties()->transform = scale;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
// Case 6: Skew.
gfx::Transform skew;
skew.Skew(10.0, 0.0);
child_->test_properties()->transform = skew;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
// Case 7: Translucent.
child_->test_properties()->transform = gfx::Transform();
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
child_->test_properties()->opacity = 0.5f;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
// Case 8: Sanity check: restore transform and opacity.
child_->test_properties()->transform = gfx::Transform();
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
child_->test_properties()->opacity = 1.f;
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
// Case 9: Non-opaque content.
child_->SetContentsOpaque(false);
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
// Case 10: Sanity check: restore content opaqueness.
child_->SetContentsOpaque(true);
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
}
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, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
// Add opacity animation.
child_->test_properties()->opacity = 0.9f;
child_->layer_tree_impl()->property_trees()->needs_rebuild = true;
SetElementIdsForTesting();
AddOpacityTransitionToElementWithAnimation(child_->element_id(), timeline(),
10.0, 0.9f, 0.1f, false);
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
// Text LCD should be adjusted while animation is active.
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, grand_child_->CanUseLCDText());
}
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, nullptr, nullptr, nullptr);
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
SetElementIdsForTesting();
// Mark contents non-opaque within the first animation frame.
child_->SetContentsOpaque(false);
AddOpacityTransitionToElementWithAnimation(child_->element_id(), timeline(),
10.0, 0.9f, 0.1f, false);
ExecuteCalculateDrawProperties(root_, 1.f, 1.f, nullptr, nullptr, nullptr);
// LCD text should be disabled for non-opaque layers even during animations.
EXPECT_EQ(expect_lcd_text, root_->CanUseLCDText());
EXPECT_EQ(expect_not_lcd_text, child_->CanUseLCDText());
EXPECT_EQ(expect_lcd_text, grand_child_->CanUseLCDText());
}
INSTANTIATE_TEST_SUITE_P(LayerTreeHostCommonTest,
LCDTextTest,
testing::Combine(testing::Bool(),
testing::Bool(),
testing::Bool()));
TEST_F(LayerTreeHostCommonTest, SubtreeHidden_SingleLayerImpl) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.pending_tree(), 2);
child->SetBounds(gfx::Size(40, 40));
child->SetDrawsContent(true);
LayerImpl* child_layer = child.get();
std::unique_ptr<LayerImpl> grand_child =
LayerImpl::Create(host_impl.pending_tree(), 3);
grand_child->SetBounds(gfx::Size(30, 30));
grand_child->SetDrawsContent(true);
grand_child->test_properties()->hide_layer_and_subtree = true;
LayerImpl* grand_child_layer = grand_child.get();
child->test_properties()->AddChild(std::move(grand_child));
root->test_properties()->AddChild(std::move(child));
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
// We should have one render surface and two layers. The grand child has
// hidden itself.
ASSERT_EQ(1u, render_surface_list.size());
ASSERT_EQ(2, GetRenderSurface(root_layer)->num_contributors());
EXPECT_TRUE(root_layer->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child_layer->contributes_to_drawn_render_surface());
}
TEST_F(LayerTreeHostCommonTest, SubtreeHidden_TwoLayersImpl) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.pending_tree(), 2);
child->SetBounds(gfx::Size(40, 40));
child->SetDrawsContent(true);
child->test_properties()->hide_layer_and_subtree = true;
LayerImpl* child_layer = child.get();
std::unique_ptr<LayerImpl> grand_child =
LayerImpl::Create(host_impl.pending_tree(), 3);
grand_child->SetBounds(gfx::Size(30, 30));
grand_child->SetDrawsContent(true);
LayerImpl* grand_child_layer = grand_child.get();
child->test_properties()->AddChild(std::move(grand_child));
root->test_properties()->AddChild(std::move(child));
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
// We should have one render surface and one layer. The child has
// hidden itself and the grand child.
ASSERT_EQ(1u, render_surface_list.size());
ASSERT_EQ(1, GetRenderSurface(root_layer)->num_contributors());
EXPECT_TRUE(root_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child_layer->contributes_to_drawn_render_surface());
}
TEST_F(LayerTreeHostCommonTest, SubtreeHiddenWithCopyRequest) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> copy_grand_parent =
LayerImpl::Create(host_impl.pending_tree(), 2);
copy_grand_parent->SetBounds(gfx::Size(40, 40));
copy_grand_parent->SetDrawsContent(true);
LayerImpl* copy_grand_parent_layer = copy_grand_parent.get();
std::unique_ptr<LayerImpl> copy_parent =
LayerImpl::Create(host_impl.pending_tree(), 3);
copy_parent->SetBounds(gfx::Size(30, 30));
copy_parent->SetDrawsContent(true);
copy_parent->test_properties()->force_render_surface = true;
LayerImpl* copy_parent_layer = copy_parent.get();
std::unique_ptr<LayerImpl> copy_request =
LayerImpl::Create(host_impl.pending_tree(), 4);
copy_request->SetBounds(gfx::Size(20, 20));
copy_request->SetDrawsContent(true);
copy_request->test_properties()->force_render_surface = true;
LayerImpl* copy_layer = copy_request.get();
std::unique_ptr<LayerImpl> copy_child =
LayerImpl::Create(host_impl.pending_tree(), 5);
copy_child->SetBounds(gfx::Size(20, 20));
copy_child->SetDrawsContent(true);
LayerImpl* copy_child_layer = copy_child.get();
std::unique_ptr<LayerImpl> copy_grand_child =
LayerImpl::Create(host_impl.pending_tree(), 6);
copy_grand_child->SetBounds(gfx::Size(20, 20));
copy_grand_child->SetDrawsContent(true);
LayerImpl* copy_grand_child_layer = copy_grand_child.get();
std::unique_ptr<LayerImpl> copy_grand_parent_sibling_before =
LayerImpl::Create(host_impl.pending_tree(), 7);
copy_grand_parent_sibling_before->SetBounds(gfx::Size(40, 40));
copy_grand_parent_sibling_before->SetDrawsContent(true);
LayerImpl* copy_grand_parent_sibling_before_layer =
copy_grand_parent_sibling_before.get();
std::unique_ptr<LayerImpl> copy_grand_parent_sibling_after =
LayerImpl::Create(host_impl.pending_tree(), 8);
copy_grand_parent_sibling_after->SetBounds(gfx::Size(40, 40));
copy_grand_parent_sibling_after->SetDrawsContent(true);
LayerImpl* copy_grand_parent_sibling_after_layer =
copy_grand_parent_sibling_after.get();
copy_child->test_properties()->AddChild(std::move(copy_grand_child));
copy_request->test_properties()->AddChild(std::move(copy_child));
copy_parent->test_properties()->AddChild(std::move(copy_request));
copy_grand_parent->test_properties()->AddChild(std::move(copy_parent));
root->test_properties()->AddChild(
std::move(copy_grand_parent_sibling_before));
root->test_properties()->AddChild(std::move(copy_grand_parent));
root->test_properties()->AddChild(std::move(copy_grand_parent_sibling_after));
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
// 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->test_properties()->hide_layer_and_subtree = true;
copy_grand_parent_sibling_before_layer->test_properties()
->hide_layer_and_subtree = true;
copy_grand_parent_sibling_after_layer->test_properties()
->hide_layer_and_subtree = true;
copy_grand_child_layer->test_properties()->hide_layer_and_subtree = true;
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
auto& effect_tree =
root_layer->layer_tree_impl()->property_trees()->effect_tree;
EXPECT_TRUE(effect_tree.Node(root_layer->effect_tree_index())
->subtree_has_copy_request);
EXPECT_TRUE(effect_tree.Node(copy_grand_parent_layer->effect_tree_index())
->subtree_has_copy_request);
EXPECT_TRUE(effect_tree.Node(copy_parent_layer->effect_tree_index())
->subtree_has_copy_request);
EXPECT_TRUE(effect_tree.Node(copy_layer->effect_tree_index())
->subtree_has_copy_request);
// 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_list.size());
EXPECT_EQ(static_cast<uint64_t>(root_layer->id()),
render_surface_list.at(0)->id());
EXPECT_EQ(static_cast<uint64_t>(copy_grand_parent_layer->id()),
render_surface_list.at(1)->id());
EXPECT_EQ(static_cast<uint64_t>(copy_parent_layer->id()),
render_surface_list.at(2)->id());
EXPECT_EQ(static_cast<uint64_t>(copy_layer->id()),
render_surface_list.at(3)->id());
// The root render surface should have 2 contributing layers.
EXPECT_EQ(2, GetRenderSurface(root_layer)->num_contributors());
EXPECT_TRUE(root_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(copy_grand_parent_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(copy_grand_parent_sibling_before_layer
->contributes_to_drawn_render_surface());
EXPECT_FALSE(copy_grand_parent_sibling_after_layer
->contributes_to_drawn_render_surface());
// 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(1, GetRenderSurface(copy_parent_layer)->num_contributors());
EXPECT_FALSE(copy_parent_layer->contributes_to_drawn_render_surface());
// The copy layer's render surface should have 2 contributing layers.
ASSERT_EQ(2, GetRenderSurface(copy_layer)->num_contributors());
EXPECT_TRUE(copy_layer->contributes_to_drawn_render_surface());
EXPECT_TRUE(copy_child_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(copy_grand_child_layer->contributes_to_drawn_render_surface());
// 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->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = tree.Node(copy_grand_parent_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
node = tree.Node(copy_parent_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
node = tree.Node(copy_layer->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
node = tree.Node(copy_child_layer->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
node = tree.Node(copy_grand_child_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
// Though copy_layer is drawn, it shouldn't contribute to drawn surface as its
// actually hidden.
EXPECT_FALSE(GetRenderSurface(copy_layer)->contributes_to_drawn_surface());
}
TEST_F(LayerTreeHostCommonTest, ClippedOutCopyRequest) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
std::unique_ptr<LayerImpl> copy_parent =
LayerImpl::Create(host_impl.pending_tree(), 2);
copy_parent->SetDrawsContent(true);
copy_parent->SetMasksToBounds(true);
std::unique_ptr<LayerImpl> copy_layer =
LayerImpl::Create(host_impl.pending_tree(), 3);
copy_layer->SetBounds(gfx::Size(30, 30));
copy_layer->SetDrawsContent(true);
copy_layer->test_properties()->force_render_surface = true;
std::unique_ptr<LayerImpl> copy_child =
LayerImpl::Create(host_impl.pending_tree(), 4);
copy_child->SetBounds(gfx::Size(20, 20));
copy_child->SetDrawsContent(true);
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
copy_layer->test_properties()->AddChild(std::move(copy_child));
copy_parent->test_properties()->AddChild(std::move(copy_layer));
root->test_properties()->AddChild(std::move(copy_parent));
RenderSurfaceList render_surface_list;
LayerImpl* root_layer = root.get();
root_layer->layer_tree_impl()->SetRootLayerForTesting(std::move(root));
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
// We should have two render surface, as the others are clipped out.
ASSERT_EQ(2u, render_surface_list.size());
EXPECT_EQ(static_cast<uint64_t>(root_layer->id()),
render_surface_list.at(0)->id());
// The root render surface should have only 2 contributing layer, since the
// other layers are clipped away.
ASSERT_EQ(2, GetRenderSurface(root_layer)->num_contributors());
EXPECT_TRUE(root_layer->contributes_to_drawn_render_surface());
}
TEST_F(LayerTreeHostCommonTest, SingularTransformAndCopyRequests) {
LayerImpl* root = root_layer_for_testing();
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
LayerImpl* singular_transform_layer = AddChild<LayerImpl>(root);
singular_transform_layer->SetBounds(gfx::Size(100, 100));
singular_transform_layer->SetDrawsContent(true);
gfx::Transform singular;
singular.Scale3d(6.f, 6.f, 0.f);
singular_transform_layer->test_properties()->transform = singular;
LayerImpl* copy_layer = AddChild<LayerImpl>(singular_transform_layer);
copy_layer->SetBounds(gfx::Size(100, 100));
copy_layer->SetDrawsContent(true);
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
LayerImpl* copy_child = AddChild<LayerImpl>(copy_layer);
copy_child->SetBounds(gfx::Size(100, 100));
copy_child->SetDrawsContent(true);
LayerImpl* copy_grand_child = AddChild<LayerImpl>(copy_child);
copy_grand_child->SetBounds(gfx::Size(100, 100));
copy_grand_child->SetDrawsContent(true);
copy_grand_child->test_properties()->transform = singular;
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
// A layer with singular transform should not contribute to drawn render
// surface.
EXPECT_FALSE(singular_transform_layer->contributes_to_drawn_render_surface());
// Even though copy_layer and copy_child have singular screen space transform,
// they still contribute to drawn render surface as their transform to the
// closest ancestor with copy request is not singular.
EXPECT_TRUE(copy_layer->contributes_to_drawn_render_surface());
EXPECT_TRUE(copy_child->contributes_to_drawn_render_surface());
// copy_grand_child's transform to its closest ancestor with copy request is
// also singular. So, it doesn't contribute to drawn render surface.
EXPECT_FALSE(copy_grand_child->contributes_to_drawn_render_surface());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectInNonRootCopyRequest) {
LayerImpl* root = root_layer_for_testing();
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
root->SetMasksToBounds(true);
LayerImpl* copy_layer = AddChild<LayerImpl>(root);
copy_layer->SetBounds(gfx::Size(100, 100));
copy_layer->SetDrawsContent(true);
copy_layer->test_properties()->force_render_surface = true;
LayerImpl* copy_child = AddChild<LayerImpl>(copy_layer);
copy_child->test_properties()->position = gfx::PointF(40.f, 40.f);
copy_child->SetBounds(gfx::Size(20, 20));
copy_child->SetDrawsContent(true);
LayerImpl* copy_clip = AddChild<LayerImpl>(copy_layer);
copy_clip->SetBounds(gfx::Size(55, 55));
copy_clip->SetMasksToBounds(true);
LayerImpl* copy_clipped_child = AddChild<LayerImpl>(copy_clip);
copy_clipped_child->test_properties()->position = gfx::PointF(40.f, 40.f);
copy_clipped_child->SetBounds(gfx::Size(20, 20));
copy_clipped_child->SetDrawsContent(true);
LayerImpl* copy_surface = AddChild<LayerImpl>(copy_clip);
copy_surface->test_properties()->position = gfx::PointF(45.f, 45.f);
copy_surface->SetBounds(gfx::Size(20, 20));
copy_surface->SetDrawsContent(true);
copy_surface->test_properties()->force_render_surface = true;
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
EXPECT_EQ(gfx::Rect(100, 100), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(20, 20), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(15, 15), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_surface->visible_layer_rect());
// Case 2: When the non root copy request layer is clipped.
copy_layer->SetBounds(gfx::Size(50, 50));
copy_layer->SetMasksToBounds(true);
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
EXPECT_EQ(gfx::Rect(50, 50), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(5, 5), copy_surface->visible_layer_rect());
// Case 3: When there is device scale factor.
float device_scale_factor = 2.f;
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root, device_scale_factor);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
EXPECT_EQ(gfx::Rect(50, 50), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(5, 5), copy_surface->visible_layer_rect());
}
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_for_testing();
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->test_properties()->clip_parent = clip_parent;
std::unique_ptr<std::set<LayerImpl*>> clip_children(new std::set<LayerImpl*>);
clip_children->insert(clip_child);
clip_parent->test_properties()->clip_children = std::move(clip_children);
intervening->SetMasksToBounds(true);
clip_parent->SetMasksToBounds(true);
gfx::Transform scale_transform;
scale_transform.Scale(2, 2);
root->SetBounds(gfx::Size(50, 50));
render_surface->SetBounds(gfx::Size(10, 10));
render_surface->test_properties()->force_render_surface = true;
clip_parent->test_properties()->transform = scale_transform;
clip_parent->test_properties()->position = gfx::PointF(1.f, 1.f);
clip_parent->SetBounds(gfx::Size(10, 10));
intervening->test_properties()->position = gfx::PointF(1.f, 1.f);
intervening->SetBounds(gfx::Size(5, 5));
clip_child->test_properties()->position = gfx::PointF(1.f, 1.f);
clip_child->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawProperties(root);
ASSERT_TRUE(GetRenderSurface(root));
ASSERT_TRUE(GetRenderSurface(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), clip_parent->clip_rect());
ASSERT_EQ(clip_parent->clip_rect(), clip_child->clip_rect());
ASSERT_EQ(gfx::Rect(3, 3, 10, 10), intervening->clip_rect());
// 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),
GetRenderSurface(render_surface)->content_rect());
// 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(
GetRenderSurface(render_surface)->content_rect()));
ASSERT_FALSE(intervening->clip_rect().Contains(
GetRenderSurface(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_for_testing();
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->test_properties()->clip_parent = clip_parent;
intervening->SetMasksToBounds(true);
clip_parent->SetMasksToBounds(true);
gfx::Transform translation_transform;
translation_transform.Translate(2, 2);
root->SetBounds(gfx::Size(50, 50));
clip_parent->test_properties()->position = gfx::PointF(1.f, 1.f);
clip_parent->SetBounds(gfx::Size(40, 40));
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
intervening->test_properties()->position = gfx::PointF(1.f, 1.f);
intervening->SetBounds(gfx::Size(5, 5));
render_surface2->SetBounds(gfx::Size(10, 10));
render_surface2->test_properties()->force_render_surface = true;
clip_child->test_properties()->position = gfx::PointF(-10.f, -10.f);
clip_child->SetBounds(gfx::Size(60, 60));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_TRUE(GetRenderSurface(render_surface1));
EXPECT_TRUE(GetRenderSurface(render_surface2));
// render_surface1 should apply the clip from clip_parent. Though there is a
// clip child, render_surface1 can apply the clip as there are no clips
// between the clip parent and render_surface1
EXPECT_EQ(gfx::Rect(1, 1, 40, 40),
GetRenderSurface(render_surface1)->clip_rect());
EXPECT_TRUE(GetRenderSurface(render_surface1)->is_clipped());
EXPECT_EQ(gfx::Rect(), render_surface1->clip_rect());
EXPECT_FALSE(render_surface1->is_clipped());
// render_surface2 could have expanded, as there is a clip between
// clip_child's clip (clip_parent) and render_surface2's clip (intervening).
// So, it should not be clipped (their bounds would no longer be reliable).
// We should resort to layer clipping in this case.
EXPECT_EQ(gfx::Rect(0, 0, 0, 0),
GetRenderSurface(render_surface2)->clip_rect());
EXPECT_FALSE(GetRenderSurface(render_surface2)->is_clipped());
// This value is inherited from the clipping ancestor layer, 'intervening'.
EXPECT_EQ(gfx::Rect(0, 0, 5, 5), render_surface2->clip_rect());
EXPECT_TRUE(render_surface2->is_clipped());
// The content rects of render_surface2 should have expanded to contain the
// clip child.
EXPECT_EQ(gfx::Rect(0, 0, 40, 40),
GetRenderSurface(render_surface1)->content_rect());
EXPECT_EQ(gfx::Rect(-10, -10, 60, 60),
GetRenderSurface(render_surface2)->content_rect());
// The clip child should have inherited the clip parent's clip (projected to
// the right space, of course), but as render_surface1 already applies that
// clip, clip_child need not apply it again.
EXPECT_EQ(gfx::Rect(), clip_child->clip_rect());
EXPECT_EQ(gfx::Rect(9, 9, 40, 40), clip_child->visible_layer_rect());
EXPECT_FALSE(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_for_testing();
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->test_properties()->clip_parent = clip_parent;
intervening->SetMasksToBounds(true);
clip_parent->SetMasksToBounds(true);
intervening->SetScrollable(gfx::Size(1, 1));
intervening->SetElementId(LayerIdToElementIdForTesting(intervening->id()));
gfx::Transform translation_transform;
translation_transform.Translate(2, 2);
root->SetBounds(gfx::Size(50, 50));
clip_parent->test_properties()->transform = translation_transform;
clip_parent->test_properties()->position = gfx::PointF(1.f, 1.f);
clip_parent->SetBounds(gfx::Size(40, 40));
render_surface1->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->force_render_surface = true;
intervening->test_properties()->position = gfx::PointF(1.f, 1.f);
intervening->SetBounds(gfx::Size(5, 5));
render_surface2->SetBounds(gfx::Size(10, 10));
render_surface2->test_properties()->force_render_surface = true;
clip_child->test_properties()->position = gfx::PointF(-10.f, -10.f);
clip_child->SetBounds(gfx::Size(60, 60));
BuildPropertyTreesForTesting();
intervening->SetCurrentScrollOffset(gfx::ScrollOffset(3, 3));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_TRUE(GetRenderSurface(render_surface1));
EXPECT_TRUE(GetRenderSurface(render_surface2));
// render_surface1 should apply the clip from clip_parent. Though there is a
// clip child, render_surface1 can apply the clip as there are no clips
// between the clip parent and render_surface1
EXPECT_EQ(gfx::Rect(3, 3, 40, 40),
GetRenderSurface(render_surface1)->clip_rect());
EXPECT_TRUE(GetRenderSurface(render_surface1)->is_clipped());
EXPECT_EQ(gfx::Rect(), render_surface1->clip_rect());
EXPECT_FALSE(render_surface1->is_clipped());
// render_surface2 could have expanded, as there is a clip between
// clip_child's clip (clip_parent) and render_surface2's clip (intervening).
// So, it should not be clipped (their bounds would no longer be reliable).
// We should resort to layer clipping in this case.
EXPECT_EQ(gfx::Rect(0, 0, 0, 0),
GetRenderSurface(render_surface2)->clip_rect());
EXPECT_FALSE(GetRenderSurface(render_surface2)->is_clipped());
// This value is inherited from the clipping ancestor layer, 'intervening'.
EXPECT_EQ(gfx::Rect(0, 0, 5, 5), render_surface2->clip_rect());
EXPECT_TRUE(render_surface2->is_clipped());
// The content rects of render_surface2 should have expanded to contain the
// clip child.
EXPECT_EQ(gfx::Rect(0, 0, 40, 40),
GetRenderSurface(render_surface1)->content_rect());
EXPECT_EQ(gfx::Rect(-10, -10, 60, 60),
GetRenderSurface(render_surface2)->content_rect());
// The clip child should have inherited the clip parent's clip (projected to
// the right space, of course), but as render_surface1 already applies that
// clip, clip_child need not apply it again.
EXPECT_EQ(gfx::Rect(), clip_child->clip_rect());
EXPECT_EQ(gfx::Rect(12, 12, 40, 40), clip_child->visible_layer_rect());
EXPECT_FALSE(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_for_testing();
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->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
root->SetBounds(gfx::Size(50, 50));
clip_parent->SetBounds(gfx::Size(40, 40));
clip_parent->SetMasksToBounds(true);
intervening->SetBounds(gfx::Size(5, 5));
intervening->SetMasksToBounds(true);
clip_child->SetBounds(gfx::Size(60, 60));
child->SetBounds(gfx::Size(60, 60));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
// Neither the clip child nor its descendant should have inherited the clip
// from |intervening|.
EXPECT_EQ(gfx::Rect(0, 0, 40, 40), clip_child->clip_rect());
EXPECT_TRUE(clip_child->is_clipped());
EXPECT_EQ(gfx::Rect(0, 0, 40, 40), child->visible_layer_rect());
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)
// + clip_layer (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_for_testing();
LayerImpl* clip_parent = AddChildToRoot<LayerImpl>();
LayerImpl* clip_layer = AddChild<LayerImpl>(clip_parent);
LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_layer);
LayerImpl* clip_child = AddChild<LayerImpl>(render_surface1);
LayerImpl* render_surface2 = AddChild<LayerImpl>(clip_layer);
LayerImpl* non_clip_child = AddChild<LayerImpl>(render_surface2);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
clip_parent->SetMasksToBounds(true);
clip_layer->SetMasksToBounds(true);
render_surface1->SetDrawsContent(true);
clip_child->SetDrawsContent(true);
render_surface2->SetDrawsContent(true);
non_clip_child->SetDrawsContent(true);
root->SetBounds(gfx::Size(15, 15));
clip_parent->SetBounds(gfx::Size(10, 10));
clip_layer->SetBounds(gfx::Size(10, 10));
render_surface1->test_properties()->position = gfx::PointF(5, 5);
render_surface1->SetBounds(gfx::Size(5, 5));
render_surface1->test_properties()->force_render_surface = true;
render_surface2->SetBounds(gfx::Size(5, 5));
render_surface2->test_properties()->force_render_surface = true;
clip_child->test_properties()->position = gfx::PointF(-1, 1);
clip_child->SetBounds(gfx::Size(10, 10));
non_clip_child->SetBounds(gfx::Size(5, 5));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_TRUE(GetRenderSurface(render_surface1));
EXPECT_TRUE(GetRenderSurface(render_surface2));
EXPECT_EQ(gfx::Rect(-5, -5, 10, 10), render_surface1->clip_rect());
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),
GetRenderSurface(render_surface1)->clip_rect());
EXPECT_FALSE(GetRenderSurface(render_surface1)->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),
GetRenderSurface(render_surface1)->content_rect());
// This render surface should clip. It has no unclipped descendants.
EXPECT_EQ(gfx::Rect(0, 0, 10, 10),
GetRenderSurface(render_surface2)->clip_rect());
EXPECT_TRUE(GetRenderSurface(render_surface2)->is_clipped());
EXPECT_FALSE(render_surface2->is_clipped());
// It also shouldn't have grown to accomodate the clip child.
EXPECT_EQ(gfx::Rect(0, 0, 5, 5),
GetRenderSurface(render_surface2)->content_rect());
}
TEST_F(LayerTreeHostCommonTest,
CreateRenderSurfaceWhenFlattenInsideRenderingContext) {
// Verifies that Render Surfaces are created at the edge of rendering context.
LayerImpl* root = root_layer_for_testing();
LayerImpl* child1 = AddChildToRoot<LayerImpl>();
LayerImpl* child2 = AddChild<LayerImpl>(child1);
LayerImpl* child3 = AddChild<LayerImpl>(child2);
root->SetDrawsContent(true);
gfx::Size bounds(100, 100);
root->SetBounds(bounds);
child1->SetBounds(bounds);
child1->SetDrawsContent(true);
child1->test_properties()->should_flatten_transform = false;
child1->test_properties()->sorting_context_id = 1;
child2->SetBounds(bounds);
child2->SetDrawsContent(true);
child2->test_properties()->sorting_context_id = 1;
child3->SetBounds(bounds);
child3->SetDrawsContent(true);
child3->test_properties()->sorting_context_id = 1;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
// Verify which render surfaces were created.
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(child1), GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(child2), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(child3), GetRenderSurface(child2));
}
TEST_F(LayerTreeHostCommonTest, DoNotIncludeBackfaceInvisibleSurfaces) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* back_facing = AddChild<LayerImpl>(root);
LayerImpl* render_surface1 = AddChild<LayerImpl>(back_facing);
LayerImpl* child1 = AddChild<LayerImpl>(render_surface1);
LayerImpl* flattener = AddChild<LayerImpl>(back_facing);
LayerImpl* render_surface2 = AddChild<LayerImpl>(flattener);
LayerImpl* child2 = AddChild<LayerImpl>(render_surface2);
child1->SetDrawsContent(true);
child2->SetDrawsContent(true);
root->SetBounds(gfx::Size(50, 50));
back_facing->SetBounds(gfx::Size(50, 50));
back_facing->test_properties()->should_flatten_transform = false;
render_surface1->SetBounds(gfx::Size(30, 30));
render_surface1->test_properties()->should_flatten_transform = false;
render_surface1->test_properties()->force_render_surface = true;
render_surface1->test_properties()->double_sided = false;
child1->SetBounds(gfx::Size(20, 20));
flattener->SetBounds(gfx::Size(30, 30));
render_surface2->SetBounds(gfx::Size(30, 30));
render_surface2->test_properties()->should_flatten_transform = false;
render_surface2->test_properties()->force_render_surface = true;
render_surface2->test_properties()->double_sided = false;
child2->SetBounds(gfx::Size(20, 20));
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(3u, render_surface_list_impl()->size());
EXPECT_EQ(2, render_surface_list_impl()->at(0)->num_contributors());
EXPECT_EQ(1, render_surface_list_impl()->at(1)->num_contributors());
gfx::Transform rotation_transform;
rotation_transform.RotateAboutXAxis(180.0);
back_facing->test_properties()->transform = 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_list_impl()->size());
EXPECT_EQ(1, render_surface_list_impl()->at(0)->num_contributors());
}
TEST_F(LayerTreeHostCommonTest, DoNotIncludeBackfaceInvisibleLayers) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
root->test_properties()->should_flatten_transform = false;
child->SetBounds(gfx::Size(30, 30));
child->test_properties()->double_sided = false;
child->test_properties()->should_flatten_transform = false;
grand_child->SetBounds(gfx::Size(20, 20));
grand_child->SetDrawsContent(true);
grand_child->SetUseParentBackfaceVisibility(true);
grand_child->test_properties()->should_flatten_transform = false;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1u, render_surface_list_impl()->size());
EXPECT_TRUE(grand_child->contributes_to_drawn_render_surface());
// A ll layers with invisible backfgaces should be checked.
EXPECT_FALSE(root->should_check_backface_visibility());
EXPECT_TRUE(child->should_check_backface_visibility());
EXPECT_TRUE(grand_child->should_check_backface_visibility());
gfx::Transform rotation_transform;
rotation_transform.RotateAboutXAxis(180.0);
child->test_properties()->transform = rotation_transform;
child->test_properties()->sorting_context_id = 1;
grand_child->test_properties()->sorting_context_id = 1;
child->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1u, render_surface_list_impl()->size());
EXPECT_EQ(0, render_surface_list_impl()->at(0)->num_contributors());
// We should check for backface visibilty of child as it has a rotation
// transform. We should also check for grand_child as it uses the backface
// visibility of its parent.
EXPECT_FALSE(root->should_check_backface_visibility());
EXPECT_TRUE(child->should_check_backface_visibility());
EXPECT_TRUE(grand_child->should_check_backface_visibility());
grand_child->SetUseParentBackfaceVisibility(false);
grand_child->test_properties()->double_sided = false;
grand_child->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1u, render_surface_list_impl()->size());
EXPECT_EQ(0, render_surface_list_impl()->at(0)->num_contributors());
// We should check the backface visibility of child as it has a rotation
// transform and for grand_child as it is in a 3d rendering context and not
// the root of it.
EXPECT_FALSE(root->should_check_backface_visibility());
EXPECT_TRUE(child->should_check_backface_visibility());
EXPECT_TRUE(grand_child->should_check_backface_visibility());
}
TEST_F(LayerTreeHostCommonTest, TransformAnimationUpdatesBackfaceVisibility) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* back_facing = AddChild<LayerImpl>(root);
LayerImpl* render_surface1 = AddChild<LayerImpl>(back_facing);
LayerImpl* render_surface2 = AddChild<LayerImpl>(back_facing);
gfx::Transform rotate_about_y;
rotate_about_y.RotateAboutYAxis(180.0);
root->SetBounds(gfx::Size(50, 50));
root->test_properties()->should_flatten_transform = false;
root->test_properties()->sorting_context_id = 1;
back_facing->test_properties()->transform = rotate_about_y;
back_facing->SetBounds(gfx::Size(50, 50));
back_facing->test_properties()->should_flatten_transform = false;
back_facing->test_properties()->sorting_context_id = 1;
render_surface1->SetBounds(gfx::Size(30, 30));
render_surface1->test_properties()->should_flatten_transform = false;
render_surface1->test_properties()->double_sided = false;
render_surface1->test_properties()->force_render_surface = true;
render_surface1->test_properties()->sorting_context_id = 1;
render_surface2->SetBounds(gfx::Size(30, 30));
render_surface2->test_properties()->should_flatten_transform = false;
render_surface2->test_properties()->double_sided = false;
render_surface2->test_properties()->force_render_surface = true;
render_surface2->test_properties()->sorting_context_id = 1;
SetElementIdsForTesting();
ExecuteCalculateDrawProperties(root);
const EffectTree& tree =
root->layer_tree_impl()->property_trees()->effect_tree;
EXPECT_TRUE(tree.Node(render_surface1->effect_tree_index())
->hidden_by_backface_visibility);
EXPECT_TRUE(tree.Node(render_surface2->effect_tree_index())
->hidden_by_backface_visibility);
root->layer_tree_impl()->SetTransformMutated(back_facing->element_id(),
gfx::Transform());
root->layer_tree_impl()->SetTransformMutated(render_surface2->element_id(),
rotate_about_y);
ExecuteCalculateDrawProperties(root);
EXPECT_FALSE(tree.Node(render_surface1->effect_tree_index())
->hidden_by_backface_visibility);
EXPECT_TRUE(tree.Node(render_surface2->effect_tree_index())
->hidden_by_backface_visibility);
root->layer_tree_impl()->SetTransformMutated(render_surface1->element_id(),
rotate_about_y);
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(tree.Node(render_surface1->effect_tree_index())
->hidden_by_backface_visibility);
EXPECT_TRUE(tree.Node(render_surface2->effect_tree_index())
->hidden_by_backface_visibility);
}
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_for_testing();
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->test_properties()->scroll_parent = scroll_parent;
root->SetBounds(gfx::Size(50, 50));
scroll_parent_border->SetBounds(gfx::Size(40, 40));
scroll_parent_clip->SetBounds(gfx::Size(30, 30));
scroll_parent->SetBounds(gfx::Size(50, 50));
scroll_child->SetBounds(gfx::Size(50, 50));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(gfx::Rect(0, 0, 30, 30), scroll_child->clip_rect());
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_for_testing();
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->test_properties()->scroll_parent = scroll_parent;
root->SetBounds(gfx::Size(50, 50));
scroll_child_target->SetBounds(gfx::Size(50, 50));
scroll_child_target->test_properties()->force_render_surface = true;
scroll_child->SetBounds(gfx::Size(50, 50));
scroll_parent_target->test_properties()->position = gfx::PointF(10, 10);
scroll_parent_target->SetBounds(gfx::Size(50, 50));
scroll_parent_target->SetMasksToBounds(true);
scroll_parent_target->test_properties()->force_render_surface = true;
scroll_parent->SetBounds(gfx::Size(50, 50));
float device_scale_factor = 1.5f;
RenderSurfaceList render_surface_list_impl;
gfx::Size device_viewport_size =
gfx::Size(root->bounds().width() * device_scale_factor,
root->bounds().height() * device_scale_factor);
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, device_viewport_size, gfx::Transform(), &render_surface_list_impl);
inputs.device_scale_factor = device_scale_factor;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_EQ(scroll_child->effect_tree_index(),
scroll_child_target->effect_tree_index());
EXPECT_EQ(scroll_child->visible_layer_rect(), gfx::Rect(10, 10, 40, 40));
EXPECT_EQ(scroll_child->clip_rect(), gfx::Rect(15, 15, 75, 75));
gfx::Transform scale;
scale.Scale(1.5f, 1.5f);
EXPECT_TRANSFORMATION_MATRIX_EQ(scroll_child->DrawTransform(), scale);
}
TEST_F(LayerTreeHostCommonTest, SingularTransformSubtreesDoNotDraw) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChildToRoot<LayerImpl>();
LayerImpl* child = AddChild<LayerImpl>(parent);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
root->test_properties()->sorting_context_id = 1;
parent->SetBounds(gfx::Size(30, 30));
parent->SetDrawsContent(true);
parent->test_properties()->force_render_surface = true;
parent->test_properties()->sorting_context_id = 1;
child->SetBounds(gfx::Size(20, 20));
child->SetDrawsContent(true);
child->test_properties()->force_render_surface = true;
child->test_properties()->sorting_context_id = 1;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(3u, render_surface_list_impl()->size());
gfx::Transform singular_transform;
singular_transform.Scale3d(
SkDoubleToMScalar(1.0), SkDoubleToMScalar(1.0), SkDoubleToMScalar(0.0));
child->test_properties()->transform = singular_transform;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(2u, render_surface_list_impl()->size());
// Ensure that the entire subtree under a layer with singular transform does
// not get rendered.
parent->test_properties()->transform = singular_transform;
child->test_properties()->transform = gfx::Transform();
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1u, render_surface_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_for_testing();
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);
root->SetBounds(gfx::Size(50, 50));
scroll_parent_border->SetBounds(gfx::Size(40, 40));
scroll_parent_clip->SetBounds(gfx::Size(30, 30));
scroll_parent_clip->SetMasksToBounds(true);
scroll_parent->SetBounds(gfx::Size(50, 50));
scroll_child->SetBounds(gfx::Size(50, 50));
scroll_child->test_properties()->scroll_parent = scroll_parent;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(gfx::Rect(0, 0, 30, 30), scroll_child->clip_rect());
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_for_testing();
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->test_properties()->scroll_parent = scroll_parent;
scroll_parent_border->test_properties()->scroll_parent = scroll_grandparent;
root->SetBounds(gfx::Size(50, 50));
scroll_grandparent_border->SetBounds(gfx::Size(40, 40));
scroll_grandparent_clip->SetBounds(gfx::Size(20, 20));
scroll_grandparent->SetBounds(gfx::Size(50, 50));
scroll_parent_border->SetBounds(gfx::Size(40, 40));
scroll_parent_clip->SetBounds(gfx::Size(30, 30));
scroll_parent->SetBounds(gfx::Size(50, 50));
scroll_child->SetBounds(gfx::Size(50, 50));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(gfx::Rect(0, 0, 20, 20), scroll_child->clip_rect());
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(3, GetRenderSurface(root)->num_contributors());
EXPECT_TRUE(scroll_child->contributes_to_drawn_render_surface());
EXPECT_TRUE(scroll_parent->contributes_to_drawn_render_surface());
EXPECT_TRUE(scroll_grandparent->contributes_to_drawn_render_surface());
}
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_for_testing();
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->test_properties()->scroll_parent = scroll_parent;
scroll_parent_border->test_properties()->scroll_parent = scroll_grandparent;
root->SetBounds(gfx::Size(50, 50));
scroll_grandparent_border->SetBounds(gfx::Size(40, 40));
scroll_grandparent_clip->SetBounds(gfx::Size(20, 20));
scroll_grandparent->SetBounds(gfx::Size(50, 50));
render_surface1->SetBounds(gfx::Size(50, 50));
render_surface1->test_properties()->force_render_surface = true;
scroll_parent_border->SetBounds(gfx::Size(40, 40));
scroll_parent_clip->SetBounds(gfx::Size(30, 30));
scroll_parent->SetBounds(gfx::Size(50, 50));
render_surface2->SetBounds(gfx::Size(50, 50));
render_surface2->test_properties()->force_render_surface = true;
scroll_child->SetBounds(gfx::Size(50, 50));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(gfx::Rect(0, 0, 20, 20), scroll_child->clip_rect());
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_list's order should be unaffected.
EXPECT_EQ(3u, render_surface_list_impl()->size());
EXPECT_EQ(GetRenderSurface(root), render_surface_list_impl()->at(0));
EXPECT_EQ(GetRenderSurface(render_surface2),
render_surface_list_impl()->at(1));
EXPECT_EQ(GetRenderSurface(render_surface1),
render_surface_list_impl()->at(2));
}
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
//
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* fixed = AddChild<LayerImpl>(render_surface);
LayerImpl* child = AddChild<LayerImpl>(fixed);
render_surface->test_properties()->force_render_surface = true;
root->test_properties()->is_container_for_fixed_position_layers = true;
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
fixed->test_properties()->position_constraint = constraint;
root->SetBounds(gfx::Size(50, 50));
render_surface->test_properties()->position = gfx::PointF(7.f, 9.f);
render_surface->SetBounds(gfx::Size(50, 50));
render_surface->SetDrawsContent(true);
fixed->test_properties()->position = gfx::PointF(10.f, 15.f);
fixed->SetBounds(gfx::Size(50, 50));
fixed->SetDrawsContent(true);
child->test_properties()->position = gfx::PointF(1.f, 2.f);
child->SetBounds(gfx::Size(50, 50));
child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
TransformTree& transform_tree =
host_impl()->active_tree()->property_trees()->transform_tree;
EffectTree& effect_tree =
host_impl()->active_tree()->property_trees()->effect_tree;
gfx::Transform expected_fixed_draw_transform;
expected_fixed_draw_transform.Translate(10.f, 15.f);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_fixed_draw_transform,
draw_property_utils::DrawTransform(fixed, transform_tree, effect_tree));
gfx::Transform expected_fixed_screen_space_transform;
expected_fixed_screen_space_transform.Translate(17.f, 24.f);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_fixed_screen_space_transform,
draw_property_utils::ScreenSpaceTransform(fixed, transform_tree));
gfx::Transform expected_child_draw_transform;
expected_child_draw_transform.Translate(11.f, 17.f);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_child_draw_transform,
draw_property_utils::DrawTransform(child, transform_tree, effect_tree));
gfx::Transform expected_child_screen_space_transform;
expected_child_screen_space_transform.Translate(18.f, 26.f);
EXPECT_TRANSFORMATION_MATRIX_EQ(
expected_child_screen_space_transform,
draw_property_utils::ScreenSpaceTransform(child, transform_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;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root_ptr =
LayerImpl::Create(host_impl.active_tree(), 1);
LayerImpl* root = root_ptr.get();
std::unique_ptr<LayerImpl> container =
LayerImpl::Create(host_impl.active_tree(), 2);
LayerImpl* container_layer = container.get();
std::unique_ptr<LayerImpl> scroller =
LayerImpl::Create(host_impl.active_tree(), 3);
LayerImpl* scroll_layer = scroller.get();
std::unique_ptr<LayerImpl> fixed =
LayerImpl::Create(host_impl.active_tree(), 4);
LayerImpl* fixed_layer = fixed.get();
container->test_properties()->is_container_for_fixed_position_layers = true;
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
fixed->test_properties()->position_constraint = constraint;
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
gfx::Transform container_transform;
container_transform.Translate3d(10.0, 20.0, 0.0);
gfx::Vector2dF container_offset = container_transform.To2dTranslation();
root->SetBounds(gfx::Size(50, 50));
container->test_properties()->transform = container_transform;
container->SetBounds(gfx::Size(40, 40));
container->SetDrawsContent(true);
scroller->SetBounds(gfx::Size(30, 30));
scroller->SetScrollable(container->bounds());
scroller->SetDrawsContent(true);
fixed->SetBounds(gfx::Size(50, 50));
fixed->SetDrawsContent(true);
scroller->test_properties()->AddChild(std::move(fixed));
container->test_properties()->AddChild(std::move(scroller));
root->test_properties()->AddChild(std::move(container));
root->layer_tree_impl()->SetRootLayerForTesting(std::move(root_ptr));
root->layer_tree_impl()->BuildPropertyTreesForTesting();
// Rounded to integers already.
{
gfx::Vector2dF scroll_delta(3.0, 5.0);
SetScrollOffsetDelta(scroll_layer, scroll_delta);
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
root->layer_tree_impl()
->property_trees()
->transform_tree.set_source_to_parent_updates_allowed(false);
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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);
SetScrollOffsetDelta(scroll_layer, scroll_delta);
gfx::Vector2dF rounded_scroll_delta(4.f, 8.f);
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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.
{
SetScrollOffsetDelta(scroll_layer, gfx::Vector2dF());
gfx::Transform scaled_container_transform = container_transform;
scaled_container_transform.Scale3d(2.0, 2.0, 1.0);
container_layer->test_properties()->transform = scaled_container_transform;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
gfx::Vector2dF scroll_delta(4.5f, 8.5f);
SetScrollOffsetDelta(scroll_layer, scroll_delta);
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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->test_properties()->transform = container_transform;
}
}
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_for_testing();
LayerImpl* animated_layer = AddChildToRoot<FakePictureLayerImpl>();
LayerImpl* surface = AddChild<LayerImpl>(animated_layer);
LayerImpl* container = AddChild<LayerImpl>(surface);
LayerImpl* scroller = AddChild<LayerImpl>(container);
gfx::Transform start_scale;
start_scale.Scale(1.5f, 1.5f);
root->SetBounds(gfx::Size(50, 50));
animated_layer->test_properties()->transform = start_scale;
animated_layer->SetBounds(gfx::Size(50, 50));
surface->SetBounds(gfx::Size(50, 50));
surface->test_properties()->force_render_surface = true;
container->SetBounds(gfx::Size(50, 50));
scroller->SetBounds(gfx::Size(100, 100));
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
scroller->SetScrollable(container->bounds());
scroller->SetDrawsContent(true);
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);
SetElementIdsForTesting();
AddAnimatedTransformToElementWithAnimation(animated_layer->element_id(),
timeline_impl(), 1.0,
start_operations, end_operations);
BuildPropertyTreesForTesting();
gfx::Vector2dF scroll_delta(5.f, 9.f);
SetScrollOffsetDelta(scroller, scroll_delta);
ExecuteCalculateDrawProperties(root);
gfx::Vector2dF expected_draw_transform_translation(-7.5f, -13.5f);
EXPECT_VECTOR2DF_EQ(expected_draw_transform_translation,
scroller->DrawTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, ScrollSnappingWithScrollChild) {
// This test verifies that a scrolling child of a scrolling layer doesn't get
// snapped to integer coordinates.
//
// + root
// + container
// + scroller
// + scroll_child
//
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> scroll_child = Layer::Create();
root->AddChild(container);
root->AddChild(scroll_child);
container->AddChild(scroller);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
scroll_child->SetScrollParent(scroller.get());
gfx::Transform rotate;
rotate.RotateAboutYAxis(30);
root->SetBounds(gfx::Size(50, 50));
container->SetBounds(gfx::Size(50, 50));
scroller->SetBounds(gfx::Size(100, 100));
scroller->SetScrollable(container->bounds());
scroller->SetPosition(gfx::PointF(10.3f, 10.3f));
scroll_child->SetBounds(gfx::Size(10, 10));
scroll_child->SetTransform(rotate);
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* scroll_child_impl = layer_tree_impl->LayerById(scroll_child->id());
gfx::Vector2dF scroll_delta(5.f, 9.f);
SetScrollOffsetDelta(scroller_impl, scroll_delta);
ExecuteCalculateDrawProperties(root_impl);
gfx::Vector2dF expected_scroller_screen_space_transform_translation(5.f, 1.f);
EXPECT_VECTOR2DF_EQ(expected_scroller_screen_space_transform_translation,
scroller_impl->ScreenSpaceTransform().To2dTranslation());
gfx::Transform expected_scroll_child_screen_space_transform;
expected_scroll_child_screen_space_transform.Translate(-5.3f, -9.3f);
expected_scroll_child_screen_space_transform.RotateAboutYAxis(30);
EXPECT_TRANSFORMATION_MATRIX_EQ(expected_scroll_child_screen_space_transform,
scroll_child_impl->ScreenSpaceTransform());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionTop) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 10.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(10, 20, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(10, 20));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(10.f, 20.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(5.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(5.f, 15.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the Y coordinate should now be clamped.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 15.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the end of the sticky container (note: this element does not
// have its own layer as it does not need to be composited).
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 50.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, -10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionTopScrollParent) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
root->AddChild(sticky_pos);
sticky_pos->SetScrollParent(scroller.get());
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
// The sticky layer has already been scrolled on the main thread side, and has
// stuck. This test then checks that further changes from cc-only scrolling
// are handled correctly.
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 10.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(10, 20, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(200, 200));
container->SetBounds(gfx::Size(100, 100));
container->SetPosition(gfx::PointF(50, 50));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(60, 70));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(60.f, 70.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(5.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(55.f, 65.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the Y coordinate should now be clamped.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 15.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(45.f, 60.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(45.f, 60.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the end of the sticky container (note: this element does not
// have its own layer as it does not need to be composited).
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 50.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(45.f, 40.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionSubpixelScroll) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_bottom = true;
sticky_position.bottom_offset = 10.0f;
sticky_position.constraint_box_rect = gfx::RectF(0, 0, 100, 100);
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 200, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 100, 500);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(100, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 200));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 0.8f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 80.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionBottom) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_bottom = true;
sticky_position.bottom_offset = 10.0f;
sticky_position.constraint_box_rect = gfx::RectF(0, 0, 100, 100);
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 150, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 100, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 150));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
// Initially the sticky element is moved up to the top of the container.
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 100.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 95.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Once we get past the top of the container it moves to be aligned 10px
// up from the the bottom of the scroller.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 80.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 30.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 80.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Once we scroll past its initial location, it sticks there.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 150.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionBottomOuterViewportDelta) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> page_scale = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> outer_clip = Layer::Create();
scoped_refptr<Layer> outer_viewport = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(page_scale);
page_scale->AddChild(scroller);
scroller->AddChild(outer_clip);
outer_clip->AddChild(outer_viewport);
outer_viewport->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
ViewportLayers viewport_layers;
viewport_layers.page_scale = page_scale;
viewport_layers.inner_viewport_container = root;
viewport_layers.outer_viewport_container = outer_clip;
viewport_layers.inner_viewport_scroll = scroller;
viewport_layers.outer_viewport_scroll = outer_viewport;
host()->RegisterViewportLayers(viewport_layers);
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_bottom = true;
sticky_position.bottom_offset = 10.0f;
sticky_position.constraint_box_rect = gfx::RectF(0, 0, 100, 100);
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 70, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 60, 100, 100);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
scroller->SetScrollable(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(100, 1000));
outer_clip->SetBounds(gfx::Size(100, 100));
outer_viewport->SetScrollable(gfx::Size(100, 100));
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 70));
ExecuteCalculateDrawProperties(root.get(), 1.f, 1.f, root.get(),
scroller.get(), outer_viewport.get());
host()->CommitAndCreateLayerImplTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
ASSERT_EQ(outer_viewport->id(),
layer_tree_impl->OuterViewportScrollLayer()->id());
LayerImpl* inner_scroll = layer_tree_impl->InnerViewportScrollLayer();
LayerImpl* outer_scroll = layer_tree_impl->OuterViewportScrollLayer();
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
LayerImpl* outer_clip_impl = layer_tree_impl->LayerById(outer_clip->id());
// Initially the sticky element is moved to the bottom of the container.
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 70.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// We start to hide the toolbar, but not far enough that the sticky element
// should be moved up yet.
outer_clip_impl->SetViewportBoundsDelta(gfx::Vector2dF(0.f, -10.f));
ExecuteCalculateDrawProperties(root_impl, 1.f, 1.f, root_impl, inner_scroll,
outer_scroll);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 70.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// On hiding more of the toolbar the sticky element starts to stick.
outer_clip_impl->SetViewportBoundsDelta(gfx::Vector2dF(0.f, -20.f));
ExecuteCalculateDrawProperties(root_impl, 1.f, 1.f, root_impl, inner_scroll,
outer_scroll);
// On hiding more the sticky element stops moving as it has reached its
// limit.
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 60.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
outer_clip_impl->SetViewportBoundsDelta(gfx::Vector2dF(0.f, -30.f));
ExecuteCalculateDrawProperties(root_impl, 1.f, 1.f, root_impl, inner_scroll,
outer_scroll);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 60.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionLeftRight) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_left = true;
sticky_position.is_anchored_right = true;
sticky_position.left_offset = 10.f;
sticky_position.right_offset = 10.f;
sticky_position.constraint_box_rect = gfx::RectF(0, 0, 100, 100);
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(145, 0, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(100, 0, 100, 100);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(145, 0));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
// Initially the sticky element is moved the leftmost side of the container.
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(100.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(5.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(95.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Once we get past the left side of the container it moves to be aligned 10px
// up from the the right of the scroller.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(25.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(80.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(30.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(80.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// When we get to the sticky element's original position we stop sticking
// to the right.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(95.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(50.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(105.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(40.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// The element starts sticking to the left once we scroll far enough.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(150.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(10.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(155.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(10.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Finally it stops sticking when it hits the right side of the container.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(190.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(195.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
// This test ensures that the compositor sticky position code correctly accounts
// for the element having been given a position from the main thread sticky
// position code.
TEST_F(LayerTreeHostCommonTest, StickyPositionMainThreadUpdates) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 10.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(10, 20, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(10, 20));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(10.f, 20.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(5.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(5.f, 15.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the Y coordinate should now be clamped.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(15.f, 15.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Now the main thread commits the new position of the sticky element.
scroller->SetScrollOffset(gfx::ScrollOffset(15, 15));
// Shift the layer by -offset_for_position_sticky.
sticky_pos->SetPosition(gfx::PointF(10, 25) - gfx::Vector2dF(0, 5));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
layer_tree_impl = host()->host_impl()->active_tree();
root_impl = layer_tree_impl->LayerById(root->id());
scroller_impl = layer_tree_impl->LayerById(scroller->id());
sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
// The element should still be where it was before. We reset the delta to
// (0, 0) because we have synced a scroll offset of (15, 15) from the main
// thread.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// And if we scroll a little further it remains there.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 10.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(-5.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
// This tests the main thread updates with a composited sticky container. In
// this case the position received from main is relative to the container but
// the constraint rects are relative to the ancestor scroller.
TEST_F(LayerTreeHostCommonTest, StickyPositionCompositedContainer) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_container = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_container);
sticky_container->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 10.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(20, 30, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(20, 20, 30, 30);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_container->SetPosition(gfx::PointF(20, 20));
sticky_container->SetBounds(gfx::Size(30, 30));
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 10));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 30.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 25.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the Y coordinate should now be clamped.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Now the main thread commits the new position of the sticky element.
scroller->SetScrollOffset(gfx::ScrollOffset(0, 25));
// Shift the layer by -offset_for_position_sticky.
sticky_pos->SetPosition(gfx::PointF(0, 15) - gfx::Vector2dF(0, 5));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
layer_tree_impl = host()->host_impl()->active_tree();
root_impl = layer_tree_impl->LayerById(root->id());
scroller_impl = layer_tree_impl->LayerById(scroller->id());
sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
// The element should still be where it was before. We reset the delta to
// (0, 0) because we have synced a scroll offset of (0, 25) from the main
// thread.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 0.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// And if we scroll a little further it remains there.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// And hits the bottom of the container.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 10.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(20.f, 5.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
// A transform on a sticky element should not affect its sticky position.
TEST_F(LayerTreeHostCommonTest, StickyPositionScaledStickyBox) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
gfx::Transform t;
t.Scale(2, 2);
sticky_pos->SetTransform(t);
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 0.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 20, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 20));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 20.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 15.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 5.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the box is positioned at the scroller
// edge.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 30.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 0.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the end of the sticky container.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 50.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, -10.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
// Tests that a transform does not affect the sticking points. The sticky
// element will however move relative to the viewport due to its transform.
TEST_F(LayerTreeHostCommonTest, StickyPositionScaledContainer) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> sticky_container = Layer::Create();
scoped_refptr<Layer> sticky_pos = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(sticky_container);
sticky_container->AddChild(sticky_pos);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
gfx::Transform t;
t.Scale(2, 2);
sticky_container->SetTransform(t);
LayerStickyPositionConstraint sticky_position;
sticky_position.is_sticky = true;
sticky_position.is_anchored_top = true;
sticky_position.top_offset = 0.0f;
sticky_position.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 20, 10, 10);
sticky_position.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 50);
sticky_pos->SetStickyPositionConstraint(sticky_position);
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetScrollable(container->bounds());
sticky_container->SetBounds(gfx::Size(50, 50));
sticky_pos->SetBounds(gfx::Size(10, 10));
sticky_pos->SetPosition(gfx::PointF(0, 20));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* sticky_pos_impl = layer_tree_impl->LayerById(sticky_pos->id());
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 40.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than sticking point, sticky element should move with scroll as
// we haven't gotten to the initial sticky item location yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 15.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 25.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the sticking point, the box is positioned at the scroller
// edge but is also scaled by its container so it begins to move down.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 25.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 25.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 30.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 30.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll past the end of the sticky container.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 50.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 30.f),
sticky_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, StickyPositionNested) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> outer_sticky = Layer::Create();
scoped_refptr<Layer> inner_sticky = Layer::Create();
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(outer_sticky);
outer_sticky->AddChild(inner_sticky);
host()->SetRootLayer(root);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
outer_sticky->SetElementId(LayerIdToElementIdForTesting(outer_sticky->id()));
root->SetBounds(gfx::Size(100, 100));
container->SetBounds(gfx::Size(100, 100));
scroller->SetBounds(gfx::Size(100, 1000));
scroller->SetScrollable(container->bounds());
outer_sticky->SetBounds(gfx::Size(10, 50));
outer_sticky->SetPosition(gfx::PointF(0, 50));
inner_sticky->SetBounds(gfx::Size(10, 10));
inner_sticky->SetPosition(gfx::PointF(0, 0));
LayerStickyPositionConstraint outer_sticky_pos;
outer_sticky_pos.is_sticky = true;
outer_sticky_pos.is_anchored_top = true;
outer_sticky_pos.top_offset = 10.0f;
outer_sticky_pos.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 50, 10, 50);
outer_sticky_pos.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 0, 50, 400);
outer_sticky->SetStickyPositionConstraint(outer_sticky_pos);
LayerStickyPositionConstraint inner_sticky_pos;
inner_sticky_pos.is_sticky = true;
inner_sticky_pos.is_anchored_top = true;
inner_sticky_pos.top_offset = 25.0f;
inner_sticky_pos.scroll_container_relative_sticky_box_rect =
gfx::Rect(0, 50, 10, 10);
inner_sticky_pos.scroll_container_relative_containing_block_rect =
gfx::Rect(0, 50, 10, 50);
inner_sticky_pos.nearest_element_shifting_containing_block =
outer_sticky->element_id();
inner_sticky->SetStickyPositionConstraint(inner_sticky_pos);
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* outer_sticky_impl = layer_tree_impl->LayerById(outer_sticky->id());
LayerImpl* inner_sticky_impl = layer_tree_impl->LayerById(inner_sticky->id());
ExecuteCalculateDrawProperties(root_impl);
// Before any scrolling is done, the sticky elements should still be at their
// original positions.
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 50.f),
outer_sticky_impl->ScreenSpaceTransform().To2dTranslation());
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 50.f),
inner_sticky_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll less than the sticking point. Both sticky elements should move with
// scroll as we haven't gotten to the sticky item locations yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 5.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 45.f),
outer_sticky_impl->ScreenSpaceTransform().To2dTranslation());
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 45.f),
inner_sticky_impl->ScreenSpaceTransform().To2dTranslation());
// Scroll such that the inner sticky should stick, but the outer one should
// keep going as it hasn't reached its position yet.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 30.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 20.f),
outer_sticky_impl->ScreenSpaceTransform().To2dTranslation());
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 25.f),
inner_sticky_impl->ScreenSpaceTransform().To2dTranslation());
// Keep going, both should stick.
SetScrollOffsetDelta(scroller_impl, gfx::Vector2dF(0.f, 100.f));
ExecuteCalculateDrawProperties(root_impl);
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 10.f),
outer_sticky_impl->ScreenSpaceTransform().To2dTranslation());
EXPECT_VECTOR2DF_EQ(
gfx::Vector2dF(0.f, 25.f),
inner_sticky_impl->ScreenSpaceTransform().To2dTranslation());
}
TEST_F(LayerTreeHostCommonTest, NonFlatContainerForFixedPosLayer) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> fixed_pos = Layer::Create();
scroller->SetIsContainerForFixedPositionLayers(true);
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(fixed_pos);
host()->SetRootLayer(root);
LayerPositionConstraint fixed_position;
fixed_position.set_is_fixed_position(true);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
fixed_pos->SetPositionConstraint(fixed_position);
root->SetBounds(gfx::Size(50, 50));
container->SetBounds(gfx::Size(50, 50));
scroller->SetBounds(gfx::Size(50, 50));
scroller->SetScrollable(container->bounds());
fixed_pos->SetBounds(gfx::Size(50, 50));
gfx::Transform rotate;
rotate.RotateAboutXAxis(20);
container->SetTransform(rotate);
ExecuteCalculateDrawProperties(root.get());
TransformTree& tree =
root->layer_tree_host()->property_trees()->transform_tree;
gfx::Transform transform;
tree.ComputeTranslation(fixed_pos->transform_tree_index(),
container->transform_tree_index(), &transform);
EXPECT_TRUE(transform.IsIdentity());
}
TEST_F(LayerTreeHostCommonTest, ScrollSnappingWithFixedPosChild) {
// This test verifies that a fixed pos child of a scrolling layer doesn't get
// snapped to integer coordinates.
//
// + root
// + container
// + scroller
// + fixed_pos
//
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> container = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> fixed_pos = Layer::Create();
scroller->SetIsContainerForFixedPositionLayers(true);
root->AddChild(container);
container->AddChild(scroller);
scroller->AddChild(fixed_pos);
host()->SetRootLayer(root);
LayerPositionConstraint fixed_position;
fixed_position.set_is_fixed_position(true);
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
fixed_pos->SetPositionConstraint(fixed_position);
root->SetBounds(gfx::Size(50, 50));
container->SetBounds(gfx::Size(50, 50));
scroller->SetBounds(gfx::Size(100, 100));
scroller->SetScrollable(container->bounds());
scroller->SetPosition(gfx::PointF(10.3f, 10.3f));
fixed_pos->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawProperties(root.get());
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* scroller_impl = layer_tree_impl->LayerById(scroller->id());
LayerImpl* fixed_pos_impl = layer_tree_impl->LayerById(fixed_pos->id());
gfx::Vector2dF scroll_delta(5.f, 9.f);
SetScrollOffsetDelta(scroller_impl, scroll_delta);
ExecuteCalculateDrawProperties(root_impl);
gfx::Vector2dF expected_scroller_screen_space_transform_translation(5.f, 1.f);
EXPECT_VECTOR2DF_EQ(expected_scroller_screen_space_transform_translation,
scroller_impl->ScreenSpaceTransform().To2dTranslation());
gfx::Vector2dF expected_fixed_pos_screen_space_transform_translation(10.3f,
10.3f);
EXPECT_VECTOR2DF_EQ(expected_fixed_pos_screen_space_transform_translation,
fixed_pos_impl->ScreenSpaceTransform().To2dTranslation());
}
class AnimationScaleFactorTrackingLayerImpl : public LayerImpl {
public:
static std::unique_ptr<AnimationScaleFactorTrackingLayerImpl> Create(
LayerTreeImpl* tree_impl,
int id) {
return base::WrapUnique(
new AnimationScaleFactorTrackingLayerImpl(tree_impl, id));
}
~AnimationScaleFactorTrackingLayerImpl() override = default;
private:
explicit AnimationScaleFactorTrackingLayerImpl(LayerTreeImpl* tree_impl,
int id)
: LayerImpl(tree_impl, id) {
SetDrawsContent(true);
}
};
TEST_F(LayerTreeHostCommonTest, MaximumAnimationScaleFactor) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
LayerTreeSettings settings = host()->GetSettings();
settings.layer_transforms_should_scale_layer_contents = true;
FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider,
&task_graph_runner);
std::unique_ptr<AnimationScaleFactorTrackingLayerImpl> grand_parent =
AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<AnimationScaleFactorTrackingLayerImpl> parent =
AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 2);
std::unique_ptr<AnimationScaleFactorTrackingLayerImpl> child =
AnimationScaleFactorTrackingLayerImpl::Create(host_impl.active_tree(), 3);
std::unique_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();
AnimationScaleFactorTrackingLayerImpl* grand_parent_raw = grand_parent.get();
grand_parent->SetBounds(gfx::Size(1, 2));
parent->SetBounds(gfx::Size(1, 2));
child->SetBounds(gfx::Size(1, 2));
grand_child->SetBounds(gfx::Size(1, 2));
child->test_properties()->AddChild(std::move(grand_child));
parent->test_properties()->AddChild(std::move(child));
grand_parent->test_properties()->AddChild(std::move(parent));
host_impl.active_tree()->SetRootLayerForTesting(std::move(grand_parent));
ExecuteCalculateDrawProperties(grand_parent_raw);
// No layers have animations.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
TransformOperations translation;
translation.AppendTranslate(1.f, 2.f, 3.f);
scoped_refptr<AnimationTimeline> timeline;
timeline = AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl.animation_host()->AddAnimationTimeline(timeline);
host_impl.active_tree()->SetElementIdsForTesting();
scoped_refptr<SingleKeyframeEffectAnimation> grand_parent_animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline->AttachAnimation(grand_parent_animation);
grand_parent_animation->AttachElement(grand_parent_raw->element_id());
scoped_refptr<SingleKeyframeEffectAnimation> parent_animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline->AttachAnimation(parent_animation);
parent_animation->AttachElement(parent_raw->element_id());
scoped_refptr<SingleKeyframeEffectAnimation> child_animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline->AttachAnimation(child_animation);
child_animation->AttachElement(child_raw->element_id());
scoped_refptr<SingleKeyframeEffectAnimation> grand_child_animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline->AttachAnimation(grand_child_animation);
grand_child_animation->AttachElement(grand_child_raw->element_id());
AddAnimatedTransformToAnimation(parent_animation.get(), 1.0,
TransformOperations(), translation);
// No layers have scale-affecting animations.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
TransformOperations scale;
scale.AppendScale(5.f, 4.f, 3.f);
AddAnimatedTransformToAnimation(child_animation.get(), 1.0,
TransformOperations(), scale);
child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// Only |child| has a scale-affecting animation.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(5.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(5.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(1.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(1.f, GetStartingAnimationScale(grand_child_raw));
AddAnimatedTransformToAnimation(grand_parent_animation.get(), 1.0,
TransformOperations(), scale);
grand_parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |grand_parent| and |child| have scale-affecting animations.
EXPECT_EQ(5.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(5.f, GetMaximumAnimationScale(parent_raw));
// 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, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(1.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(1.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
AddAnimatedTransformToAnimation(parent_animation.get(), 1.0,
TransformOperations(), scale);
parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |grand_parent|, |parent|, and |child| have scale-affecting animations.
EXPECT_EQ(5.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(1.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
grand_parent_animation->AbortKeyframeModelsWithProperty(
TargetProperty::TRANSFORM, false);
parent_animation->AbortKeyframeModelsWithProperty(TargetProperty::TRANSFORM,
false);
child_animation->AbortKeyframeModelsWithProperty(TargetProperty::TRANSFORM,
false);
TransformOperations perspective;
perspective.AppendPerspective(10.f);
AddAnimatedTransformToAnimation(child_animation.get(), 1.0,
TransformOperations(), perspective);
child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |child| has a scale-affecting animation but computing the maximum of this
// animation is not supported.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
child_animation->AbortKeyframeModelsWithProperty(TargetProperty::TRANSFORM,
false);
gfx::Transform scale_matrix;
scale_matrix.Scale(1.f, 2.f);
grand_parent_raw->test_properties()->transform = scale_matrix;
parent_raw->test_properties()->transform = scale_matrix;
grand_parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
AddAnimatedTransformToAnimation(parent_animation.get(), 1.0,
TransformOperations(), scale);
ExecuteCalculateDrawProperties(grand_parent_raw);
// |grand_parent| and |parent| each have scale 2.f. |parent| has a scale
// animation with maximum scale 5.f.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(10.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(10.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(10.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(2.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(2.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(2.f, GetStartingAnimationScale(grand_child_raw));
gfx::Transform perspective_matrix;
perspective_matrix.ApplyPerspectiveDepth(2.f);
child_raw->test_properties()->transform = perspective_matrix;
grand_parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |child| has a transform that's neither a translation nor a scale.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(10.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(2.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
parent_raw->test_properties()->transform = perspective_matrix;
grand_parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |parent| and |child| have transforms that are neither translations nor
// scales.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
parent_raw->test_properties()->transform = gfx::Transform();
child_raw->test_properties()->transform = gfx::Transform();
grand_parent_raw->test_properties()->transform = perspective_matrix;
grand_parent_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
// |grand_parent| has a transform that's neither a translation nor a scale.
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(child_raw));
EXPECT_EQ(0.f, GetMaximumAnimationScale(grand_child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(parent_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(child_raw));
EXPECT_EQ(0.f, GetStartingAnimationScale(grand_child_raw));
}
static void GatherDrawnLayers(LayerTreeImpl* tree_impl,
std::set<LayerImpl*>* drawn_layers) {
for (EffectTreeLayerListIterator it(tree_impl);
it.state() != EffectTreeLayerListIterator::State::END; ++it) {
if (it.state() == EffectTreeLayerListIterator::State::LAYER)
drawn_layers->insert(it.current_layer());
if (it.state() != EffectTreeLayerListIterator::State::CONTRIBUTING_SURFACE)
continue;
if (it.current_render_surface()->MaskLayer())
drawn_layers->insert(it.current_render_surface()->MaskLayer());
}
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceLayerListMembership) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> grand_parent =
LayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<LayerImpl> parent =
LayerImpl::Create(host_impl.active_tree(), 3);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.active_tree(), 5);
std::unique_ptr<LayerImpl> grand_child1 =
LayerImpl::Create(host_impl.active_tree(), 7);
std::unique_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();
grand_parent->SetBounds(gfx::Size(1, 2));
parent->SetBounds(gfx::Size(1, 2));
child->SetBounds(gfx::Size(1, 2));
grand_child1->SetBounds(gfx::Size(1, 2));
grand_child2->SetBounds(gfx::Size(1, 2));
child->test_properties()->AddChild(std::move(grand_child1));
child->test_properties()->AddChild(std::move(grand_child2));
parent->test_properties()->AddChild(std::move(child));
grand_parent->test_properties()->AddChild(std::move(parent));
host_impl.active_tree()->SetRootLayerForTesting(std::move(grand_parent));
// Start with nothing being drawn.
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child2_raw->contributes_to_drawn_render_surface());
std::set<LayerImpl*> expected;
std::set<LayerImpl*> actual;
GatherDrawnLayers(host_impl.active_tree(), &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->test_properties()->force_render_surface = true;
grand_child1_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &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);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
expected.insert(grand_child1_raw);
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &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->test_properties()->force_render_surface = false;
grand_child1_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
child_raw->test_properties()->force_render_surface = true;
grand_child2_raw->SetDrawsContent(true);
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
expected.insert(grand_child2_raw);
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &actual);
EXPECT_EQ(expected, actual);
// Add a mask layer to child.
child_raw->test_properties()->SetMaskLayer(
LayerImpl::Create(host_impl.active_tree(), 6));
child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_raw->test_properties()
->mask_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
expected.insert(grand_child2_raw);
expected.insert(child_raw->test_properties()->mask_layer);
expected.clear();
expected.insert(grand_child2_raw);
expected.insert(child_raw->test_properties()->mask_layer);
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &actual);
EXPECT_EQ(expected, actual);
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_raw->test_properties()
->mask_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
expected.insert(grand_child2_raw);
expected.insert(child_raw->test_properties()->mask_layer);
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &actual);
EXPECT_EQ(expected, actual);
child_raw->layer_tree_impl()->property_trees()->needs_rebuild = true;
// With nothing drawing, we should have no layers.
grand_child2_raw->SetDrawsContent(false);
ExecuteCalculateDrawProperties(grand_parent_raw);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(child_raw->test_properties()
->mask_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &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);
EXPECT_FALSE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_raw->test_properties()
->mask_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_FALSE(grand_child2_raw->contributes_to_drawn_render_surface());
expected.clear();
expected.insert(child_raw);
expected.insert(child_raw->test_properties()->mask_layer);
actual.clear();
GatherDrawnLayers(host_impl.active_tree(), &actual);
EXPECT_EQ(expected, actual);
child_raw->test_properties()->SetMaskLayer(nullptr);
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);
EXPECT_TRUE(grand_parent_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(parent_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(child_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child1_raw->contributes_to_drawn_render_surface());
EXPECT_TRUE(grand_child2_raw->contributes_to_drawn_render_surface());
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(host_impl.active_tree(), &actual);
EXPECT_EQ(expected, actual);
}
TEST_F(LayerTreeHostCommonTest, DrawPropertyScales) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
LayerTreeSettings settings = host()->GetSettings();
settings.layer_transforms_should_scale_layer_contents = true;
FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider,
&task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> child1 =
LayerImpl::Create(host_impl.active_tree(), 2);
LayerImpl* child1_layer = child1.get();
std::unique_ptr<LayerImpl> child2 =
LayerImpl::Create(host_impl.active_tree(), 3);
LayerImpl* child2_layer = child2.get();
gfx::Transform scale_transform_child1, scale_transform_child2;
scale_transform_child1.Scale(2, 3);
scale_transform_child2.Scale(4, 5);
root->SetBounds(gfx::Size(1, 1));
root->SetDrawsContent(true);
child1_layer->test_properties()->transform = scale_transform_child1;
child1_layer->SetBounds(gfx::Size(1, 1));
child1_layer->SetDrawsContent(true);
child1_layer->test_properties()->SetMaskLayer(
LayerImpl::Create(host_impl.active_tree(), 4));
root->test_properties()->AddChild(std::move(child1));
root->test_properties()->AddChild(std::move(child2));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.active_tree()->SetElementIdsForTesting();
ExecuteCalculateDrawProperties(root_layer);
TransformOperations scale;
scale.AppendScale(5.f, 8.f, 3.f);
scoped_refptr<AnimationTimeline> timeline =
AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl.animation_host()->AddAnimationTimeline(timeline);
child2_layer->test_properties()->transform = scale_transform_child2;
child2_layer->SetBounds(gfx::Size(1, 1));
child2_layer->SetDrawsContent(true);
AddAnimatedTransformToElementWithAnimation(
child2_layer->element_id(), timeline, 1.0, TransformOperations(), scale);
root_layer->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->test_properties()->mask_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(5.f, child2_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(child1_layer));
EXPECT_FLOAT_EQ(8.f, GetMaximumAnimationScale(child2_layer));
// 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;
RenderSurfaceList render_surface_list;
gfx::Size device_viewport_size =
gfx::Size(root_layer->bounds().width() * device_scale_factor,
root_layer->bounds().height() * device_scale_factor);
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, device_viewport_size, &render_surface_list);
inputs.page_scale_factor = page_scale_factor;
inputs.can_adjust_raster_scales = true;
inputs.page_scale_layer = root_layer;
inputs.page_scale_transform_node = inputs.property_trees->transform_tree.Node(
inputs.page_scale_layer->transform_tree_index());
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_FLOAT_EQ(3.f, root_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(9.f, child1_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(
9.f,
child1_layer->test_properties()->mask_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(15.f, child2_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(child1_layer));
EXPECT_FLOAT_EQ(24.f, GetMaximumAnimationScale(child2_layer));
// 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->layer_tree_impl()->property_trees()->needs_rebuild = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_FLOAT_EQ(12.f, root_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(36.f, child1_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(
36.f,
child1_layer->test_properties()->mask_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(60.f, child2_layer->GetIdealContentsScale());
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(child1_layer));
EXPECT_FLOAT_EQ(96.f, GetMaximumAnimationScale(child2_layer));
}
TEST_F(LayerTreeHostCommonTest, AnimationScales) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
LayerTreeSettings settings = host()->GetSettings();
settings.layer_transforms_should_scale_layer_contents = true;
FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider,
&task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> child1 =
LayerImpl::Create(host_impl.active_tree(), 2);
LayerImpl* child1_layer = child1.get();
std::unique_ptr<LayerImpl> child2 =
LayerImpl::Create(host_impl.active_tree(), 3);
LayerImpl* child2_layer = child2.get();
root->test_properties()->AddChild(std::move(child1));
child1_layer->test_properties()->AddChild(std::move(child2));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.active_tree()->SetElementIdsForTesting();
gfx::Transform scale_transform_child1, scale_transform_child2;
scale_transform_child1.Scale(2, 3);
scale_transform_child2.Scale(4, 5);
root_layer->SetBounds(gfx::Size(1, 1));
child1_layer->test_properties()->transform = scale_transform_child1;
child1_layer->SetBounds(gfx::Size(1, 1));
child2_layer->test_properties()->transform = scale_transform_child2;
child2_layer->SetBounds(gfx::Size(1, 1));
TransformOperations scale;
scale.AppendScale(5.f, 8.f, 3.f);
scoped_refptr<AnimationTimeline> timeline =
AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl.animation_host()->AddAnimationTimeline(timeline);
AddAnimatedTransformToElementWithAnimation(
child2_layer->element_id(), timeline, 1.0, TransformOperations(), scale);
// Correctly computes animation scale when rebuilding property trees.
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_layer);
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(child1_layer));
EXPECT_FLOAT_EQ(24.f, GetMaximumAnimationScale(child2_layer));
EXPECT_FLOAT_EQ(0.f, GetStartingAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetStartingAnimationScale(child1_layer));
EXPECT_FLOAT_EQ(3.f, GetStartingAnimationScale(child2_layer));
// Correctly updates animation scale when layer property changes.
child1_layer->test_properties()->transform = gfx::Transform();
root_layer->layer_tree_impl()->SetTransformMutated(child1_layer->element_id(),
gfx::Transform());
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = false;
ExecuteCalculateDrawProperties(root_layer);
EXPECT_FLOAT_EQ(8.f, GetMaximumAnimationScale(child2_layer));
EXPECT_FLOAT_EQ(1.f, GetStartingAnimationScale(child2_layer));
// Do not update animation scale if already updated.
host_impl.active_tree()->property_trees()->SetAnimationScalesForTesting(
child2_layer->transform_tree_index(), 100.f, 100.f);
EXPECT_FLOAT_EQ(100.f, GetMaximumAnimationScale(child2_layer));
EXPECT_FLOAT_EQ(100.f, GetStartingAnimationScale(child2_layer));
}
TEST_F(LayerTreeHostCommonTest,
AnimationScaleWhenLayerTransformShouldNotScaleLayerBounds) {
// Returns empty scale if layer_transforms_should_scale_layer_contents is
// false.
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
LayerTreeSettings settings = host()->GetSettings();
settings.layer_transforms_should_scale_layer_contents = false;
FakeLayerTreeHostImpl host_impl(settings, &task_runner_provider,
&task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.active_tree(), 2);
LayerImpl* child_layer = child.get();
root->test_properties()->AddChild(std::move(child));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.active_tree()->SetElementIdsForTesting();
gfx::Transform scale_transform_child;
scale_transform_child.Scale(4, 5);
root_layer->SetBounds(gfx::Size(1, 1));
child_layer->test_properties()->transform = scale_transform_child;
child_layer->SetBounds(gfx::Size(1, 1));
TransformOperations scale;
scale.AppendScale(5.f, 8.f, 3.f);
scoped_refptr<AnimationTimeline> timeline =
AnimationTimeline::Create(AnimationIdProvider::NextTimelineId());
host_impl.animation_host()->AddAnimationTimeline(timeline);
AddAnimatedTransformToElementWithAnimation(
child_layer->element_id(), timeline, 1.0, TransformOperations(), scale);
root_layer->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root_layer);
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetMaximumAnimationScale(child_layer));
EXPECT_FLOAT_EQ(0.f, GetStartingAnimationScale(root_layer));
EXPECT_FLOAT_EQ(0.f, GetStartingAnimationScale(child_layer));
}
TEST_F(LayerTreeHostCommonTest, VisibleContentRectInChildRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip = AddChild<LayerImpl>(root);
LayerImpl* content = AddChild<LayerImpl>(clip);
root->SetBounds(gfx::Size(768 / 2, 3000));
root->SetDrawsContent(true);
clip->SetBounds(gfx::Size(768 / 2, 10000));
clip->SetMasksToBounds(true);
content->SetBounds(gfx::Size(768 / 2, 10000));
content->SetDrawsContent(true);
content->test_properties()->force_render_surface = true;
gfx::Size device_viewport_size(768, 582);
RenderSurfaceList render_surface_list_impl;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, device_viewport_size, gfx::Transform(), &render_surface_list_impl);
inputs.device_scale_factor = 2.f;
inputs.page_scale_factor = 1.f;
inputs.page_scale_layer = nullptr;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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_layer_rect());
// 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_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, ViewportBoundsDeltaAffectVisibleContentRect) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &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 browser controls.
gfx::Size device_viewport_size = gfx::Size(300, 600);
host_impl.active_tree()->SetDeviceViewportSize(device_viewport_size);
host_impl.active_tree()->SetRootLayerForTesting(
LayerImpl::Create(host_impl.active_tree(), 1));
LayerImpl* root = host_impl.active_tree()->root_layer_for_testing();
root->SetBounds(root_size);
root->SetMasksToBounds(true);
// Make root the inner viewport scroll layer. This ensures the later call to
// |SetViewportBoundsDelta| will be on a viewport layer.
LayerTreeImpl::ViewportLayerIds viewport_ids;
viewport_ids.inner_viewport_scroll = root->id();
host_impl.active_tree()->SetViewportLayersFromIds(viewport_ids);
root->test_properties()->AddChild(
LayerImpl::Create(host_impl.active_tree(), 2));
LayerImpl* sublayer = root->test_properties()->children[0];
sublayer->SetBounds(sublayer_size);
sublayer->SetDrawsContent(true);
host_impl.active_tree()->BuildPropertyTreesForTesting();
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, device_viewport_size, &render_surface_list);
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
EXPECT_EQ(gfx::Rect(root_size), sublayer->visible_layer_rect());
root->SetViewportBoundsDelta(gfx::Vector2dF(0.0, 50.0));
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&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, NodesAffectedByViewportBoundsDeltaGetUpdated) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> page_scale_layer = Layer::Create();
scoped_refptr<Layer> inner_viewport_container_layer = Layer::Create();
scoped_refptr<Layer> inner_viewport_scroll_layer = Layer::Create();
scoped_refptr<Layer> outer_viewport_container_layer = Layer::Create();
scoped_refptr<Layer> outer_viewport_scroll_layer = Layer::Create();
root->AddChild(inner_viewport_container_layer);
inner_viewport_container_layer->AddChild(page_scale_layer);
page_scale_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->SetIsContainerForFixedPositionLayers(true);
outer_viewport_scroll_layer->SetIsContainerForFixedPositionLayers(true);
outer_viewport_scroll_layer->SetIsResizedByBrowserControls(true);
host()->SetRootLayer(root);
ViewportLayers viewport_layers;
viewport_layers.page_scale = page_scale_layer;
viewport_layers.inner_viewport_container = inner_viewport_container_layer;
viewport_layers.outer_viewport_container = outer_viewport_container_layer;
viewport_layers.inner_viewport_scroll = inner_viewport_scroll_layer;
viewport_layers.outer_viewport_scroll = outer_viewport_scroll_layer;
host()->RegisterViewportLayers(viewport_layers);
scoped_refptr<Layer> fixed_to_inner = Layer::Create();
scoped_refptr<Layer> fixed_to_outer = Layer::Create();
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);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
TransformTree& transform_tree = host()->property_trees()->transform_tree;
EXPECT_TRUE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta());
LayerPositionConstraint fixed_to_left;
fixed_to_left.set_is_fixed_position(true);
fixed_to_inner->SetPositionConstraint(fixed_to_left);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_TRUE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta());
fixed_to_outer->SetPositionConstraint(fixed_to_left);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_FALSE(transform_tree.HasNodesAffectedByOuterViewportBoundsDelta());
}
TEST_F(LayerTreeHostCommonTest, VisibleContentRectForAnimatedLayer) {
host_impl()->CreatePendingTree();
std::unique_ptr<LayerImpl> pending_root =
LayerImpl::Create(host_impl()->pending_tree(), 1);
LayerImpl* root = pending_root.get();
host_impl()->pending_tree()->SetRootLayerForTesting(std::move(pending_root));
std::unique_ptr<LayerImpl> animated_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 2);
LayerImpl* animated = animated_ptr.get();
root->test_properties()->AddChild(std::move(animated_ptr));
animated->SetDrawsContent(true);
host_impl()->pending_tree()->SetElementIdsForTesting();
root->SetBounds(gfx::Size(100, 100));
root->SetMasksToBounds(true);
root->test_properties()->force_render_surface = true;
animated->test_properties()->opacity = 0.f;
animated->SetBounds(gfx::Size(20, 20));
AddOpacityTransitionToElementWithAnimation(
animated->element_id(), timeline_impl(), 10.0, 0.f, 1.f, false);
animated->test_properties()->opacity_can_animate = true;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root);
EXPECT_FALSE(animated->visible_layer_rect().IsEmpty());
}
TEST_F(LayerTreeHostCommonTest,
VisibleContentRectForAnimatedLayerWithSingularTransform) {
host_impl()->CreatePendingTree();
std::unique_ptr<LayerImpl> root_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 1);
LayerImpl* root = root_ptr.get();
host_impl()->pending_tree()->SetRootLayerForTesting(std::move(root_ptr));
std::unique_ptr<LayerImpl> clip_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 2);
LayerImpl* clip = clip_ptr.get();
root->test_properties()->AddChild(std::move(clip_ptr));
std::unique_ptr<LayerImpl> animated_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 3);
LayerImpl* animated = animated_ptr.get();
clip->test_properties()->AddChild(std::move(animated_ptr));
std::unique_ptr<LayerImpl> surface_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 4);
LayerImpl* surface = surface_ptr.get();
animated->test_properties()->AddChild(std::move(surface_ptr));
std::unique_ptr<LayerImpl> descendant_of_keyframe_model_ptr =
LayerImpl::Create(host_impl()->pending_tree(), 5);
LayerImpl* descendant_of_keyframe_model =
descendant_of_keyframe_model_ptr.get();
surface->test_properties()->AddChild(
std::move(descendant_of_keyframe_model_ptr));
host_impl()->pending_tree()->SetElementIdsForTesting();
root->SetDrawsContent(true);
animated->SetDrawsContent(true);
surface->SetDrawsContent(true);
descendant_of_keyframe_model->SetDrawsContent(true);
gfx::Transform uninvertible_matrix;
uninvertible_matrix.Scale3d(6.f, 6.f, 0.f);
root->SetBounds(gfx::Size(100, 100));
clip->SetBounds(gfx::Size(10, 10));
clip->SetMasksToBounds(true);
animated->test_properties()->transform = uninvertible_matrix;
animated->SetBounds(gfx::Size(120, 120));
surface->SetBounds(gfx::Size(100, 100));
surface->test_properties()->force_render_surface = true;
descendant_of_keyframe_model->SetBounds(gfx::Size(200, 200));
TransformOperations start_transform_operations;
start_transform_operations.AppendMatrix(uninvertible_matrix);
TransformOperations end_transform_operations;
AddAnimatedTransformToElementWithAnimation(
animated->element_id(), timeline_impl(), 10.0, start_transform_operations,
end_transform_operations);
ExecuteCalculateDrawProperties(root);
// Since animated has singular transform, it is not be part of render
// surface layer list but should be rastered.
EXPECT_FALSE(animated->contributes_to_drawn_render_surface());
EXPECT_TRUE(animated->raster_even_if_not_drawn());
// 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_layer_rect());
// 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_layer_rect());
EXPECT_EQ(gfx::Rect(2, 2),
descendant_of_keyframe_model->visible_layer_rect());
gfx::Transform zero_matrix;
zero_matrix.Scale3d(0.f, 0.f, 0.f);
root->layer_tree_impl()->SetTransformMutated(animated->element_id(),
zero_matrix);
ExecuteCalculateDrawProperties(root);
// The animated layer will be treated as fully visible when we combine clips
// in screen space.
EXPECT_EQ(gfx::Rect(120, 120), animated->visible_layer_rect());
// 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(100, 100), surface->visible_layer_rect());
EXPECT_EQ(gfx::Rect(200, 200),
descendant_of_keyframe_model->visible_layer_rect());
host_impl()->ActivateSyncTree();
LayerImpl* active_root = host_impl()->active_tree()->LayerById(root->id());
ExecuteCalculateDrawProperties(active_root);
// Since the animated has singular transform, it is not be part of render
// surface layer list.
LayerImpl* active_animated =
host_impl()->active_tree()->LayerById(animated->id());
EXPECT_TRUE(active_root->contributes_to_drawn_render_surface());
EXPECT_FALSE(active_animated->contributes_to_drawn_render_surface());
// Since animated has singular transform, it is not be part of render
// surface layer list but should be rastered.
EXPECT_TRUE(animated->raster_even_if_not_drawn());
EXPECT_EQ(gfx::Rect(120, 120), active_animated->visible_layer_rect());
}
// Verify that having animated opacity but current opacity 1 still creates
// a render surface.
TEST_F(LayerTreeHostCommonTest, AnimatedOpacityCreatesRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grandchild = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
child->SetBounds(gfx::Size(50, 50));
child->SetDrawsContent(true);
grandchild->SetBounds(gfx::Size(50, 50));
grandchild->SetDrawsContent(true);
SetElementIdsForTesting();
AddOpacityTransitionToElementWithAnimation(
child->element_id(), timeline_impl(), 10.0, 1.f, 0.2f, false);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1.f, child->Opacity());
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(grandchild), GetRenderSurface(child));
}
static bool FilterIsAnimating(LayerImpl* layer) {
MutatorHost* host = layer->layer_tree_impl()->mutator_host();
return host->IsAnimatingFilterProperty(layer->element_id(),
layer->GetElementTypeForAnimation());
}
// Verify that having an animated filter (but no current filter, as these
// are mutually exclusive) correctly creates a render surface.
TEST_F(LayerTreeHostCommonTest, AnimatedFilterCreatesRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grandchild = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
child->SetBounds(gfx::Size(50, 50));
grandchild->SetBounds(gfx::Size(50, 50));
SetElementIdsForTesting();
AddAnimatedFilterToElementWithAnimation(child->element_id(), timeline_impl(),
10.0, 0.1f, 0.2f);
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(grandchild), GetRenderSurface(child));
EXPECT_TRUE(GetRenderSurface(root)->Filters().IsEmpty());
EXPECT_TRUE(GetRenderSurface(child)->Filters().IsEmpty());
EXPECT_FALSE(FilterIsAnimating(root));
EXPECT_TRUE(FilterIsAnimating(child));
EXPECT_FALSE(FilterIsAnimating(grandchild));
}
bool HasPotentiallyRunningFilterAnimation(const LayerImpl& layer) {
MutatorHost* host = layer.layer_tree_impl()->mutator_host();
return host->HasPotentiallyRunningFilterAnimation(
layer.element_id(), layer.GetElementTypeForAnimation());
}
// Verify that having a filter animation with a delayed start time creates a
// render surface.
TEST_F(LayerTreeHostCommonTest, DelayedFilterAnimationCreatesRenderSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grandchild = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
child->SetBounds(gfx::Size(50, 50));
grandchild->SetBounds(gfx::Size(50, 50));
SetElementIdsForTesting();
std::unique_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));
std::unique_ptr<KeyframeModel> keyframe_model =
KeyframeModel::Create(std::move(curve), 0, 1, TargetProperty::FILTER);
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
AddKeyframeModelToElementWithAnimation(child->element_id(), timeline_impl(),
std::move(keyframe_model));
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_NE(GetRenderSurface(child), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(grandchild), GetRenderSurface(child));
EXPECT_TRUE(GetRenderSurface(root)->Filters().IsEmpty());
EXPECT_TRUE(GetRenderSurface(child)->Filters().IsEmpty());
EXPECT_FALSE(FilterIsAnimating(root));
EXPECT_FALSE(HasPotentiallyRunningFilterAnimation(*root));
EXPECT_FALSE(FilterIsAnimating(child));
EXPECT_TRUE(HasPotentiallyRunningFilterAnimation(*child));
EXPECT_FALSE(FilterIsAnimating(grandchild));
EXPECT_FALSE(HasPotentiallyRunningFilterAnimation(*grandchild));
}
// Ensures that the property tree code accounts for offsets between fixed
// position layers and their respective containers.
TEST_F(LayerTreeHostCommonTest, PropertyTreesAccountForFixedParentOffset) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grandchild = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
root->SetMasksToBounds(true);
root->test_properties()->is_container_for_fixed_position_layers = true;
child->test_properties()->position = gfx::PointF(1000, 1000);
child->SetBounds(gfx::Size(50, 50));
grandchild->test_properties()->position = gfx::PointF(-1000, -1000);
grandchild->SetBounds(gfx::Size(50, 50));
grandchild->SetDrawsContent(true);
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
grandchild->test_properties()->position_constraint = constraint;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0, 50, 50), grandchild->visible_layer_rect());
}
// 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) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grandchild = AddChild<LayerImpl>(child);
root->SetBounds(gfx::Size(50, 50));
root->SetMasksToBounds(true);
root->test_properties()->is_container_for_fixed_position_layers = true;
child->test_properties()->position = gfx::PointF(1000, 1000);
child->SetBounds(gfx::Size(50, 50));
child->test_properties()->is_container_for_fixed_position_layers = true;
grandchild->test_properties()->position = gfx::PointF(-1000, -1000);
grandchild->SetBounds(gfx::Size(50, 50));
grandchild->SetDrawsContent(true);
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
grandchild->test_properties()->position_constraint = constraint;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0, 50, 50), grandchild->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, OnlyApplyFixedPositioningOnce) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* frame_clip = AddChild<LayerImpl>(root);
LayerImpl* fixed = AddChild<LayerImpl>(frame_clip);
gfx::Transform translate_z;
translate_z.Translate3d(0, 0, 10);
root->SetBounds(gfx::Size(800, 800));
root->test_properties()->is_container_for_fixed_position_layers = true;
frame_clip->test_properties()->position = gfx::PointF(500, 100);
frame_clip->SetBounds(gfx::Size(100, 100));
frame_clip->SetMasksToBounds(true);
fixed->SetBounds(gfx::Size(1000, 1000));
fixed->SetDrawsContent(true);
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
fixed->test_properties()->position_constraint = constraint;
ExecuteCalculateDrawProperties(root);
gfx::Rect expected(0, 0, 100, 100);
EXPECT_EQ(expected, fixed->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, FixedClipsShouldBeAssociatedWithTheRightNode) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* frame_clip = AddChild<LayerImpl>(root);
LayerImpl* scroller = AddChild<LayerImpl>(frame_clip);
LayerImpl* fixed = AddChild<LayerImpl>(scroller);
root->SetBounds(gfx::Size(800, 800));
root->SetDrawsContent(true);
root->test_properties()->is_container_for_fixed_position_layers = true;
frame_clip->test_properties()->position = gfx::PointF(500, 100);
frame_clip->SetBounds(gfx::Size(100, 100));
frame_clip->SetMasksToBounds(true);
frame_clip->SetDrawsContent(true);
scroller->SetBounds(gfx::Size(1000, 1000));
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
scroller->SetElementId(LayerIdToElementIdForTesting(scroller->id()));
scroller->SetScrollable(frame_clip->bounds());
scroller->SetDrawsContent(true);
fixed->test_properties()->position = gfx::PointF(100, 100);
fixed->SetBounds(gfx::Size(50, 50));
fixed->SetMasksToBounds(true);
fixed->SetDrawsContent(true);
fixed->test_properties()->force_render_surface = true;
BuildPropertyTreesForTesting();
scroller->SetCurrentScrollOffset(gfx::ScrollOffset(100, 100));
LayerPositionConstraint constraint;
constraint.set_is_fixed_position(true);
fixed->test_properties()->position_constraint = constraint;
ExecuteCalculateDrawProperties(root);
gfx::Rect expected(0, 0, 50, 50);
EXPECT_EQ(expected, fixed->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, ChangingAxisAlignmentTriggersRebuild) {
gfx::Transform translate;
gfx::Transform rotate;
translate.Translate(10, 10);
rotate.Rotate(45);
scoped_refptr<Layer> root = Layer::Create();
root->SetBounds(gfx::Size(800, 800));
root->SetIsContainerForFixedPositionLayers(true);
host()->SetRootLayer(root);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(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) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
gfx::Transform scale_matrix;
scale_matrix.Scale(2.f, 2.f);
root->SetBounds(gfx::Size(100, 100));
root->SetDrawsContent(true);
child->test_properties()->transform = scale_matrix;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(10, 10), child->visible_layer_rect());
child->test_properties()->transform_origin = gfx::Point3F(10.f, 10.f, 10.f);
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(5, 5, 5, 5), child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, UpdateScrollChildPosition) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* scroll_parent = AddChild<LayerImpl>(root);
LayerImpl* scroll_child = AddChild<LayerImpl>(scroll_parent);
gfx::Transform scale;
scale.Scale(2.f, 2.f);
root->SetBounds(gfx::Size(50, 50));
scroll_child->test_properties()->transform = scale;
scroll_child->SetBounds(gfx::Size(40, 40));
scroll_child->SetDrawsContent(true);
scroll_parent->SetBounds(gfx::Size(30, 30));
scroll_parent->SetScrollable(gfx::Size(50, 50));
scroll_parent->SetElementId(
LayerIdToElementIdForTesting(scroll_parent->id()));
scroll_parent->SetDrawsContent(true);
scroll_child->test_properties()->scroll_parent = scroll_parent;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(25, 25), scroll_child->visible_layer_rect());
scroll_child->test_properties()->position = gfx::PointF(0, -10.f);
scroll_parent->SetCurrentScrollOffset(gfx::ScrollOffset(0.f, 10.f));
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 5, 25, 25), scroll_child->visible_layer_rect());
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 10, 25, 25), scroll_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, HasCopyRequestsInTargetSubtree) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> child1 = Layer::Create();
scoped_refptr<Layer> child2 = Layer::Create();
scoped_refptr<Layer> grandchild = Layer::Create();
scoped_refptr<Layer> greatgrandchild = Layer::Create();
root->AddChild(child1);
root->AddChild(child2);
child1->AddChild(grandchild);
grandchild->AddChild(greatgrandchild);
host()->SetRootLayer(root);
child1->RequestCopyOfOutput(viz::CopyOutputRequest::CreateStubForTesting());
greatgrandchild->RequestCopyOfOutput(
viz::CopyOutputRequest::CreateStubForTesting());
child2->SetOpacity(0.f);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_TRUE(LayerSubtreeHasCopyRequest(root.get()));
EXPECT_TRUE(LayerSubtreeHasCopyRequest(child1.get()));
EXPECT_FALSE(LayerSubtreeHasCopyRequest(child2.get()));
EXPECT_TRUE(LayerSubtreeHasCopyRequest(grandchild.get()));
EXPECT_TRUE(LayerSubtreeHasCopyRequest(greatgrandchild.get()));
}
TEST_F(LayerTreeHostCommonTest, SkippingSubtreeMain) {
FakeContentLayerClient client;
scoped_refptr<Layer> root = Layer::Create();
client.set_bounds(root->bounds());
scoped_refptr<Layer> child = Layer::Create();
child->SetIsDrawable(true);
scoped_refptr<Layer> grandchild = Layer::Create();
grandchild->SetIsDrawable(true);
scoped_refptr<FakePictureLayer> greatgrandchild(
FakePictureLayer::Create(&client));
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(10, 10));
grandchild->SetBounds(gfx::Size(10, 10));
greatgrandchild->SetBounds(gfx::Size(10, 10));
root->AddChild(child);
child->AddChild(grandchild);
grandchild->AddChild(greatgrandchild);
host()->SetRootLayer(root);
host()->SetElementIdsForTesting();
// Check the non-skipped case.
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
const LayerList* update_list = GetUpdateLayerList();
EXPECT_TRUE(VerifyLayerInList(grandchild, update_list));
// 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.
gfx::Transform singular;
singular.matrix().set(0, 0, 0);
child->SetTransform(singular);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_FALSE(VerifyLayerInList(grandchild, update_list));
child->SetTransform(gfx::Transform());
child->SetHideLayerAndSubtree(true);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_FALSE(VerifyLayerInList(grandchild, update_list));
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 keyframe_model_id = 0;
std::unique_ptr<KeyframeModel> keyframe_model = KeyframeModel::Create(
std::unique_ptr<AnimationCurve>(new FakeTransformTransition(1.0)),
keyframe_model_id, 1, TargetProperty::TRANSFORM);
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
AddKeyframeModelToElementWithAnimation(child->element_id(), timeline(),
std::move(keyframe_model));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_TRUE(VerifyLayerInList(grandchild, update_list));
RemoveKeyframeModelFromElementWithExistingKeyframeEffect(
child->element_id(), timeline(), keyframe_model_id);
child->SetTransform(gfx::Transform());
child->SetOpacity(0.f);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_FALSE(VerifyLayerInList(grandchild, update_list));
// 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(
viz::CopyOutputRequest::CreateStubForTesting());
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_TRUE(VerifyLayerInList(grandchild, update_list));
// Add an opacity animation with a start delay.
keyframe_model_id = 1;
keyframe_model = KeyframeModel::Create(
std::unique_ptr<AnimationCurve>(new FakeFloatTransition(1.0, 0.f, 1.f)),
keyframe_model_id, 1, TargetProperty::OPACITY);
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
AddKeyframeModelToElementWithExistingKeyframeEffect(
child->element_id(), timeline(), std::move(keyframe_model));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
update_list = GetUpdateLayerList();
EXPECT_TRUE(VerifyLayerInList(grandchild, update_list));
}
TEST_F(LayerTreeHostCommonTest, SkippingLayerImpl) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.active_tree(), 1);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.active_tree(), 2);
std::unique_ptr<LayerImpl> grandchild =
LayerImpl::Create(host_impl.active_tree(), 3);
std::unique_ptr<FakePictureLayerImpl> greatgrandchild(
FakePictureLayerImpl::Create(host_impl.active_tree(), 4));
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
grandchild->SetBounds(gfx::Size(10, 10));
grandchild->SetDrawsContent(true);
greatgrandchild->SetDrawsContent(true);
LayerImpl* root_ptr = root.get();
LayerImpl* child_ptr = child.get();
LayerImpl* grandchild_ptr = grandchild.get();
child->test_properties()->AddChild(std::move(grandchild));
root->test_properties()->AddChild(std::move(child));
host_impl.active_tree()->SetRootLayerForTesting(std::move(root));
host_impl.active_tree()->SetElementIdsForTesting();
// Check the non-skipped case.
// ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
// EXPECT_EQ(gfx::Rect(10, 10), grandchild_ptr->visible_layer_rect());
// 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_layer_rect(gfx::Rect());
gfx::Transform singular;
singular.matrix().set(0, 0, 0);
// This line is used to make the results of skipping and not skipping layers
// different.
singular.matrix().set(0, 1, 1);
gfx::Transform rotate;
rotate.Rotate(90);
gfx::Transform rotate_back_and_translate;
rotate_back_and_translate.RotateAboutYAxis(180);
rotate_back_and_translate.Translate(-10, 0);
child_ptr->test_properties()->transform = singular;
host_impl.active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
child_ptr->test_properties()->transform = gfx::Transform();
child_ptr->test_properties()->hide_layer_and_subtree = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
child_ptr->test_properties()->hide_layer_and_subtree = false;
child_ptr->test_properties()->opacity = 0.f;
host_impl.active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
child_ptr->test_properties()->opacity = 1.f;
root_ptr->test_properties()->transform = singular;
// Force transform tree to have a node for child, so that ancestor's
// invertible transform can be tested.
child_ptr->test_properties()->transform = rotate;
host_impl.active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
root_ptr->test_properties()->transform = gfx::Transform();
child_ptr->test_properties()->transform = gfx::Transform();
root_ptr->test_properties()->opacity = 0.f;
child_ptr->test_properties()->opacity = 0.7f;
host_impl.active_tree()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
root_ptr->test_properties()->opacity = 1.f;
child_ptr->test_properties()->opacity = 0.f;
// 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_ptr->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
root_ptr->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(10, 10), grandchild_ptr->visible_layer_rect());
host_impl.active_tree()->property_trees()->effect_tree.ClearCopyRequests();
child_ptr->test_properties()->opacity = 1.f;
// A double sided render surface with backface visible should not be skipped
grandchild_ptr->set_visible_layer_rect(gfx::Rect());
child_ptr->test_properties()->force_render_surface = true;
child_ptr->test_properties()->double_sided = true;
child_ptr->test_properties()->transform = rotate_back_and_translate;
root_ptr->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(10, 10), grandchild_ptr->visible_layer_rect());
child_ptr->test_properties()->transform = gfx::Transform();
std::unique_ptr<KeyframedTransformAnimationCurve> curve(
KeyframedTransformAnimationCurve::Create());
TransformOperations start;
start.AppendTranslate(1.f, 2.f, 3.f);
gfx::Transform transform;
transform.Scale3d(1.0, 2.0, 3.0);
TransformOperations operation;
operation.AppendMatrix(transform);
curve->AddKeyframe(
TransformKeyframe::Create(base::TimeDelta(), start, nullptr));
curve->AddKeyframe(TransformKeyframe::Create(
base::TimeDelta::FromSecondsD(1.0), operation, nullptr));
std::unique_ptr<KeyframeModel> transform_animation(
KeyframeModel::Create(std::move(curve), 3, 3, TargetProperty::TRANSFORM));
scoped_refptr<SingleKeyframeEffectAnimation> animation(
SingleKeyframeEffectAnimation::Create(1));
timeline()->AttachAnimation(animation);
// TODO(smcgruer): Should attach a timeline and element rather than call this
// directly. See http://crbug.com/771316
host_impl.animation_host()->RegisterKeyframeEffectForElement(
root_ptr->element_id(), animation->keyframe_effect());
animation->AddKeyframeModel(std::move(transform_animation));
grandchild_ptr->set_visible_layer_rect(gfx::Rect());
root_ptr->test_properties()->transform = singular;
child_ptr->test_properties()->transform = singular;
root_ptr->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(0, 0), grandchild_ptr->visible_layer_rect());
host_impl.animation_host()->UnregisterKeyframeEffectForElement(
root_ptr->element_id(), animation->keyframe_effect());
host_impl.animation_host()->RemoveFromTicking(animation.get());
timeline()->DetachAnimation(animation);
}
TEST_F(LayerTreeHostCommonTest, LayerSkippingInSubtreeOfSingularTransform) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
LayerImpl* grand_child = AddChild<LayerImpl>(child);
SetElementIdsForTesting();
gfx::Transform singular;
singular.matrix().set(0, 0, 0);
singular.matrix().set(0, 1, 1);
root->SetBounds(gfx::Size(10, 10));
child->test_properties()->transform = singular;
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
grand_child->SetBounds(gfx::Size(10, 10));
grand_child->SetDrawsContent(true);
std::unique_ptr<KeyframedTransformAnimationCurve> curve(
KeyframedTransformAnimationCurve::Create());
TransformOperations start;
start.AppendTranslate(1.f, 2.f, 3.f);
gfx::Transform transform;
transform.Scale3d(1.0, 2.0, 3.0);
TransformOperations operation;
operation.AppendMatrix(transform);
curve->AddKeyframe(
TransformKeyframe::Create(base::TimeDelta(), start, nullptr));
curve->AddKeyframe(TransformKeyframe::Create(
base::TimeDelta::FromSecondsD(1.0), operation, nullptr));
std::unique_ptr<KeyframeModel> transform_animation(
KeyframeModel::Create(std::move(curve), 3, 3, TargetProperty::TRANSFORM));
scoped_refptr<SingleKeyframeEffectAnimation> animation(
SingleKeyframeEffectAnimation::Create(1));
timeline()->AttachAnimation(animation);
// TODO(smcgruer): Should attach a timeline and element rather than call this
// directly. See http://crbug.com/771316
host_impl()->animation_host()->RegisterKeyframeEffectForElement(
grand_child->element_id(), animation->keyframe_effect());
animation->AddKeyframeModel(std::move(transform_animation));
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0), grand_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(0, 0), child->visible_layer_rect());
host_impl()->animation_host()->UnregisterKeyframeEffectForElement(
grand_child->element_id(), animation->keyframe_effect());
host_impl()->animation_host()->RemoveFromTicking(animation.get());
timeline()->DetachAnimation(animation);
}
TEST_F(LayerTreeHostCommonTest, SkippingPendingLayerImpl) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
std::unique_ptr<LayerImpl> child =
LayerImpl::Create(host_impl.pending_tree(), 2);
std::unique_ptr<LayerImpl> grandchild =
LayerImpl::Create(host_impl.pending_tree(), 3);
std::unique_ptr<FakePictureLayerImpl> greatgrandchild(
FakePictureLayerImpl::Create(host_impl.pending_tree(), 4));
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
grandchild->SetBounds(gfx::Size(10, 10));
grandchild->SetDrawsContent(true);
greatgrandchild->SetDrawsContent(true);
LayerImpl* root_ptr = root.get();
LayerImpl* grandchild_ptr = grandchild.get();
child->test_properties()->AddChild(std::move(grandchild));
root->test_properties()->AddChild(std::move(child));
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
host_impl.pending_tree()->SetElementIdsForTesting();
// Check the non-skipped case.
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(10, 10), grandchild_ptr->visible_layer_rect());
std::unique_ptr<KeyframedFloatAnimationCurve> curve(
KeyframedFloatAnimationCurve::Create());
std::unique_ptr<TimingFunction> func =
CubicBezierTimingFunction::CreatePreset(
CubicBezierTimingFunction::EaseType::EASE);
curve->AddKeyframe(
FloatKeyframe::Create(base::TimeDelta(), 0.9f, std::move(func)));
curve->AddKeyframe(
FloatKeyframe::Create(base::TimeDelta::FromSecondsD(1.0), 0.3f, nullptr));
std::unique_ptr<KeyframeModel> keyframe_model(
KeyframeModel::Create(std::move(curve), 3, 3, TargetProperty::OPACITY));
scoped_refptr<SingleKeyframeEffectAnimation> animation(
SingleKeyframeEffectAnimation::Create(1));
timeline()->AttachAnimation(animation);
// TODO(smcgruer): Should attach a timeline and element rather than call this
// directly. See http://crbug.com/771316
host_impl.animation_host()->RegisterKeyframeEffectForElement(
root_ptr->element_id(), animation->keyframe_effect());
animation->AddKeyframeModel(std::move(keyframe_model));
root_ptr->test_properties()->opacity = 0.f;
grandchild_ptr->set_visible_layer_rect(gfx::Rect());
root_ptr->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root_ptr);
EXPECT_EQ(gfx::Rect(10, 10), grandchild_ptr->visible_layer_rect());
host_impl.animation_host()->UnregisterKeyframeEffectForElement(
root_ptr->element_id(), animation->keyframe_effect());
host_impl.animation_host()->RemoveFromTicking(animation.get());
timeline()->DetachAnimation(animation);
}
TEST_F(LayerTreeHostCommonTest, SkippingLayer) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(10, 10));
child->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(10, 10), child->visible_layer_rect());
child->set_visible_layer_rect(gfx::Rect());
child->test_properties()->hide_layer_and_subtree = true;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0), child->visible_layer_rect());
child->test_properties()->hide_layer_and_subtree = false;
child->SetBounds(gfx::Size());
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0), child->visible_layer_rect());
child->SetBounds(gfx::Size(10, 10));
gfx::Transform rotate;
child->test_properties()->double_sided = false;
rotate.RotateAboutXAxis(180.f);
child->test_properties()->transform = rotate;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0), child->visible_layer_rect());
child->test_properties()->double_sided = true;
child->test_properties()->transform = gfx::Transform();
child->test_properties()->opacity = 0.f;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(0, 0), child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, LayerTreeRebuildTest) {
// Ensure that the treewalk in LayerTreeHostCommom::
// PreCalculateMetaInformation happens when its required.
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> parent = Layer::Create();
scoped_refptr<Layer> child = Layer::Create();
root->SetBounds(gfx::Size(100, 100));
parent->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(100, 100));
child->SetClipParent(root.get());
root->AddChild(parent);
parent->AddChild(child);
host()->SetRootLayer(root);
child->RequestCopyOfOutput(viz::CopyOutputRequest::CreateStubForTesting());
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_TRUE(LayerSubtreeHasCopyRequest(root.get()));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_TRUE(LayerSubtreeHasCopyRequest(root.get()));
}
TEST_F(LayerTreeHostCommonTest, ResetPropertyTreeIndices) {
scoped_refptr<Layer> root = Layer::Create();
root->SetBounds(gfx::Size(800, 800));
gfx::Transform translate_z;
translate_z.Translate3d(0, 0, 10);
scoped_refptr<Layer> child = Layer::Create();
child->SetTransform(translate_z);
child->SetBounds(gfx::Size(100, 100));
root->AddChild(child);
host()->SetRootLayer(root);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_NE(-1, child->transform_tree_index());
child->RemoveFromParent();
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_EQ(-1, child->transform_tree_index());
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceClipsSubtree) {
// Ensure that a Clip Node is added when a render surface applies clip.
LayerImpl* root = root_layer_for_testing();
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);
// 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);
root->SetBounds(gfx::Size(30, 30));
significant_transform->test_properties()->transform = transform1;
significant_transform->SetBounds(gfx::Size(30, 30));
layer_clips_subtree->SetBounds(gfx::Size(30, 30));
layer_clips_subtree->SetMasksToBounds(true);
layer_clips_subtree->test_properties()->force_render_surface = true;
render_surface->test_properties()->transform = transform2;
render_surface->SetBounds(gfx::Size(30, 30));
render_surface->test_properties()->force_render_surface = true;
test_layer->SetBounds(gfx::Size(30, 30));
test_layer->SetDrawsContent(true);
float device_scale_factor = 1.f;
float page_scale_factor = 1.f;
LayerImpl* page_scale_layer = nullptr;
LayerImpl* inner_viewport_scroll_layer = nullptr;
LayerImpl* outer_viewport_scroll_layer = nullptr;
ExecuteCalculateDrawProperties(root, device_scale_factor, page_scale_factor,
page_scale_layer, inner_viewport_scroll_layer,
outer_viewport_scroll_layer);
EXPECT_TRUE(GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(significant_transform), GetRenderSurface(root));
EXPECT_TRUE(GetRenderSurface(layer_clips_subtree));
EXPECT_NE(GetRenderSurface(render_surface), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(test_layer), GetRenderSurface(render_surface));
EXPECT_EQ(gfx::Rect(30, 20), test_layer->visible_layer_rect());
}
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_for_testing();
LayerImpl* clip_layer = AddChild<LayerImpl>(root);
LayerImpl* target_layer = AddChild<LayerImpl>(clip_layer);
LayerImpl* test_layer = AddChild<LayerImpl>(target_layer);
gfx::Transform transform;
transform.RotateAboutYAxis(45);
root->SetBounds(gfx::Size(30, 30));
clip_layer->test_properties()->transform = transform;
clip_layer->SetBounds(gfx::Size(30, 30));
clip_layer->SetMasksToBounds(true);
target_layer->test_properties()->transform = transform;
target_layer->SetBounds(gfx::Size(30, 30));
target_layer->SetMasksToBounds(true);
test_layer->SetBounds(gfx::Size(30, 30));
test_layer->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(30, 30), test_layer->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest,
RenderSurfaceWithUnclippedDescendantsClipsSubtree) {
// Ensure clip rect is calculated correctly when render surface has unclipped
// descendants.
LayerImpl* root = root_layer_for_testing();
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);
gfx::Transform translate;
translate.Translate(2.0, 2.0);
root->SetBounds(gfx::Size(30, 30));
clip_parent->test_properties()->transform = translate;
clip_parent->SetBounds(gfx::Size(30, 30));
clip_parent->SetMasksToBounds(true);
between_clip_parent_and_child->test_properties()->transform = translate;
between_clip_parent_and_child->SetBounds(gfx::Size(30, 30));
between_clip_parent_and_child->SetMasksToBounds(true);
render_surface->SetBounds(gfx::Size(30, 30));
render_surface->test_properties()->force_render_surface = true;
test_layer->SetBounds(gfx::Size(30, 30));
test_layer->SetDrawsContent(true);
render_surface->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(render_surface);
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(test_layer->is_clipped());
EXPECT_FALSE(test_layer->render_target()->is_clipped());
EXPECT_EQ(gfx::Rect(-2, -2, 30, 30), test_layer->clip_rect());
EXPECT_EQ(gfx::Rect(28, 28), test_layer->drawable_content_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_for_testing();
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);
root->SetBounds(gfx::Size(30, 10));
clip_parent->SetBounds(gfx::Size(30, 30));
render_surface->SetBounds(gfx::Size(10, 15));
render_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(10, 10));
clip_child->SetDrawsContent(true);
child->SetBounds(gfx::Size(40, 40));
child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(30, 10), child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest,
RenderSurfaceClipsSubtreeAndHasUnclippedDescendants) {
LayerImpl* root = root_layer_for_testing();
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);
root->SetBounds(gfx::Size(30, 30));
root->SetMasksToBounds(true);
clip_parent->SetBounds(gfx::Size(30, 30));
render_surface->SetBounds(gfx::Size(50, 50));
render_surface->SetMasksToBounds(true);
render_surface->SetDrawsContent(true);
render_surface->test_properties()->force_render_surface = true;
test_layer1->SetBounds(gfx::Size(50, 50));
test_layer1->SetDrawsContent(true);
clip_child->SetBounds(gfx::Size(50, 50));
clip_child->SetDrawsContent(true);
test_layer2->SetBounds(gfx::Size(50, 50));
test_layer2->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(30, 30), render_surface->visible_layer_rect());
EXPECT_EQ(gfx::Rect(30, 30), 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_for_testing();
LayerImpl* clip_parent = AddChildToRoot<LayerImpl>();
LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent);
LayerImpl* clip_child = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(50, 50));
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->SetDrawsContent(true);
render_surface->SetBounds(gfx::Size(30, 30));
render_surface->test_properties()->force_render_surface = true;
render_surface->SetMasksToBounds(true);
render_surface->SetDrawsContent(true);
clip_child->SetBounds(gfx::Size(50, 50));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
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_for_testing();
LayerImpl* unclipped_surface = AddChildToRoot<LayerImpl>();
LayerImpl* clip_parent = AddChild<LayerImpl>(unclipped_surface);
LayerImpl* clip_layer = AddChild<LayerImpl>(clip_parent);
LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_layer);
LayerImpl* unclipped_desc_surface2 =
AddChild<LayerImpl>(unclipped_desc_surface);
LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface2);
LayerImpl* clipped_surface = AddChild<LayerImpl>(clip_child);
root->SetBounds(gfx::Size(80, 80));
unclipped_surface->SetBounds(gfx::Size(50, 50));
unclipped_surface->SetMasksToBounds(true);
unclipped_surface->SetDrawsContent(true);
unclipped_surface->test_properties()->force_render_surface = true;
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent->SetMasksToBounds(true);
clip_layer->SetMasksToBounds(true);
clip_layer->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
unclipped_desc_surface2->SetBounds(gfx::Size(60, 60));
unclipped_desc_surface2->SetDrawsContent(true);
unclipped_desc_surface2->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(100, 100));
clipped_surface->SetBounds(gfx::Size(70, 70));
clipped_surface->SetDrawsContent(true);
clipped_surface->test_properties()->force_render_surface = true;
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(50, 50),
GetRenderSurface(unclipped_surface)->content_rect());
EXPECT_EQ(gfx::Rect(50, 50),
GetRenderSurface(unclipped_desc_surface)->content_rect());
EXPECT_EQ(gfx::Rect(60, 60),
GetRenderSurface(unclipped_desc_surface2)->content_rect());
EXPECT_EQ(gfx::Rect(50, 50),
GetRenderSurface(clipped_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_for_testing();
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);
gfx::Transform translate;
translate.Translate(10, 10);
gfx::Transform rotate;
rotate.RotateAboutXAxis(10);
root->SetBounds(gfx::Size(100, 100));
surface->SetBounds(gfx::Size(100, 100));
surface->SetMasksToBounds(true);
surface->test_properties()->force_render_surface = true;
surface->test_properties()->transform = rotate;
clip_layer->SetBounds(gfx::Size(20, 20));
clip_layer->SetMasksToBounds(true);
clip_parent->SetBounds(gfx::Size(50, 50));
unclipped_desc_surface->test_properties()->transform = translate;
unclipped_desc_surface->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(100, 100));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(10, 10),
GetRenderSurface(unclipped_desc_surface)->content_rect());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectForDescendantOfScaledSurface) {
LayerImpl* root = root_layer_for_testing();
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);
gfx::Transform scale;
scale.Scale(2, 2);
root->SetBounds(gfx::Size(100, 100));
surface->test_properties()->transform = scale;
surface->SetBounds(gfx::Size(100, 100));
surface->SetMasksToBounds(true);
surface->test_properties()->force_render_surface = true;
clip_layer->SetBounds(gfx::Size(20, 20));
clip_layer->SetMasksToBounds(true);
clip_parent->SetBounds(gfx::Size(50, 50));
unclipped_desc_surface->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(100, 100));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(20, 20), clip_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, LayerWithInputHandlerAndZeroOpacity) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* test_layer = AddChild<LayerImpl>(render_surface);
gfx::Transform translation;
translation.Translate(10, 10);
root->SetBounds(gfx::Size(30, 30));
render_surface->SetBounds(gfx::Size(30, 30));
render_surface->SetMasksToBounds(true);
render_surface->test_properties()->force_render_surface = true;
test_layer->test_properties()->transform = translation;
test_layer->SetBounds(gfx::Size(20, 20));
test_layer->SetDrawsContent(true);
TouchActionRegion touch_action_region;
touch_action_region.Union(kTouchActionNone, gfx::Rect(0, 0, 20, 20));
test_layer->SetTouchActionRegion(std::move(touch_action_region));
test_layer->test_properties()->opacity = 0.f;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(translation,
test_layer->ScreenSpaceTransform());
}
TEST_F(LayerTreeHostCommonTest, ClipParentDrawsIntoScaledRootSurface) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip_layer = AddChild<LayerImpl>(root);
LayerImpl* clip_parent = AddChild<LayerImpl>(clip_layer);
LayerImpl* clip_parent_child = AddChild<LayerImpl>(clip_parent);
LayerImpl* unclipped_desc_surface = AddChild<LayerImpl>(clip_parent_child);
LayerImpl* clip_child = AddChild<LayerImpl>(unclipped_desc_surface);
root->SetBounds(gfx::Size(100, 100));
clip_layer->SetBounds(gfx::Size(20, 20));
clip_layer->SetMasksToBounds(true);
clip_parent->SetBounds(gfx::Size(50, 50));
clip_parent_child->SetBounds(gfx::Size(20, 20));
clip_parent_child->SetMasksToBounds(true);
unclipped_desc_surface->SetBounds(gfx::Size(100, 100));
unclipped_desc_surface->SetDrawsContent(true);
unclipped_desc_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(100, 100));
clip_child->SetDrawsContent(true);
gfx::Transform translate;
translate.Translate(10, 10);
unclipped_desc_surface->test_properties()->transform = translate;
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
float device_scale_factor = 1.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_EQ(gfx::Rect(-10, -10, 20, 20), clip_child->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), clip_child->visible_layer_rect());
device_scale_factor = 2.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_EQ(gfx::Rect(-20, -20, 40, 40), clip_child->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), clip_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, ClipChildVisibleRect) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip_parent = AddChildToRoot<LayerImpl>();
LayerImpl* render_surface = AddChild<LayerImpl>(clip_parent);
LayerImpl* clip_child = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(30, 30));
clip_parent->SetBounds(gfx::Size(40, 40));
clip_parent->SetMasksToBounds(true);
render_surface->SetBounds(gfx::Size(50, 50));
render_surface->SetMasksToBounds(true);
render_surface->SetDrawsContent(true);
render_surface->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(50, 50));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = clip_parent;
clip_parent->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
clip_parent->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(30, 30), clip_child->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest,
LayerClipRectLargerThanClippingRenderSurfaceRect) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* test_layer = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(30, 30));
root->SetMasksToBounds(true);
root->SetDrawsContent(true);
render_surface->SetBounds(gfx::Size(50, 50));
render_surface->SetMasksToBounds(true);
render_surface->SetDrawsContent(true);
render_surface->test_properties()->force_render_surface = true;
test_layer->SetBounds(gfx::Size(50, 50));
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_for_testing();
LayerImpl* hidden = AddChild<LayerImpl>(root);
LayerImpl* test = AddChild<LayerImpl>(hidden);
root->SetBounds(gfx::Size(30, 30));
hidden->SetBounds(gfx::Size(30, 30));
hidden->test_properties()->force_render_surface = true;
hidden->test_properties()->hide_layer_and_subtree = true;
test->SetBounds(gfx::Size(30, 30));
test->test_properties()->force_render_surface = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(0.f,
GetRenderSurface(test)->OwningEffectNode()->screen_space_opacity);
hidden->test_properties()->hide_layer_and_subtree = false;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(1.f,
GetRenderSurface(test)->OwningEffectNode()->screen_space_opacity);
}
TEST_F(LayerTreeHostCommonTest, TwoUnclippedRenderSurfaces) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* clip_layer = AddChild<LayerImpl>(root);
LayerImpl* render_surface1 = AddChild<LayerImpl>(clip_layer);
LayerImpl* render_surface2 = AddChild<LayerImpl>(render_surface1);
LayerImpl* clip_child = AddChild<LayerImpl>(render_surface2);
root->SetBounds(gfx::Size(30, 30));
root->SetMasksToBounds(true);
clip_layer->SetBounds(gfx::Size(30, 30));
clip_layer->SetMasksToBounds(true);
render_surface1->test_properties()->position = gfx::PointF(10, 10);
render_surface1->SetBounds(gfx::Size(30, 30));
render_surface1->SetDrawsContent(true);
render_surface1->test_properties()->force_render_surface = true;
render_surface2->SetBounds(gfx::Size(30, 30));
render_surface2->SetDrawsContent(true);
render_surface2->test_properties()->force_render_surface = true;
clip_child->SetBounds(gfx::Size(30, 30));
clip_child->SetDrawsContent(true);
clip_child->test_properties()->clip_parent = root;
root->test_properties()->clip_children =
std::make_unique<std::set<LayerImpl*>>();
root->test_properties()->clip_children->insert(clip_child);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(-10, -10, 30, 30), render_surface2->clip_rect());
}
TEST_F(LayerTreeHostCommonTest, MaskLayerDrawProperties) {
// Tests that a mask layer's draw properties are computed correctly.
LayerImpl* root = root_layer_for_testing();
LayerImpl* child = AddChild<LayerImpl>(root);
child->test_properties()->SetMaskLayer(
LayerImpl::Create(root->layer_tree_impl(), 100));
LayerImpl* mask = child->test_properties()->mask_layer;
gfx::Transform transform;
transform.Translate(10, 10);
root->SetBounds(gfx::Size(40, 40));
root->SetDrawsContent(true);
child->test_properties()->transform = transform;
child->SetBounds(gfx::Size(30, 30));
child->SetDrawsContent(false);
mask->SetBounds(gfx::Size(20, 20));
ExecuteCalculateDrawProperties(root);
// The render surface created for the mask has no contributing content, so the
// mask doesn't contribute to a drawn render surface. This means it has an
// empty visible rect, but its screen space transform can still be computed
// correctly on-demand.
EXPECT_FALSE(mask->contributes_to_drawn_render_surface());
EXPECT_EQ(gfx::Rect(), mask->visible_layer_rect());
EXPECT_TRANSFORMATION_MATRIX_EQ(transform, mask->ScreenSpaceTransform());
// Make the child's render surface have contributing content.
child->SetDrawsContent(true);
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRUE(mask->contributes_to_drawn_render_surface());
EXPECT_EQ(gfx::Rect(20, 20), mask->visible_layer_rect());
EXPECT_TRANSFORMATION_MATRIX_EQ(transform, mask->ScreenSpaceTransform());
transform.Translate(10, 10);
child->test_properties()->transform = transform;
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_TRANSFORMATION_MATRIX_EQ(transform, mask->ScreenSpaceTransform());
EXPECT_EQ(gfx::Rect(20, 20), mask->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest,
SublayerScaleWithTransformNodeBetweenTwoTargets) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChild<LayerImpl>(root);
LayerImpl* between_targets = AddChild<LayerImpl>(render_surface1);
LayerImpl* render_surface2 = AddChild<LayerImpl>(between_targets);
LayerImpl* test_layer = AddChild<LayerImpl>(render_surface2);
gfx::Transform scale;
scale.Scale(2.f, 2.f);
root->SetBounds(gfx::Size(30, 30));
render_surface1->test_properties()->transform = scale;
render_surface1->SetBounds(gfx::Size(30, 30));
render_surface1->test_properties()->force_render_surface = true;
between_targets->SetBounds(gfx::Size(30, 30));
render_surface2->SetBounds(gfx::Size(30, 30));
render_surface2->test_properties()->force_render_surface = true;
test_layer->SetBounds(gfx::Size(30, 30));
test_layer->SetDrawsContent(true);
// We want layer between the two targets to create a clip node and effect
// node but it shouldn't create a render surface.
between_targets->SetMasksToBounds(true);
between_targets->test_properties()->opacity = 0.5f;
ExecuteCalculateDrawProperties(root);
EffectTree& tree = root->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = tree.Node(render_surface1->effect_tree_index());
EXPECT_EQ(node->surface_contents_scale, gfx::Vector2dF(2.f, 2.f));
node = tree.Node(between_targets->effect_tree_index());
EXPECT_EQ(node->surface_contents_scale, gfx::Vector2dF(1.f, 1.f));
node = tree.Node(render_surface2->effect_tree_index());
EXPECT_EQ(node->surface_contents_scale, gfx::Vector2dF(2.f, 2.f));
EXPECT_EQ(gfx::Rect(15, 15), test_layer->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, NoisyTransform) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* scroll_child = AddChild<LayerImpl>(render_surface);
LayerImpl* scroll_clip = AddChild<LayerImpl>(render_surface);
LayerImpl* scroll_parent = AddChild<LayerImpl>(scroll_clip);
scroll_child->test_properties()->scroll_parent = scroll_parent;
scroll_parent->SetDrawsContent(true);
scroll_child->SetDrawsContent(true);
render_surface->test_properties()->force_render_surface = true;
// A noisy transform that's invertible.
gfx::Transform transform;
transform.matrix().setDouble(0, 0, 6.12323e-17);
transform.matrix().setDouble(0, 2, 1);
transform.matrix().setDouble(2, 2, 6.12323e-17);
transform.matrix().setDouble(2, 0, -1);
scroll_child->test_properties()->transform = transform;
render_surface->test_properties()->transform = transform;
root->SetBounds(gfx::Size(30, 30));
scroll_child->SetBounds(gfx::Size(30, 30));
scroll_parent->SetBounds(gfx::Size(30, 30));
ExecuteCalculateDrawProperties(root);
gfx::Transform expected;
expected.matrix().setDouble(0, 0, 3.749395e-33);
expected.matrix().setDouble(0, 2, 6.12323e-17);
expected.matrix().setDouble(2, 0, -1);
expected.matrix().setDouble(2, 2, 6.12323e-17);
EXPECT_TRANSFORMATION_MATRIX_EQ(expected,
scroll_child->ScreenSpaceTransform());
}
TEST_F(LayerTreeHostCommonTest, LargeTransformTest) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface1 = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(render_surface1);
child->SetDrawsContent(true);
child->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)));
root->SetBounds(gfx::Size(30, 30));
render_surface1->SetBounds(gfx::Size(30, 30));
render_surface1->test_properties()->force_render_surface = true;
// TODO(sunxd): we make child have no render surface, because if the
// child has one, the large transform applied to child will result in NaNs in
// the draw_transform of the render_surface, thus make draw property updates
// skip the child layer. We need further investigation into this to know
// what exactly happens here.
child->test_properties()->transform = large_transform;
child->SetBounds(gfx::Size(30, 30));
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::RectF(),
GetRenderSurface(render_surface1)->DrawableContentRect());
bool is_inf_or_nan = std::isinf(child->DrawTransform().matrix().get(0, 0)) ||
std::isnan(child->DrawTransform().matrix().get(0, 0));
EXPECT_TRUE(is_inf_or_nan);
is_inf_or_nan = std::isinf(child->DrawTransform().matrix().get(1, 1)) ||
std::isnan(child->DrawTransform().matrix().get(1, 1));
EXPECT_TRUE(is_inf_or_nan);
// The root layer should be in the RenderSurfaceList.
const auto* rsl = render_surface_list_impl();
EXPECT_TRUE(base::ContainsValue(*rsl, GetRenderSurface(root)));
}
TEST_F(LayerTreeHostCommonTest, PropertyTreesRebuildWithOpacityChanges) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> child = Layer::Create();
child->SetIsDrawable(true);
root->AddChild(child);
host()->SetRootLayer(root);
root->SetBounds(gfx::Size(100, 100));
child->SetBounds(gfx::Size(20, 20));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
// Changing the opacity from 1 to non-1 value should trigger rebuild of
// property trees as a new effect node will be created.
child->SetOpacity(0.5f);
PropertyTrees* property_trees = host()->property_trees();
EXPECT_TRUE(property_trees->needs_rebuild);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_NE(property_trees->effect_tree.Node(child->effect_tree_index()),
property_trees->effect_tree.Node(root->effect_tree_index()));
// child already has an effect node. Changing its opacity shouldn't trigger
// a property trees rebuild.
child->SetOpacity(0.8f);
property_trees = host()->property_trees();
EXPECT_FALSE(property_trees->needs_rebuild);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_NE(property_trees->effect_tree.Node(child->effect_tree_index()),
property_trees->effect_tree.Node(root->effect_tree_index()));
// Changing the opacity from non-1 value to 1 should trigger a rebuild of
// property trees as the effect node may no longer be needed.
child->SetOpacity(1.f);
property_trees = host()->property_trees();
EXPECT_TRUE(property_trees->needs_rebuild);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EXPECT_EQ(property_trees->effect_tree.Node(child->effect_tree_index()),
property_trees->effect_tree.Node(root->effect_tree_index()));
}
TEST_F(LayerTreeHostCommonTest, OpacityAnimationsTrackingTest) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> animated = Layer::Create();
animated->SetIsDrawable(true);
root->AddChild(animated);
host()->SetRootLayer(root);
host()->SetElementIdsForTesting();
root->SetBounds(gfx::Size(100, 100));
root->SetForceRenderSurfaceForTesting(true);
animated->SetBounds(gfx::Size(20, 20));
animated->SetOpacity(0.f);
scoped_refptr<SingleKeyframeEffectAnimation> animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline()->AttachAnimation(animation);
animation->AttachElement(animated->element_id());
int keyframe_model_id = 0;
std::unique_ptr<KeyframeModel> keyframe_model = KeyframeModel::Create(
std::unique_ptr<AnimationCurve>(new FakeFloatTransition(1.0, 0.f, 1.f)),
keyframe_model_id, 1, TargetProperty::OPACITY);
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
KeyframeModel* keyframe_model_ptr = keyframe_model.get();
AddKeyframeModelToElementWithExistingKeyframeEffect(
animated->element_id(), timeline(), std::move(keyframe_model));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
EffectTree& tree = root->layer_tree_host()->property_trees()->effect_tree;
EffectNode* node = tree.Node(animated->effect_tree_index());
EXPECT_FALSE(node->is_currently_animating_opacity);
EXPECT_TRUE(node->has_potential_opacity_animation);
keyframe_model_ptr->set_time_offset(base::TimeDelta::FromMilliseconds(0));
host()->AnimateLayers(base::TimeTicks::Max());
node = tree.Node(animated->effect_tree_index());
EXPECT_TRUE(node->is_currently_animating_opacity);
EXPECT_TRUE(node->has_potential_opacity_animation);
animation->AbortKeyframeModelsWithProperty(TargetProperty::OPACITY,
false /*needs_completion*/);
node = tree.Node(animated->effect_tree_index());
EXPECT_FALSE(node->is_currently_animating_opacity);
EXPECT_FALSE(node->has_potential_opacity_animation);
}
TEST_F(LayerTreeHostCommonTest, TransformAnimationsTrackingTest) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> animated = Layer::Create();
animated->SetIsDrawable(true);
root->AddChild(animated);
host()->SetRootLayer(root);
host()->SetElementIdsForTesting();
root->SetBounds(gfx::Size(100, 100));
root->SetForceRenderSurfaceForTesting(true);
animated->SetBounds(gfx::Size(20, 20));
scoped_refptr<SingleKeyframeEffectAnimation> animation =
SingleKeyframeEffectAnimation::Create(
AnimationIdProvider::NextAnimationId());
timeline()->AttachAnimation(animation);
animation->AttachElement(animated->element_id());
std::unique_ptr<KeyframedTransformAnimationCurve> curve(
KeyframedTransformAnimationCurve::Create());
TransformOperations start;
start.AppendTranslate(1.f, 2.f, 3.f);
gfx::Transform transform;
transform.Scale3d(1.0, 2.0, 3.0);
TransformOperations operation;
operation.AppendMatrix(transform);
curve->AddKeyframe(
TransformKeyframe::Create(base::TimeDelta(), start, nullptr));
curve->AddKeyframe(TransformKeyframe::Create(
base::TimeDelta::FromSecondsD(1.0), operation, nullptr));
std::unique_ptr<KeyframeModel> keyframe_model(
KeyframeModel::Create(std::move(curve), 3, 3, TargetProperty::TRANSFORM));
keyframe_model->set_fill_mode(KeyframeModel::FillMode::NONE);
keyframe_model->set_time_offset(base::TimeDelta::FromMilliseconds(-1000));
KeyframeModel* keyframe_model_ptr = keyframe_model.get();
AddKeyframeModelToElementWithExistingKeyframeEffect(
animated->element_id(), timeline(), std::move(keyframe_model));
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root.get());
TransformTree& tree =
root->layer_tree_host()->property_trees()->transform_tree;
TransformNode* node = tree.Node(animated->transform_tree_index());
EXPECT_FALSE(node->is_currently_animating);
EXPECT_TRUE(node->has_potential_animation);
keyframe_model_ptr->set_time_offset(base::TimeDelta::FromMilliseconds(0));
host()->AnimateLayers(base::TimeTicks::Max());
node = tree.Node(animated->transform_tree_index());
EXPECT_TRUE(node->is_currently_animating);
EXPECT_TRUE(node->has_potential_animation);
animation->AbortKeyframeModelsWithProperty(TargetProperty::TRANSFORM,
false /*needs_completion*/);
node = tree.Node(animated->transform_tree_index());
EXPECT_FALSE(node->is_currently_animating);
EXPECT_FALSE(node->has_potential_animation);
}
TEST_F(LayerTreeHostCommonTest, ScrollTreeBuilderTest) {
// Test the behavior of scroll tree builder
// Topology:
// +root1(1)[inner_viewport_container_layer]
// +-page_scale_layer
// +----parent2(2)[kHasBackgroundAttachmentFixedObjects|kScrollbarScrolling &
// scrollable, inner_viewport_scroll_layer]
// +------child6(6)[kScrollbarScrolling]
// +--------grand_child10(10)[kScrollbarScrolling]
// +----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();
scoped_refptr<Layer> page_scale_layer = Layer::Create();
scoped_refptr<Layer> parent2 = Layer::Create();
scoped_refptr<Layer> parent3 = Layer::Create();
scoped_refptr<Layer> parent4 = Layer::Create();
scoped_refptr<Layer> parent5 = Layer::Create();
scoped_refptr<Layer> child6 = Layer::Create();
scoped_refptr<Layer> child7 = Layer::Create();
scoped_refptr<Layer> child8 = Layer::Create();
scoped_refptr<Layer> child9 = Layer::Create();
scoped_refptr<Layer> grand_child10 = Layer::Create();
scoped_refptr<Layer> grand_child11 = Layer::Create();
scoped_refptr<Layer> grand_child12 = Layer::Create();
root1->AddChild(page_scale_layer);
page_scale_layer->AddChild(parent2);
page_scale_layer->AddChild(parent3);
page_scale_layer->AddChild(parent4);
page_scale_layer->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);
root1->SetBounds(gfx::Size(1, 1));
parent2->AddMainThreadScrollingReasons(
MainThreadScrollingReason::kHasBackgroundAttachmentFixedObjects);
parent2->AddMainThreadScrollingReasons(
MainThreadScrollingReason::kScrollbarScrolling);
parent2->SetElementId(LayerIdToElementIdForTesting(parent2->id()));
parent2->SetScrollable(root1->bounds());
child6->AddMainThreadScrollingReasons(
MainThreadScrollingReason::kScrollbarScrolling);
grand_child10->AddMainThreadScrollingReasons(
MainThreadScrollingReason::kScrollbarScrolling);
parent3->SetBounds(gfx::Size(2, 2));
child7->SetScrollable(parent3->bounds());
child7->SetElementId(LayerIdToElementIdForTesting(child7->id()));
child8->SetScrollParent(child7.get());
child8->SetBounds(gfx::Size(3, 3));
grand_child11->SetScrollable(child8->bounds());
grand_child11->SetElementId(
LayerIdToElementIdForTesting(grand_child11->id()));
parent5->SetNonFastScrollableRegion(gfx::Rect(0, 0, 50, 50));
parent5->SetBounds(gfx::Size(10, 10));
ViewportLayers viewport_layers;
viewport_layers.page_scale = page_scale_layer;
viewport_layers.inner_viewport_container = root1;
viewport_layers.inner_viewport_scroll = parent2;
host()->RegisterViewportLayers(viewport_layers);
ExecuteCalculateDrawPropertiesAndSaveUpdateLayerList(root1.get());
const int kRootPropertyTreeNodeId = 0;
// Property tree root
ScrollTree& scroll_tree = host()->property_trees()->scroll_tree;
PropertyTrees property_trees;
property_trees.is_main_thread = true;
property_trees.is_active = false;
ScrollTree& expected_scroll_tree = property_trees.scroll_tree;
ScrollNode* property_tree_root = expected_scroll_tree.Node(0);
property_tree_root->id = kRootPropertyTreeNodeId;
property_tree_root->parent_id = ScrollTree::kInvalidNodeId;
property_tree_root->scrollable = false;
property_tree_root->main_thread_scrolling_reasons =
MainThreadScrollingReason::kNotScrollingOnMain;
property_tree_root->transform_id = kRootPropertyTreeNodeId;
// The node owned by root1
ScrollNode scroll_root1;
scroll_root1.id = 1;
scroll_root1.bounds = root1->bounds();
scroll_root1.user_scrollable_horizontal = true;
scroll_root1.user_scrollable_vertical = true;
scroll_root1.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.element_id = parent2->element_id();
scroll_parent2.scrollable = true;
scroll_parent2.main_thread_scrolling_reasons =
parent2->GetMainThreadScrollingReasons();
scroll_parent2.container_bounds = root1->bounds();
scroll_parent2.bounds = parent2->bounds();
scroll_parent2.max_scroll_offset_affected_by_page_scale = true;
scroll_parent2.scrolls_inner_viewport = true;
scroll_parent2.user_scrollable_horizontal = true;
scroll_parent2.user_scrollable_vertical = true;
scroll_parent2.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.main_thread_scrolling_reasons =
child6->GetMainThreadScrollingReasons();
scroll_child6.should_flatten = true;
scroll_child6.user_scrollable_horizontal = true;
scroll_child6.user_scrollable_vertical = true;
scroll_child6.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.element_id = child7->element_id();
scroll_child7.scrollable = true;
scroll_child7.container_bounds = parent3->bounds();
scroll_child7.bounds = child7->bounds();
scroll_child7.user_scrollable_horizontal = true;
scroll_child7.user_scrollable_vertical = true;
scroll_child7.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.element_id = grand_child11->element_id();
scroll_grand_child11.scrollable = true;
scroll_grand_child11.container_bounds = child8->bounds();
scroll_grand_child11.user_scrollable_horizontal = true;
scroll_grand_child11.user_scrollable_vertical = true;
scroll_grand_child11.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.bounds = gfx::Size(10, 10);
scroll_parent5.should_flatten = true;
scroll_parent5.user_scrollable_horizontal = true;
scroll_parent5.user_scrollable_vertical = true;
scroll_parent5.transform_id = parent5->transform_tree_index();
expected_scroll_tree.Insert(scroll_parent5, 1);
expected_scroll_tree.SetScrollOffset(parent2->element_id(),
gfx::ScrollOffset(0, 0));
expected_scroll_tree.SetScrollOffset(child7->element_id(),
gfx::ScrollOffset(0, 0));
expected_scroll_tree.SetScrollOffset(grand_child11->element_id(),
gfx::ScrollOffset(0, 0));
expected_scroll_tree.set_needs_update(false);
EXPECT_EQ(expected_scroll_tree, scroll_tree);
// Check other layers' scroll_tree_index
EXPECT_EQ(scroll_root1.id, page_scale_layer->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());
}
TEST_F(LayerTreeHostCommonTest, CanAdjustRasterScaleTest) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* render_surface = AddChild<LayerImpl>(root);
LayerImpl* child = AddChild<LayerImpl>(render_surface);
root->SetBounds(gfx::Size(50, 50));
render_surface->SetBounds(gfx::Size(10, 10));
render_surface->test_properties()->force_render_surface = true;
gfx::Transform transform;
transform.Scale(5.f, 5.f);
render_surface->test_properties()->transform = transform;
child->SetDrawsContent(true);
child->SetMasksToBounds(true);
child->SetBounds(gfx::Size(10, 10));
ExecuteCalculateDrawPropertiesWithoutAdjustingRasterScales(root);
// Check surface draw properties.
EXPECT_EQ(gfx::Rect(10, 10),
GetRenderSurface(render_surface)->content_rect());
EXPECT_EQ(transform, GetRenderSurface(render_surface)->draw_transform());
EXPECT_EQ(gfx::RectF(50.0f, 50.0f),
GetRenderSurface(render_surface)->DrawableContentRect());
// Check child layer draw properties.
EXPECT_EQ(gfx::Rect(10, 10), child->visible_layer_rect());
EXPECT_EQ(gfx::Transform(), child->DrawTransform());
EXPECT_EQ(gfx::Rect(10, 10), child->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), child->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonTest, SurfaceContentsScaleChangeWithCopyRequestTest) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* scale_layer = AddChild<LayerImpl>(root);
LayerImpl* copy_layer = AddChild<LayerImpl>(scale_layer);
LayerImpl* clip_layer = AddChild<LayerImpl>(copy_layer);
LayerImpl* test_layer = AddChild<LayerImpl>(clip_layer);
root->SetBounds(gfx::Size(150, 150));
scale_layer->SetBounds(gfx::Size(30, 30));
gfx::Transform transform;
transform.Scale(5.f, 5.f);
scale_layer->test_properties()->transform = transform;
// Need to persist the render surface after copy request is cleared.
copy_layer->test_properties()->force_render_surface = true;
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
clip_layer->SetDrawsContent(true);
clip_layer->SetMasksToBounds(true);
clip_layer->SetBounds(gfx::Size(10, 10));
test_layer->SetDrawsContent(true);
test_layer->SetMasksToBounds(true);
test_layer->SetBounds(gfx::Size(20, 20));
ExecuteCalculateDrawPropertiesWithoutAdjustingRasterScales(root);
// Check surface with copy request draw properties.
EXPECT_EQ(gfx::Rect(50, 50), GetRenderSurface(copy_layer)->content_rect());
EXPECT_EQ(gfx::Transform(), GetRenderSurface(copy_layer)->draw_transform());
EXPECT_EQ(gfx::RectF(50.0f, 50.0f),
GetRenderSurface(copy_layer)->DrawableContentRect());
// Check test layer draw properties.
EXPECT_EQ(gfx::Rect(10, 10), test_layer->visible_layer_rect());
EXPECT_EQ(transform, test_layer->DrawTransform());
EXPECT_EQ(gfx::Rect(50, 50), test_layer->clip_rect());
EXPECT_EQ(gfx::Rect(50, 50), test_layer->drawable_content_rect());
// Clear the copy request and call UpdateSurfaceContentsScale.
host_impl()->active_tree()->property_trees()->effect_tree.ClearCopyRequests();
ExecuteCalculateDrawPropertiesWithoutAdjustingRasterScales(root);
// Check surface draw properties without copy request.
EXPECT_EQ(gfx::Rect(10, 10), GetRenderSurface(copy_layer)->content_rect());
EXPECT_EQ(transform, GetRenderSurface(copy_layer)->draw_transform());
EXPECT_EQ(gfx::RectF(50.0f, 50.0f),
GetRenderSurface(copy_layer)->DrawableContentRect());
// Check test layer draw properties without copy request.
EXPECT_EQ(gfx::Rect(10, 10), test_layer->visible_layer_rect());
EXPECT_EQ(gfx::Transform(), test_layer->DrawTransform());
EXPECT_EQ(gfx::Rect(10, 10), test_layer->clip_rect());
EXPECT_EQ(gfx::Rect(10, 10), test_layer->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonTest, SubtreeHiddenWithCacheRenderSurface) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
host_impl.CreatePendingTree();
std::unique_ptr<LayerImpl> root =
LayerImpl::Create(host_impl.pending_tree(), 1);
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
LayerImpl* root_layer = root.get();
std::unique_ptr<LayerImpl> cache_grand_parent =
LayerImpl::Create(host_impl.pending_tree(), 2);
cache_grand_parent->SetBounds(gfx::Size(40, 40));
cache_grand_parent->SetDrawsContent(true);
LayerImpl* cache_grand_parent_layer = cache_grand_parent.get();
std::unique_ptr<LayerImpl> cache_parent =
LayerImpl::Create(host_impl.pending_tree(), 3);
cache_parent->SetBounds(gfx::Size(30, 30));
cache_parent->SetDrawsContent(true);
cache_parent->test_properties()->force_render_surface = true;
LayerImpl* cache_parent_layer = cache_parent.get();
std::unique_ptr<LayerImpl> cache_render_surface =
LayerImpl::Create(host_impl.pending_tree(), 4);
cache_render_surface->SetBounds(gfx::Size(20, 20));
cache_render_surface->SetDrawsContent(true);
cache_render_surface->test_properties()->cache_render_surface = true;
LayerImpl* cache_layer = cache_render_surface.get();
std::unique_ptr<LayerImpl> cache_child =
LayerImpl::Create(host_impl.pending_tree(), 5);
cache_child->SetBounds(gfx::Size(20, 20));
cache_child->SetDrawsContent(true);
LayerImpl* cache_child_layer = cache_child.get();
std::unique_ptr<LayerImpl> cache_grand_child =
LayerImpl::Create(host_impl.pending_tree(), 6);
cache_grand_child->SetBounds(gfx::Size(20, 20));
cache_grand_child->SetDrawsContent(true);
LayerImpl* cache_grand_child_layer = cache_grand_child.get();
std::unique_ptr<LayerImpl> cache_grand_parent_sibling_before =
LayerImpl::Create(host_impl.pending_tree(), 7);
cache_grand_parent_sibling_before->SetBounds(gfx::Size(40, 40));
cache_grand_parent_sibling_before->SetDrawsContent(true);
LayerImpl* cache_grand_parent_sibling_before_layer =
cache_grand_parent_sibling_before.get();
std::unique_ptr<LayerImpl> cache_grand_parent_sibling_after =
LayerImpl::Create(host_impl.pending_tree(), 8);
cache_grand_parent_sibling_after->SetBounds(gfx::Size(40, 40));
cache_grand_parent_sibling_after->SetDrawsContent(true);
LayerImpl* cache_grand_parent_sibling_after_layer =
cache_grand_parent_sibling_after.get();
cache_child->test_properties()->AddChild(std::move(cache_grand_child));
cache_render_surface->test_properties()->AddChild(std::move(cache_child));
cache_parent->test_properties()->AddChild(std::move(cache_render_surface));
cache_grand_parent->test_properties()->AddChild(std::move(cache_parent));
root->test_properties()->AddChild(
std::move(cache_grand_parent_sibling_before));
root->test_properties()->AddChild(std::move(cache_grand_parent));
root->test_properties()->AddChild(
std::move(cache_grand_parent_sibling_after));
host_impl.pending_tree()->SetRootLayerForTesting(std::move(root));
// Hide the cache_grand_parent and its subtree. But cache a render surface in
// that hidden subtree on cache_layer. Also hide the cache grand child and its
// subtree.
cache_grand_parent_layer->test_properties()->hide_layer_and_subtree = true;
cache_grand_parent_sibling_before_layer->test_properties()
->hide_layer_and_subtree = true;
cache_grand_parent_sibling_after_layer->test_properties()
->hide_layer_and_subtree = true;
cache_grand_child_layer->test_properties()->hide_layer_and_subtree = true;
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root_layer, root_layer->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
// 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 cache_layer.
ASSERT_EQ(4u, render_surface_list.size());
EXPECT_EQ(static_cast<uint64_t>(root_layer->id()),
render_surface_list.at(0)->id());
EXPECT_EQ(static_cast<uint64_t>(cache_grand_parent_layer->id()),
render_surface_list.at(1)->id());
EXPECT_EQ(static_cast<uint64_t>(cache_parent_layer->id()),
render_surface_list.at(2)->id());
EXPECT_EQ(static_cast<uint64_t>(cache_layer->id()),
render_surface_list.at(3)->id());
// The root render surface should have 2 contributing layers.
EXPECT_EQ(2, GetRenderSurface(root_layer)->num_contributors());
EXPECT_TRUE(root_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(cache_grand_parent_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(cache_grand_parent_sibling_before_layer
->contributes_to_drawn_render_surface());
EXPECT_FALSE(cache_grand_parent_sibling_after_layer
->contributes_to_drawn_render_surface());
// Nothing actually draws into the cache parent, so only the cache_layer will
// appear in its list, since it needs to be drawn for the cache render
// surface.
ASSERT_EQ(1, GetRenderSurface(cache_parent_layer)->num_contributors());
EXPECT_FALSE(cache_parent_layer->contributes_to_drawn_render_surface());
// The cache layer's render surface should have 2 contributing layers.
ASSERT_EQ(2, GetRenderSurface(cache_layer)->num_contributors());
EXPECT_TRUE(cache_layer->contributes_to_drawn_render_surface());
EXPECT_TRUE(cache_child_layer->contributes_to_drawn_render_surface());
EXPECT_FALSE(cache_grand_child_layer->contributes_to_drawn_render_surface());
// cache_grand_parent, cache_parent shouldn't be drawn because they are
// hidden, but the cache_layer and cache_child should be drawn for the cache
// render surface. cache grand child should not be drawn as its hidden even in
// the cache render surface.
EffectTree& tree =
root_layer->layer_tree_impl()->property_trees()->effect_tree;
EffectNode* node = tree.Node(cache_grand_parent_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
node = tree.Node(cache_parent_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
node = tree.Node(cache_layer->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
node = tree.Node(cache_child_layer->effect_tree_index());
EXPECT_TRUE(node->is_drawn);
node = tree.Node(cache_grand_child_layer->effect_tree_index());
EXPECT_FALSE(node->is_drawn);
// Though cache_layer is drawn, it shouldn't contribute to drawn surface as
// its actually hidden.
EXPECT_FALSE(GetRenderSurface(cache_layer)->contributes_to_drawn_surface());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectInNonRootCacheRenderSurface) {
LayerImpl* root = root_layer_for_testing();
root->SetBounds(gfx::Size(50, 50));
root->SetDrawsContent(true);
root->SetMasksToBounds(true);
LayerImpl* cache_render_surface_layer = AddChild<LayerImpl>(root);
cache_render_surface_layer->SetBounds(gfx::Size(120, 120));
cache_render_surface_layer->SetDrawsContent(true);
cache_render_surface_layer->test_properties()->cache_render_surface = true;
LayerImpl* copy_layer = AddChild<LayerImpl>(cache_render_surface_layer);
copy_layer->SetBounds(gfx::Size(100, 100));
copy_layer->SetDrawsContent(true);
copy_layer->test_properties()->force_render_surface = true;
LayerImpl* copy_child = AddChild<LayerImpl>(copy_layer);
copy_child->test_properties()->position = gfx::PointF(40.f, 40.f);
copy_child->SetBounds(gfx::Size(20, 20));
copy_child->SetDrawsContent(true);
LayerImpl* copy_clip = AddChild<LayerImpl>(copy_layer);
copy_clip->SetBounds(gfx::Size(55, 55));
copy_clip->SetMasksToBounds(true);
LayerImpl* copy_clipped_child = AddChild<LayerImpl>(copy_clip);
copy_clipped_child->test_properties()->position = gfx::PointF(40.f, 40.f);
copy_clipped_child->SetBounds(gfx::Size(20, 20));
copy_clipped_child->SetDrawsContent(true);
LayerImpl* cache_surface = AddChild<LayerImpl>(copy_clip);
cache_surface->test_properties()->position = gfx::PointF(45.f, 45.f);
cache_surface->SetBounds(gfx::Size(20, 20));
cache_surface->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(120, 120),
cache_render_surface_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(100, 100), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(20, 20), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(15, 15), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), cache_surface->visible_layer_rect());
// Case 2: When the non root cache render surface layer is clipped.
cache_render_surface_layer->SetBounds(gfx::Size(50, 50));
cache_render_surface_layer->SetMasksToBounds(true);
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(gfx::Rect(50, 50),
cache_render_surface_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(50, 50), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(5, 5), cache_surface->visible_layer_rect());
// Case 3: When there is device scale factor.
float device_scale_factor = 2.f;
ExecuteCalculateDrawProperties(root, device_scale_factor);
EXPECT_EQ(gfx::Rect(50, 50),
cache_render_surface_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(50, 50), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(5, 5), cache_surface->visible_layer_rect());
// Case 4: When the non root cache render surface layer is clipped and there
// is a copy request layer beneath it.
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
EXPECT_EQ(gfx::Rect(50, 50),
cache_render_surface_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(100, 100), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(20, 20), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(15, 15), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(10, 10), cache_surface->visible_layer_rect());
// Case 5: When there is another cache render surface layer under the copy
// request layer.
cache_surface->test_properties()->cache_render_surface = true;
copy_layer->test_properties()->copy_requests.push_back(
viz::CopyOutputRequest::CreateStubForTesting());
root->layer_tree_impl()->property_trees()->needs_rebuild = true;
DCHECK(!copy_layer->test_properties()->copy_requests.empty());
ExecuteCalculateDrawProperties(root);
DCHECK(copy_layer->test_properties()->copy_requests.empty());
EXPECT_EQ(gfx::Rect(50, 50),
cache_render_surface_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(100, 100), copy_layer->visible_layer_rect());
EXPECT_EQ(gfx::Rect(20, 20), copy_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(15, 15), copy_clipped_child->visible_layer_rect());
EXPECT_EQ(gfx::Rect(20, 20), cache_surface->visible_layer_rect());
}
TEST_F(LayerTreeHostCommonTest, BuildPropertyNodesForScrollChildrenInOrder) {
// This test is intended to test against a bug that scroll children were
// visited in unspecified order, which can cause data dependency to fail
// while resolving clip parent.
// Try multiple times because in the original bug the probability to fail
// was 50%, depends on the hash values.
int trial = 10;
while (trial--) {
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> scroller = Layer::Create();
scoped_refptr<Layer> scroll_sibling_1 = Layer::Create();
scoped_refptr<Layer> scroll_sibling_2 = Layer::Create();
scoped_refptr<Layer> clip_escaper = Layer::Create();
root->SetBounds(gfx::Size(100, 100));
scroll_sibling_1->SetBounds(gfx::Size(10, 20));
scroll_sibling_1->SetMasksToBounds(true);
scroll_sibling_2->SetBounds(gfx::Size(20, 10));
scroll_sibling_2->SetMasksToBounds(true);
clip_escaper->SetBounds(gfx::Size(30, 30));
clip_escaper->SetIsDrawable(true);
host()->SetRootLayer(root);
root->AddChild(scroller.get());
root->AddChild(scroll_sibling_1.get());
root->AddChild(scroll_sibling_2.get());
scroll_sibling_2->AddChild(clip_escaper.get());
// Also randomize scroll children insertion order.
if (trial & 1) {
scroll_sibling_1->SetScrollParent(scroller.get());
scroll_sibling_2->SetScrollParent(scroller.get());
} else {
scroll_sibling_2->SetScrollParent(scroller.get());
scroll_sibling_1->SetScrollParent(scroller.get());
}
clip_escaper->SetClipParent(scroll_sibling_1.get());
ExecuteCalculateDrawProperties(root.get());
EXPECT_EQ(scroll_sibling_1->clip_tree_index(),
clip_escaper->clip_tree_index());
}
}
TEST_F(LayerTreeHostCommonTest, RenderSurfaceListForTrilinearFiltering) {
LayerImpl* root = root_layer_for_testing();
LayerImpl* parent = AddChild<LayerImpl>(root);
LayerImpl* child1 = AddChild<LayerImpl>(parent);
LayerImpl* child2 = AddChild<LayerImpl>(parent);
gfx::Transform scale_matrix;
scale_matrix.Scale(.25f, .25f);
root->SetBounds(gfx::Size(200, 200));
parent->test_properties()->transform = scale_matrix;
parent->test_properties()->trilinear_filtering = true;
child1->SetBounds(gfx::Size(50, 50));
child1->SetDrawsContent(true);
child1->test_properties()->force_render_surface = true;
child2->test_properties()->position = gfx::PointF(50, 50);
child2->SetBounds(gfx::Size(50, 50));
child2->SetDrawsContent(true);
child2->test_properties()->force_render_surface = true;
RenderSurfaceList render_surface_list;
LayerTreeHostCommon::CalcDrawPropsImplInputsForTesting inputs(
root, root->bounds(), &render_surface_list);
inputs.can_adjust_raster_scales = true;
LayerTreeHostCommon::CalculateDrawPropertiesForTesting(&inputs);
ASSERT_TRUE(GetRenderSurface(parent));
EXPECT_EQ(2, GetRenderSurface(parent)->num_contributors());
EXPECT_EQ(4U, render_surface_list.size());
// The rectangle enclosing child1 and child2 (0,0 100x100), scaled by the
// scale matrix to (0,0 25x25).
EXPECT_EQ(gfx::RectF(0, 0, 25, 25),
GetRenderSurface(parent)->DrawableContentRect());
}
TEST_F(LayerTreeHostCommonTest, VisibleRectClippedByViewport) {
FakeImplTaskRunnerProvider task_runner_provider;
TestTaskGraphRunner task_graph_runner;
FakeLayerTreeHostImpl host_impl(&task_runner_provider, &task_graph_runner);
gfx::Size root_layer_size = gfx::Size(300, 500);
gfx::Size device_viewport_size = gfx::Size(300, 600);
gfx::Rect viewport_visible_rect = gfx::Rect(100, 100, 200, 200);
host_impl.active_tree()->SetDeviceViewportSize(device_viewport_size);
host_impl.active_tree()->SetViewportVisibleRect(viewport_visible_rect);
host_impl.active_tree()->SetRootLayerForTesting(
LayerImpl::Create(host_impl.active_tree(), 1));
LayerImpl* root = host_impl.active_tree()->root_layer_for_testing();
root->SetBounds(root_layer_size);
root->SetDrawsContent(true);
ExecuteCalculateDrawProperties(root);
EXPECT_EQ(viewport_visible_rect, root->visible_layer_rect());
EXPECT_EQ(gfx::Rect(root_layer_size), root->drawable_content_rect());
}
TEST_F(LayerTreeHostCommonTest, RoundedCornerBounds) {
// Layer Tree:
// +root
// +--render surface
// +----rounded corner layer 1 [should trigger render surface]
// +----layer 1
// +--rounded corner layer 2 [should trigger render surface]
// +----layer 2
// +------rounded corner layer 3 [should trigger render surface]
// +--------rounded corner layer 4 [should trigger render surface]
constexpr int kRoundedCorner1Radius = 2;
constexpr int kRoundedCorner2Radius = 5;
constexpr int kRoundedCorner3Radius = 1;
constexpr int kRoundedCorner4Radius = 1;
constexpr gfx::RectF kRoundedCornerLayer1Bound(15.f, 15.f, 20.f, 20.f);
constexpr gfx::RectF kRoundedCornerLayer2Bound(40.f, 40.f, 60.f, 60.f);
constexpr gfx::RectF kRoundedCornerLayer3Bound(0.f, 15.f, 5.f, 5.f);
constexpr gfx::RectF kRoundedCornerLayer4Bound(1.f, 1.f, 3.f, 3.f);
constexpr float kDeviceScale = 1.6f;
scoped_refptr<Layer> root = Layer::Create();
scoped_refptr<Layer> render_surface = Layer::Create();
scoped_refptr<Layer> rounded_corner_layer_1 = Layer::Create();
scoped_refptr<Layer> layer_1 = Layer::Create();
scoped_refptr<Layer> rounded_corner_layer_2 = Layer::Create();
scoped_refptr<Layer> layer_2 = Layer::Create();
scoped_refptr<Layer> rounded_corner_layer_3 = Layer::Create();
scoped_refptr<Layer> rounded_corner_layer_4 = Layer::Create();
// Set up layer tree
root->AddChild(render_surface);
root->AddChild(rounded_corner_layer_2);
render_surface->AddChild(rounded_corner_layer_1);
render_surface->AddChild(layer_1);
rounded_corner_layer_2->AddChild(layer_2);
layer_2->AddChild(rounded_corner_layer_3);
rounded_corner_layer_3->AddChild(rounded_corner_layer_4);
// Set the root layer on host.
host()->SetRootLayer(root);
// Set layer positions.
render_surface->SetPosition(gfx::PointF(0, 0));
rounded_corner_layer_1->SetPosition(kRoundedCornerLayer1Bound.origin());
layer_1->SetPosition(gfx::PointF(10.f, 10.f));
rounded_corner_layer_2->SetPosition(kRoundedCornerLayer2Bound.origin());
layer_2->SetPosition(gfx::PointF(30.f, 30.f));
rounded_corner_layer_3->SetPosition(kRoundedCornerLayer3Bound.origin());
rounded_corner_layer_4->SetPosition(kRoundedCornerLayer4Bound.origin());
// Set up layer bounds.
root->SetBounds(gfx::Size(100, 100));
render_surface->SetBounds(gfx::Size(50, 50));
rounded_corner_layer_1->SetBounds(
gfx::ToRoundedSize(kRoundedCornerLayer1Bound.size()));
layer_1->SetBounds(gfx::Size(10, 10));
rounded_corner_layer_2->SetBounds(
gfx::ToRoundedSize(kRoundedCornerLayer2Bound.size()));
layer_2->SetBounds(gfx::Size(25, 25));
rounded_corner_layer_3->SetBounds(
gfx::ToRoundedSize(kRoundedCornerLayer3Bound.size()));
rounded_corner_layer_4->SetBounds(
gfx::ToRoundedSize(kRoundedCornerLayer4Bound.size()));
// Add Layer transforms.
gfx::Transform layer_2_transform;
constexpr gfx::Vector2dF kLayer2Translation(10.f, 10.f);
layer_2_transform.Translate(kLayer2Translation);
layer_2->SetTransform(layer_2_transform);
gfx::Transform rounded_corner_layer_3_transform;
constexpr float kRoundedCorner3Scale = 2.f;
rounded_corner_layer_3_transform.Scale(kRoundedCorner3Scale,
kRoundedCorner3Scale);
rounded_corner_layer_3->SetTransform(rounded_corner_layer_3_transform);
// Set the layer properties
render_surface->SetForceRenderSurfaceForTesting(true);
root->SetIsDrawable(true);
render_surface->SetIsDrawable(true);
rounded_corner_layer_1->SetIsDrawable(true);
layer_1->SetIsDrawable(true);
rounded_corner_layer_2->SetIsDrawable(true);
layer_2->SetIsDrawable(true);
rounded_corner_layer_3->SetIsDrawable(true);
rounded_corner_layer_4->SetIsDrawable(true);
// Set Rounded corners
rounded_corner_layer_1->SetRoundedCorner(
{kRoundedCorner1Radius, kRoundedCorner1Radius, kRoundedCorner1Radius,
kRoundedCorner1Radius});
rounded_corner_layer_2->SetRoundedCorner(
{kRoundedCorner2Radius, kRoundedCorner2Radius, kRoundedCorner2Radius,
kRoundedCorner2Radius});
rounded_corner_layer_3->SetRoundedCorner(
{kRoundedCorner3Radius, kRoundedCorner3Radius, kRoundedCorner3Radius,
kRoundedCorner3Radius});
rounded_corner_layer_4->SetRoundedCorner(
{kRoundedCorner4Radius, kRoundedCorner4Radius, kRoundedCorner4Radius,
kRoundedCorner4Radius});
ExecuteCalculateDrawProperties(root.get(), kDeviceScale);
const EffectTree& effect_tree =
root->layer_tree_host()->property_trees()->effect_tree;
// Each effect node should have a render surface as rounded corner triggers
// a render surface.
const EffectNode* effect_node =
effect_tree.Node(rounded_corner_layer_1->effect_tree_index());
gfx::RRectF rounded_corner_bounds_1 = effect_node->rounded_corner_bounds;
EXPECT_TRUE(effect_node->has_render_surface);
EXPECT_FLOAT_EQ(rounded_corner_bounds_1.GetSimpleRadius(),
kRoundedCorner1Radius);
EXPECT_EQ(rounded_corner_bounds_1.rect(),
gfx::RectF(kRoundedCornerLayer1Bound.size()));
effect_node = effect_tree.Node(rounded_corner_layer_2->effect_tree_index());
gfx::RRectF rounded_corner_bounds_2 = effect_node->rounded_corner_bounds;
EXPECT_TRUE(effect_node->has_render_surface);
EXPECT_FLOAT_EQ(rounded_corner_bounds_2.GetSimpleRadius(),
kRoundedCorner2Radius);
EXPECT_EQ(rounded_corner_bounds_2.rect(),
gfx::RectF(kRoundedCornerLayer2Bound.size()));
effect_node = effect_tree.Node(rounded_corner_layer_3->effect_tree_index());
gfx::RRectF rounded_corner_bounds_3 = effect_node->rounded_corner_bounds;
EXPECT_TRUE(effect_node->has_render_surface);
EXPECT_FLOAT_EQ(rounded_corner_bounds_3.GetSimpleRadius(),
kRoundedCorner3Radius);
EXPECT_EQ(rounded_corner_bounds_3.rect(),
gfx::RectF(kRoundedCornerLayer3Bound.size()));
effect_node = effect_tree.Node(rounded_corner_layer_4->effect_tree_index());
gfx::RRectF rounded_corner_bounds_4 = effect_node->rounded_corner_bounds;
EXPECT_TRUE(effect_node->has_render_surface);
EXPECT_FLOAT_EQ(rounded_corner_bounds_4.GetSimpleRadius(),
kRoundedCorner4Radius);
EXPECT_EQ(rounded_corner_bounds_4.rect(),
gfx::RectF(kRoundedCornerLayer4Bound.size()));
host()->host_impl()->CreatePendingTree();
host()->CommitAndCreatePendingTree();
// TODO(https://crbug.com/939968) This call should be handled by
// FakeLayerTreeHost instead of manually pushing the properties from the
// layer tree host to the pending tree.
root->layer_tree_host()->PushLayerTreePropertiesTo(host()->pending_tree());
host()->host_impl()->ActivateSyncTree();
LayerTreeImpl* layer_tree_impl = host()->host_impl()->active_tree();
// Get the layer impl for each Layer.
LayerImpl* root_impl = layer_tree_impl->LayerById(root->id());
LayerImpl* rounded_corner_layer_1_impl =
layer_tree_impl->LayerById(rounded_corner_layer_1->id());
LayerImpl* rounded_corner_layer_2_impl =
layer_tree_impl->LayerById(rounded_corner_layer_2->id());
LayerImpl* rounded_corner_layer_3_impl =
layer_tree_impl->LayerById(rounded_corner_layer_3->id());
LayerImpl* rounded_corner_layer_4_impl =
layer_tree_impl->LayerById(rounded_corner_layer_4->id());
// Set the root layer on host.
ExecuteCalculateDrawProperties(root_impl, kDeviceScale);
// Rounded corner layer 1
// The render target for this layer is |render_surface|, hence its target
// bounds are relative to |render_surface|.
// The offset from the origin of the render target is [15, 15] and the device
// scale factor is 1.6 thus giving the target space origin of [24, 24]. The
// corner radius is also scaled by a factor of 1.6.
const gfx::RRectF actual_self_rrect_1 =
rounded_corner_layer_1_impl->draw_properties().rounded_corner_bounds;
EXPECT_TRUE(actual_self_rrect_1.IsEmpty());
gfx::RectF bounds_in_target_space = kRoundedCornerLayer1Bound;
bounds_in_target_space.Scale(kDeviceScale);
const gfx::RRectF actual_render_target_rrect_1 =
rounded_corner_layer_1_impl->render_target()->rounded_corner_bounds();
EXPECT_EQ(actual_render_target_rrect_1.rect(), bounds_in_target_space);
EXPECT_FLOAT_EQ(actual_render_target_rrect_1.GetSimpleRadius(),
kRoundedCorner1Radius * kDeviceScale);
// Rounded corner layer 2
// The render target for this layer is |root|.
// The offset from the origin of the render target is [40, 40] and the device
// scale factor is 1.6 thus giving the target space origin of [64, 64]. The
// corner radius is also scaled by a factor of 1.6.
const gfx::RRectF actual_self_rrect_2 =
rounded_corner_layer_2_impl->draw_properties().rounded_corner_bounds;
EXPECT_TRUE(actual_self_rrect_2.IsEmpty());
bounds_in_target_space = kRoundedCornerLayer2Bound;
bounds_in_target_space.Scale(kDeviceScale);
const gfx::RRectF actual_render_target_rrect_2 =
rounded_corner_layer_2_impl->render_target()->rounded_corner_bounds();
EXPECT_EQ(actual_render_target_rrect_2.rect(), bounds_in_target_space);
EXPECT_FLOAT_EQ(actual_render_target_rrect_2.GetSimpleRadius(),
kRoundedCorner2Radius * kDeviceScale);
// Rounded corner layer 3
// The render target for this layer is |rounded_corner_2|.
// The net offset from the origin of the render target is [40, 55] and the
// device scale factor is 1.6 thus giving the target space origin of [64, 88].
// The corner radius is also scaled by a factor of 1.6 * transform scale.
const gfx::RRectF actual_self_rrect_3 =
rounded_corner_layer_3_impl->draw_properties().rounded_corner_bounds;
EXPECT_TRUE(actual_self_rrect_3.IsEmpty());
bounds_in_target_space = kRoundedCornerLayer3Bound;
bounds_in_target_space +=
layer_2->position().OffsetFromOrigin() + kLayer2Translation;
bounds_in_target_space.Scale(kDeviceScale);
gfx::SizeF transformed_size = bounds_in_target_space.size();
transformed_size.Scale(kRoundedCorner3Scale);
bounds_in_target_space.set_size(transformed_size);
const gfx::RRectF actual_render_target_rrect_3 =
rounded_corner_layer_3_impl->render_target()->rounded_corner_bounds();
EXPECT_EQ(actual_render_target_rrect_3.rect(), bounds_in_target_space);
EXPECT_FLOAT_EQ(actual_render_target_rrect_3.GetSimpleRadius(),
kRoundedCorner3Radius * kDeviceScale * kRoundedCorner3Scale);
// Rounded corner layer 4
// The render target for this layer is |rounded_corner_3|.
// The net offset from the origin of the render target is [1, 1] and the
// net scale is 1.6 * transform scale = 3.2 thus giving the target space o
// rigin of [3.2, 3.2].
// The corner radius is also scaled by a factor of 3.2.
const gfx::RRectF actual_self_rrect_4 =
rounded_corner_layer_4_impl->draw_properties().rounded_corner_bounds;
EXPECT_TRUE(actual_self_rrect_4.IsEmpty());
bounds_in_target_space = kRoundedCornerLayer4Bound;
bounds_in_target_space.Scale(kDeviceScale * kRoundedCorner3Scale);
const gfx::RRectF actual_render_target_rrect_4 =
rounded_corner_layer_4_impl->render_target()->rounded_corner_bounds();
EXPECT_EQ(actual_render_target_rrect_4.rect(), bounds_in_target_space);
EXPECT_FLOAT_EQ(actual_render_target_rrect_4.GetSimpleRadius(),
kRoundedCorner4Radius * kDeviceScale * kRoundedCorner3Scale);
}
} // namespace
} // namespace cc