#include "cc/trees/damage_tracker.h"
#include <stddef.h>
#include <limits>
#include "base/memory/ptr_util.h"
#include "base/memory/raw_ptr.h"
#include "cc/base/math_util.h"
#include "cc/layers/layer_impl.h"
#include "cc/layers/view_transition_content_layer_impl.h"
#include "cc/paint/filter_operation.h"
#include "cc/paint/filter_operations.h"
#include "cc/test/fake_picture_layer_impl.h"
#include "cc/test/fake_raster_source.h"
#include "cc/test/layer_tree_impl_test_base.h"
#include "cc/test/property_tree_test_utils.h"
#include "cc/trees/effect_node.h"
#include "cc/trees/layer_tree_impl.h"
#include "cc/trees/single_thread_proxy.h"
#include "cc/trees/transform_node.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/vector2d_f.h"
namespace cc {
namespace {
class TestLayerImpl : public LayerImpl {
public:
static std::unique_ptr<TestLayerImpl> Create(LayerTreeImpl* tree_impl,
int id) {
return base::WrapUnique(new TestLayerImpl(tree_impl, id));
}
void AddDamageRect(const gfx::Rect& damage_rect);
gfx::Rect GetDamageRect() const override;
void ResetChangeTracking() override;
private:
TestLayerImpl(LayerTreeImpl* tree_impl, int id);
gfx::Rect damage_rect_;
};
TestLayerImpl::TestLayerImpl(LayerTreeImpl* tree_impl, int id)
: LayerImpl(tree_impl, id) {}
void TestLayerImpl::AddDamageRect(const gfx::Rect& damage_rect) {
damage_rect_.Union(damage_rect);
}
gfx::Rect TestLayerImpl::GetDamageRect() const {
return damage_rect_;
}
void TestLayerImpl::ResetChangeTracking() {
LayerImpl::ResetChangeTracking();
damage_rect_.SetRect(0, 0, 0, 0);
}
class TestViewTransitionContentLayerImpl
: public ViewTransitionContentLayerImpl {
public:
static std::unique_ptr<TestViewTransitionContentLayerImpl> Create(
LayerTreeImpl* tree_impl,
int id,
const viz::ViewTransitionElementResourceId& resource_id,
bool is_live_content_layer) {
return base::WrapUnique(new TestViewTransitionContentLayerImpl(
tree_impl, id, resource_id, is_live_content_layer));
}
void AddDamageRect(const gfx::Rect& damage_rect);
gfx::Rect GetDamageRect() const override;
void ResetChangeTracking() override;
private:
TestViewTransitionContentLayerImpl(
LayerTreeImpl* tree_impl,
int id,
const viz::ViewTransitionElementResourceId& resource_id,
bool is_live_content_layer);
gfx::Rect damage_rect_;
};
TestViewTransitionContentLayerImpl::TestViewTransitionContentLayerImpl(
LayerTreeImpl* tree_impl,
int id,
const viz::ViewTransitionElementResourceId& resource_id,
bool is_live_content_layer)
: ViewTransitionContentLayerImpl(tree_impl,
id,
resource_id,
is_live_content_layer) {}
void TestViewTransitionContentLayerImpl::AddDamageRect(
const gfx::Rect& damage_rect) {
damage_rect_.Union(damage_rect);
}
gfx::Rect TestViewTransitionContentLayerImpl::GetDamageRect() const {
return damage_rect_;
}
void TestViewTransitionContentLayerImpl::ResetChangeTracking() {
LayerImpl::ResetChangeTracking();
damage_rect_.SetRect(0, 0, 0, 0);
}
void ClearDamageForAllSurfaces(LayerImpl* root) {
for (auto* layer : *root->layer_tree_impl()) {
if (GetRenderSurface(layer))
GetRenderSurface(layer)->damage_tracker()->DidDrawDamagedArea();
layer->ResetChangeTracking();
}
}
void SetCopyRequest(LayerImpl* root) {
auto* root_node =
root->layer_tree_impl()->property_trees()->effect_tree_mutable().Node(
root->effect_tree_index());
root_node->has_copy_request = true;
root->layer_tree_impl()
->property_trees()
->effect_tree_mutable()
.set_needs_update(true);
}
class DamageTrackerTest : public LayerTreeImplTestBase, public testing::Test {
public:
LayerImpl* CreateTestTreeWithOneSurface(int number_of_children) {
ClearLayersAndProperties();
LayerImpl* root = root_layer();
root->SetBounds(gfx::Size(500, 500));
root->layer_tree_impl()->SetDeviceViewportRect(gfx::Rect(root->bounds()));
root->SetDrawsContent(true);
SetupRootProperties(root);
child_layers_.resize(number_of_children);
for (int i = 0; i < number_of_children; ++i) {
auto* child = AddLayer<TestLayerImpl>();
child->SetBounds(gfx::Size(30, 30));
child->SetDrawsContent(true);
child_layers_[i] = child;
CopyProperties(root, child);
child->SetOffsetToTransformParent(gfx::Vector2dF(100.f, 100.f));
}
SetElementIdsForTesting();
return root;
}
LayerImpl* CreateTestTreeWithTwoSurfaces() {
ClearLayersAndProperties();
LayerImpl* root = root_layer();
root->SetBounds(gfx::Size(500, 500));
root->layer_tree_impl()->SetDeviceViewportRect(gfx::Rect(root->bounds()));
root->SetDrawsContent(true);
SetupRootProperties(root);
child1_ = AddLayer<TestLayerImpl>();
grand_child1_ = AddLayer<TestLayerImpl>();
grand_child2_ = AddLayer<TestLayerImpl>();
child2_ = AddLayer<TestLayerImpl>();
SetElementIdsForTesting();
child1_->SetBounds(gfx::Size(30, 30));
child1_->SetDrawsContent(false);
CopyProperties(root, child1_);
CreateTransformNode(child1_).post_translation =
gfx::Vector2dF(100.f, 100.f);
CreateEffectNode(child1_).render_surface_reason =
RenderSurfaceReason::kTest;
grand_child1_->SetBounds(gfx::Size(6, 8));
grand_child1_->SetDrawsContent(true);
CopyProperties(child1_, grand_child1_);
grand_child1_->SetOffsetToTransformParent(gfx::Vector2dF(200.f, 200.f));
grand_child2_->SetBounds(gfx::Size(6, 8));
grand_child2_->SetDrawsContent(true);
CopyProperties(child1_, grand_child2_);
grand_child2_->SetOffsetToTransformParent(gfx::Vector2dF(190.f, 190.f));
child2_->SetBounds(gfx::Size(18, 18));
child2_->SetDrawsContent(true);
CopyProperties(root, child2_);
child2_->SetOffsetToTransformParent(gfx::Vector2dF(11.f, 11.f));
return root;
}
LayerImpl* CreateTestTreeWithFourSurfaces() {
ClearLayersAndProperties();
LayerImpl* root = root_layer();
root->SetBounds(gfx::Size(500, 500));
root->layer_tree_impl()->SetDeviceViewportRect(gfx::Rect(root->bounds()));
root->SetDrawsContent(true);
SetupRootProperties(root);
child1_ = AddLayer<TestLayerImpl>();
grand_child1_ = AddLayer<TestLayerImpl>();
grand_child2_ = AddLayer<TestLayerImpl>();
grand_child3_ = AddLayer<TestLayerImpl>();
grand_child4_ = AddLayer<TestLayerImpl>();
child2_ = AddLayer<TestLayerImpl>();
SetElementIdsForTesting();
child1_->SetBounds(gfx::Size(30, 30));
child1_->SetDrawsContent(false);
CopyProperties(root, child1_);
CreateTransformNode(child1_).post_translation =
gfx::Vector2dF(100.f, 100.f);
CreateEffectNode(child1_).render_surface_reason =
RenderSurfaceReason::kTest;
grand_child1_->SetBounds(gfx::Size(6, 8));
grand_child1_->SetDrawsContent(true);
CopyProperties(child1_, grand_child1_);
grand_child1_->SetOffsetToTransformParent(gfx::Vector2dF(200.f, 200.f));
grand_child2_->SetBounds(gfx::Size(6, 8));
grand_child2_->SetDrawsContent(true);
CopyProperties(child1_, grand_child2_);
grand_child2_->SetOffsetToTransformParent(gfx::Vector2dF(190.f, 190.f));
grand_child3_->SetBounds(gfx::Size(6, 8));
grand_child3_->SetDrawsContent(true);
CopyProperties(child1_, grand_child3_);
CreateEffectNode(grand_child3_).render_surface_reason =
RenderSurfaceReason::kTest;
CreateTransformNode(grand_child3_).post_translation =
gfx::Vector2dF(170.f, 170.f);
grand_child4_->SetBounds(gfx::Size(15, 16));
grand_child4_->SetDrawsContent(true);
CopyProperties(child1_, grand_child4_);
CreateEffectNode(grand_child4_).render_surface_reason =
RenderSurfaceReason::kTest;
CreateTransformNode(grand_child4_).post_translation =
gfx::Vector2dF(180.f, 180.f);
child2_->SetBounds(gfx::Size(18, 18));
child2_->SetDrawsContent(true);
CopyProperties(root, child2_);
child2_->SetOffsetToTransformParent(gfx::Vector2dF(11.f, 11.f));
return root;
}
LayerImpl* CreateAndSetUpTestTreeWithOneSurface(int number_of_children = 1) {
LayerImpl* root = CreateTestTreeWithOneSurface(number_of_children);
EmulateDrawingOneFrame(root);
return root;
}
LayerImpl* CreateAndSetUpTestTreeWithTwoSurfaces() {
LayerImpl* root = CreateTestTreeWithTwoSurfaces();
EmulateDrawingOneFrame(root);
return root;
}
LayerImpl* CreateAndSetUpTestTreeWithTwoSurfacesDrawingFullyVisible() {
LayerImpl* root = CreateTestTreeWithTwoSurfaces();
root->layer_tree_impl()
->property_trees()
->effect_tree_mutable()
.Node(child1_->effect_tree_index())
->backdrop_filters.Append(
FilterOperation::CreateZoomFilter(2.f , 0 ));
EmulateDrawingOneFrame(root);
return root;
}
LayerImpl* CreateAndSetUpTestTreeWithFourSurfaces() {
LayerImpl* root = CreateTestTreeWithFourSurfaces();
EmulateDrawingOneFrame(root);
return root;
}
void EmulateDrawingOneFrame(LayerImpl* root,
float device_scale_factor = 1.f) {
root->layer_tree_impl()->SetDeviceScaleFactor(device_scale_factor);
root->layer_tree_impl()->set_needs_update_draw_properties();
UpdateDrawProperties(root->layer_tree_impl());
DamageTracker::UpdateDamageTracking(root->layer_tree_impl());
root->layer_tree_impl()->ResetAllChangeTracking();
}
protected:
void ClearLayersAndProperties() {
host_impl()->active_tree()->DetachLayersKeepingRootLayerForTesting();
host_impl()->active_tree()->property_trees()->clear();
child_layers_.clear();
child1_ = child2_ = grand_child1_ = grand_child2_ = grand_child3_ =
grand_child4_ = nullptr;
}
std::vector<TestLayerImpl*> child_layers_;
raw_ptr<TestLayerImpl> child1_ = nullptr;
raw_ptr<TestLayerImpl> child2_ = nullptr;
raw_ptr<TestLayerImpl> grand_child1_ = nullptr;
raw_ptr<TestLayerImpl> grand_child2_ = nullptr;
raw_ptr<TestLayerImpl> grand_child3_ = nullptr;
raw_ptr<TestLayerImpl> grand_child4_ = nullptr;
};
TEST_F(DamageTrackerTest, SanityCheckTestTreeWithOneSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
EXPECT_EQ(2, GetRenderSurface(root)->num_contributors());
EXPECT_TRUE(root->contributes_to_drawn_render_surface());
EXPECT_TRUE(child->contributes_to_drawn_render_surface());
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(500, 500).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, SanityCheckTestTreeWithTwoSurfaces) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_NE(GetRenderSurface(child1_), GetRenderSurface(root));
EXPECT_EQ(GetRenderSurface(child2_), GetRenderSurface(root));
EXPECT_EQ(3, GetRenderSurface(root)->num_contributors());
EXPECT_EQ(2, GetRenderSurface(child1_)->num_contributors());
EXPECT_EQ(gfx::Rect(190, 190, 16, 18).ToString(),
child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(500, 500).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForUpdateRects) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(10, 11, 12, 13));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(110, 111, 12, 13).ToString(),
root_damage_rect.ToString());
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(10, 11, 12, 13));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(110, 111, 12, 13).ToString(),
root_damage_rect.ToString());
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(20, 25, 1, 2));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(120, 125, 1, 2).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForLayerDamageRects) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
TestLayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(10, 11, 12, 13));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(110, 111, 12, 13)));
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(10, 11, 12, 13));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(110, 111, 12, 13)));
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(20, 25, 1, 2));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(120, 125, 1, 2)));
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(20, 25, 1, 2));
child->AddDamageRect(gfx::Rect(10, 15, 3, 4));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(120, 125, 1, 2)));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(110, 115, 3, 4)));
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForLayerUpdateAndDamageRects) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
TestLayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(5, 6, 12, 13));
child->UnionUpdateRect(gfx::Rect(15, 16, 14, 10));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(105, 106, 24, 20)));
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(10, 11, 12, 13));
child->UnionUpdateRect(gfx::Rect(10, 11, 14, 15));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(110, 111, 14, 15)));
ClearDamageForAllSurfaces(root);
child->AddDamageRect(gfx::Rect(20, 25, 2, 3));
child->UnionUpdateRect(gfx::Rect(5, 10, 7, 8));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(true, root_damage_rect.Contains(gfx::Rect(105, 110, 17, 18)));
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForPropertyChanges) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(10, 11, 12, 13));
root->layer_tree_impl()->SetOpacityMutated(child->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
ASSERT_EQ(2, GetRenderSurface(root)->num_contributors());
gfx::Rect expected_rect = gfx::Rect(100, 100, 30, 30);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(expected_rect.ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(root_damage_rect.IsEmpty());
ClearDamageForAllSurfaces(root);
CreateTransformNode(child).post_translation =
child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
gfx::Transform translation;
translation.Translate(100.f, 130.f);
root->layer_tree_impl()->SetTransformMutated(child->element_id(),
translation);
EmulateDrawingOneFrame(root);
expected_rect.Union(gfx::Rect(200, 230, 30, 30));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(expected_rect, root_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest,
VerifyDamageForPropertyChangesFromContributingContents) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(child1_->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
CreateEffectNode(child2_).render_surface_reason = RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(child2_->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
CreateEffectNode(grand_child1_).render_surface_reason =
RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(grand_child1_->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, EffectPropertyChangeNoSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
auto& effect_node = CreateEffectNode(child);
effect_node.opacity = 0.5;
effect_node.has_potential_opacity_animation = true;
EmulateDrawingOneFrame(root);
EXPECT_EQ(root->transform_tree_index(), child->transform_tree_index());
EXPECT_NE(root->effect_tree_index(), child->effect_tree_index());
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(child->element_id(), 0.4f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, TransformPropertyChangeNoSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Transform trans1;
trans1.Scale(2, 1);
CreateTransformNode(child).local = trans1;
EmulateDrawingOneFrame(root);
EXPECT_NE(root->transform_tree_index(), child->transform_tree_index());
EXPECT_EQ(root->effect_tree_index(), child->effect_tree_index());
ClearDamageForAllSurfaces(root);
gfx::Transform trans2;
trans2.Scale(1, 2);
root->layer_tree_impl()->SetTransformMutated(child->element_id(), trans2);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest,
VerifyDamageForUpdateAndDamageRectsFromContributingContents) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
child1_->SetDrawsContent(true);
ClearDamageForAllSurfaces(root);
child1_->AddDamageRect(gfx::Rect(105, 106, 12, 15));
child1_->UnionUpdateRect(gfx::Rect(115, 116, 12, 15));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
child2_->AddDamageRect(gfx::Rect(11, 11, 12, 15));
child2_->UnionUpdateRect(gfx::Rect(12, 12, 12, 15));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
grand_child1_->AddDamageRect(gfx::Rect(1, 0, 2, 5));
grand_child1_->UnionUpdateRect(gfx::Rect(2, 1, 2, 5));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageWhenSurfaceRemoved) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
LayerImpl* surface = child1_;
LayerImpl* child = grand_child1_;
child->SetDrawsContent(true);
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
SetRenderSurfaceReason(surface, RenderSurfaceReason::kNone);
child->SetDrawsContent(false);
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(290, 290, 16, 18).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForTransformedLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Transform rotation;
rotation.Rotate(45.0);
ClearDamageForAllSurfaces(root);
auto& transform_node = CreateTransformNode(child);
transform_node.origin = gfx::Point3F(child->bounds().width() * 0.5f,
child->bounds().height() * 0.5f, 0.f);
transform_node.post_translation = gfx::Vector2dF(85.f, 85.f);
child->SetOffsetToTransformParent(gfx::Vector2dF());
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
child->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(85, 85, 45, 45).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetTransformMutated(child->element_id(), rotation);
EmulateDrawingOneFrame(root);
float expected_width = 30.f * sqrt(2.f);
float expected_position = 100.f - 0.5f * expected_width;
gfx::Rect expected_rect = gfx::ToEnclosingRect(gfx::RectF(
expected_position, expected_position, expected_width, expected_width));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(expected_rect.ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForPerspectiveClippedLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Transform transform;
transform.Translate3d(550.0, 500.0, 0.0);
transform.ApplyPerspectiveDepth(1.0);
transform.RotateAboutYAxis(45.0);
transform.Translate3d(-50.0, -50.0, 0.0);
child->SetBounds(gfx::Size(100, 100));
CreateTransformNode(child).local = transform;
child->SetOffsetToTransformParent(gfx::Vector2dF());
EmulateDrawingOneFrame(root);
gfx::RectF test_rect(gfx::SizeF(child->bounds()));
bool clipped = false;
MathUtil::MapQuad(transform, gfx::QuadF(test_rect), &clipped);
EXPECT_TRUE(clipped);
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(child->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
gfx::Rect damage_we_care_about = gfx::Rect(gfx::Size(500, 500));
EXPECT_TRUE(root_damage_rect.Contains(damage_we_care_about));
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForBlurredSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
LayerImpl* surface = child1_;
LayerImpl* child = grand_child1_;
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(5.f));
ClearDamageForAllSurfaces(root);
SetFilter(surface, filters);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(surface)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(1, 2, 3, 4));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(286, 287, 33, 34), root_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(surface)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForImageFilter) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Rect root_damage_rect, child_damage_rect;
child->SetDrawsContent(true);
FilterOperations filters;
filters.Append(FilterOperation::CreateReferenceFilter(
sk_make_sp<BlurPaintFilter>(2, 2, SkTileMode::kDecal, nullptr)));
CreateTransformNode(child).post_translation =
child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
child->layer_tree_impl()->SetFilterMutated(child->element_id(), filters);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(94, 94, 42, 42), root_damage_rect);
EXPECT_EQ(gfx::Rect(-6, -6, 42, 42), child_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(1, 1));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(94, 94, 13, 13), root_damage_rect);
EXPECT_EQ(gfx::Rect(-6, -6, 13, 13), child_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(), root_damage_rect);
EXPECT_EQ(gfx::Rect(), child_damage_rect);
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForTransformedImageFilter) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Rect root_damage_rect, child_damage_rect;
child->SetDrawsContent(true);
FilterOperations filters;
filters.Append(FilterOperation::CreateReferenceFilter(
sk_make_sp<BlurPaintFilter>(2, 2, SkTileMode::kDecal, nullptr)));
gfx::Transform transform;
transform.RotateAboutYAxis(60);
ClearDamageForAllSurfaces(root);
auto& transform_node = CreateTransformNode(child);
transform_node.local = transform;
transform_node.post_translation = child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
auto& effect_node = CreateEffectNode(child);
effect_node.render_surface_reason = RenderSurfaceReason::kFilter;
effect_node.has_potential_filter_animation = true;
EmulateDrawingOneFrame(root);
child->layer_tree_impl()->SetFilterMutated(child->element_id(), filters);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
int blur_outset = 6;
int rotated_outset_left = blur_outset / 2;
int expected_rotated_width = (30 + 2 * blur_outset) / 2;
gfx::Rect expected_root_damage(100 - rotated_outset_left, 100 - blur_outset,
expected_rotated_width, 30 + 2 * blur_outset);
expected_root_damage.Union(gfx::Rect(100, 100, 30, 30));
EXPECT_EQ(expected_root_damage, root_damage_rect);
EXPECT_EQ(gfx::Rect(-blur_outset, -blur_outset, 30 + 2 * blur_outset,
30 + 2 * blur_outset),
child_damage_rect);
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(30, 30));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
int expect_width = 30 + 2 * blur_outset;
int expect_height = 30 + 2 * blur_outset;
EXPECT_EQ(gfx::Rect(100 - blur_outset / 2, 100 - blur_outset,
expect_width / 2, expect_height),
root_damage_rect);
EXPECT_EQ(gfx::Rect(-blur_outset, -blur_outset, expect_width, expect_height),
child_damage_rect);
}
TEST_F(DamageTrackerTest, VerifyDamageForHighDPIImageFilter) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
gfx::Rect root_damage_rect, child_damage_rect;
ClearDamageForAllSurfaces(root);
int device_scale_factor = 2;
EmulateDrawingOneFrame(root, device_scale_factor);
child->SetDrawsContent(true);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(3.f));
ClearDamageForAllSurfaces(root);
CreateTransformNode(child).post_translation =
child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
ClearDamageForAllSurfaces(root);
child->layer_tree_impl()->SetFilterMutated(child->element_id(), filters);
EmulateDrawingOneFrame(root, device_scale_factor);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
int blur_outset = 9 * device_scale_factor;
gfx::Rect expected_child_damage_rect(60, 60);
expected_child_damage_rect.Inset(-blur_outset);
gfx::Rect expected_root_damage_rect(child_damage_rect);
expected_root_damage_rect.Offset(200, 200);
EXPECT_EQ(expected_root_damage_rect, root_damage_rect);
EXPECT_EQ(expected_child_damage_rect, child_damage_rect);
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(30, 30));
EmulateDrawingOneFrame(root, device_scale_factor);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(expected_root_damage_rect, root_damage_rect);
EXPECT_EQ(expected_child_damage_rect, child_damage_rect);
}
TEST_F(DamageTrackerTest, VerifyDamageForAddingAndRemovingLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child1 = child_layers_[0];
ClearDamageForAllSurfaces(root);
LayerImpl* child2 = AddLayer<LayerImpl>();
child2->SetBounds(gfx::Size(6, 8));
child2->SetDrawsContent(true);
CopyProperties(root, child2);
child2->SetOffsetToTransformParent(gfx::Vector2dF(400.f, 380.f));
EmulateDrawingOneFrame(root);
ASSERT_EQ(3, GetRenderSurface(root)->num_contributors());
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(400, 380, 6, 8).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(root_damage_rect.IsEmpty());
{
OwnedLayerImplList layers =
host_impl()->active_tree()->DetachLayersKeepingRootLayerForTesting();
ASSERT_EQ(3u, layers.size());
ASSERT_EQ(child1, layers[1].get());
host_impl()->active_tree()->AddLayer(std::move(layers[2]));
}
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(100, 100, 30, 30).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForNewUnchangedLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
ClearDamageForAllSurfaces(root);
LayerImpl* child2 = AddLayer<LayerImpl>();
child2->SetBounds(gfx::Size(6, 8));
child2->SetDrawsContent(true);
CopyProperties(root, child2);
child2->SetOffsetToTransformParent(gfx::Vector2dF(400.f, 380.f));
host_impl()->active_tree()->ResetAllChangeTracking();
ASSERT_FALSE(child2->LayerPropertyChanged());
ASSERT_TRUE(child2->update_rect().IsEmpty());
EmulateDrawingOneFrame(root);
ASSERT_EQ(3, GetRenderSurface(root)->num_contributors());
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(400, 380, 6, 8).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForMultipleLayers) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child1 = child_layers_[0];
ClearDamageForAllSurfaces(root);
LayerImpl* child2 = AddLayer<LayerImpl>();
child2->SetBounds(gfx::Size(6, 8));
child2->SetDrawsContent(true);
CopyProperties(root, child2);
child2->SetOffsetToTransformParent(gfx::Vector2dF(400.f, 380.f));
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
child1->UnionUpdateRect(gfx::Rect(1, 2));
child2->UnionUpdateRect(gfx::Rect(3, 4));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(100, 100, 303, 284).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForNestedSurfaces) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
CreateEffectNode(child2_).render_surface_reason = RenderSurfaceReason::kTest;
CreateEffectNode(grand_child1_).render_surface_reason =
RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(grand_child1_->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(200, 200, 6, 8).ToString(), child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(300, 300, 6, 8).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child2_)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(grand_child1_)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
root->layer_tree_impl()->SetOpacityMutated(grand_child1_->element_id(), 0.7f);
root->layer_tree_impl()->SetOpacityMutated(child2_->element_id(), 0.7f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(200, 200, 6, 8).ToString(), child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(11, 11, 295, 297).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child2_)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(grand_child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForSurfaceChangeFromDescendantLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
grand_child1_->SetOffsetToTransformParent(gfx::Vector2dF(195.f, 205.f));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(190, 190, 11, 23).ToString(),
child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(290, 290, 16, 23).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForSurfaceChangeFromDescendantSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
child1_->SetDrawsContent(true);
EmulateDrawingOneFrame(root);
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
SetPostTranslation(child1_.get(), gfx::Vector2dF(105.f, 107.f));
child1_->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::UnionRects(gfx::Rect(100, 100, 206, 208),
gfx::Rect(105, 107, 206, 208))
.ToString(),
root_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(0, 0, 206, 208).ToString(), child_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
CreateEffectNode(child2_).render_surface_reason = RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
child2_->SetBounds(gfx::Size(120, 140));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child2_)
->damage_tracker()
->has_damage_from_contributing_content());
child2_->SetBounds(gfx::Size(220, 240));
root->layer_tree_impl()->SetOpacityMutated(child2_->element_id(), 0.5f);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child2_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForSurfaceChangeFromViewTransitionLayer) {
ClearLayersAndProperties();
LayerImpl* root = root_layer();
root->SetBounds(gfx::Size(500, 500));
root->layer_tree_impl()->SetDeviceViewportRect(gfx::Rect(root->bounds()));
root->SetDrawsContent(true);
SetupRootProperties(root);
LayerImpl* child1 = AddLayer<TestLayerImpl>();
LayerImpl* grand_child1 = AddLayer<TestLayerImpl>();
LayerImpl* child2 = AddLayer<TestViewTransitionContentLayerImpl>(
viz::ViewTransitionElementResourceId(3), false);
child1->SetBounds(gfx::Size(80, 80));
child1->SetDrawsContent(true);
CopyProperties(root, child1);
CreateTransformNode(child1).post_translation = gfx::Vector2dF(100.f, 100.f);
CreateEffectNode(child1).render_surface_reason = RenderSurfaceReason::kTest;
grand_child1->SetBounds(gfx::Size(10, 20));
grand_child1->SetDrawsContent(true);
CopyProperties(child1, grand_child1);
grand_child1->SetOffsetToTransformParent(gfx::Vector2dF(30.f, 30.f));
child2->SetBounds(gfx::Size(160, 160));
child2->SetDrawsContent(true);
CopyProperties(root, child2);
child2->SetOffsetToTransformParent(gfx::Vector2dF(100.f, 100.f));
SetElementIdsForTesting();
EmulateDrawingOneFrame(root);
GetRenderSurface(child1)
->OwningEffectNodeMutableForTest()
->view_transition_element_resource_id =
child2->ViewTransitionResourceId();
EmulateDrawingOneFrame(root);
gfx::Rect child1_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
grand_child1->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1)->damage_tracker()->GetDamageRectIfValid(
&child1_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(30, 30, 10, 20).ToString(),
child1_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(130, 130, 50, 70).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForSurfaceChangeFromAncestorLayer) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
gfx::Transform translation;
translation.Translate(-50.f, -50.f);
root->layer_tree_impl()->SetTransformMutated(child1_->element_id(),
translation);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(190, 190, 16, 18).ToString(),
child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(240, 240, 66, 68).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_FALSE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForAddingAndRemovingRenderSurfaces) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
SetRenderSurfaceReason(child1_.get(), RenderSurfaceReason::kNone);
EmulateDrawingOneFrame(root);
ASSERT_EQ(GetRenderSurface(child1_), GetRenderSurface(root));
ASSERT_EQ(4, GetRenderSurface(root)->num_contributors());
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(290, 290, 16, 18).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(root_damage_rect.IsEmpty());
ClearDamageForAllSurfaces(root);
SetRenderSurfaceReason(child1_.get(), RenderSurfaceReason::kTest);
EmulateDrawingOneFrame(root);
ASSERT_TRUE(GetRenderSurface(child1_));
EXPECT_EQ(3, GetRenderSurface(root)->num_contributors());
EXPECT_EQ(2, GetRenderSurface(child1_)->num_contributors());
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(190, 190, 16, 18).ToString(),
child_damage_rect.ToString());
EXPECT_EQ(gfx::Rect(290, 290, 16, 18).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyNoDamageWhenNothingChanged) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(child_damage_rect.IsEmpty());
EXPECT_TRUE(root_damage_rect.IsEmpty());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(child_damage_rect.IsEmpty());
EXPECT_TRUE(root_damage_rect.IsEmpty());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyNoDamageForUpdateRectThatDoesNotDrawContent) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
gfx::Rect child_damage_rect;
gfx::Rect root_damage_rect;
ClearDamageForAllSurfaces(root);
child1_->UnionUpdateRect(gfx::Rect(1, 2));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(child_damage_rect.IsEmpty());
EXPECT_TRUE(root_damage_rect.IsEmpty());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageForMask) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
CreateTransformNode(child).post_translation =
child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
CreateEffectNode(child);
auto* mask_layer = AddLayer<FakePictureLayerImpl>();
SetupMaskProperties(child, mask_layer);
Region empty_invalidation;
mask_layer->UpdateRasterSource(
FakeRasterSource::CreateFilled(child->bounds()), &empty_invalidation,
nullptr, nullptr);
LayerImpl* grand_child = AddLayer<LayerImpl>();
grand_child->SetBounds(gfx::Size(2, 2));
grand_child->SetDrawsContent(true);
CopyProperties(child, grand_child);
grand_child->SetOffsetToTransformParent(gfx::Vector2dF(2.f, 2.f));
EmulateDrawingOneFrame(root);
EXPECT_EQ(2, GetRenderSurface(root)->num_contributors());
EXPECT_TRUE(root->contributes_to_drawn_render_surface());
EXPECT_EQ(3, GetRenderSurface(child)->num_contributors());
EXPECT_TRUE(child->contributes_to_drawn_render_surface());
EXPECT_EQ(GetRenderSurface(child), GetRenderSurface(mask_layer));
EXPECT_TRUE(mask_layer->contributes_to_drawn_render_surface());
ClearDamageForAllSurfaces(root);
mask_layer->UnionUpdateRect(gfx::Rect(1, 2, 3, 4));
EmulateDrawingOneFrame(root);
gfx::Rect child_damage_rect;
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(1, 2, 3, 4), child_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(child_damage_rect.IsEmpty());
ClearDamageForAllSurfaces(root);
mask_layer->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(30, 30), child_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_TRUE(child_damage_rect.IsEmpty());
ClearDamageForAllSurfaces(root);
auto layers =
root->layer_tree_impl()->DetachLayersKeepingRootLayerForTesting();
ASSERT_EQ(layers[1].get(), child);
root->layer_tree_impl()->AddLayer(std::move(layers[1]));
ASSERT_EQ(layers[2].get(), mask_layer);
ASSERT_EQ(layers[3].get(), grand_child);
root->layer_tree_impl()->AddLayer(std::move(layers[3]));
CopyProperties(root, child);
CreateEffectNode(child).render_surface_reason = RenderSurfaceReason::kTest;
CopyProperties(child, grand_child);
EmulateDrawingOneFrame(root);
ASSERT_TRUE(GetRenderSurface(child));
EXPECT_TRUE(GetRenderSurface(child)->damage_tracker()->GetDamageRectIfValid(
&child_damage_rect));
EXPECT_EQ(gfx::Rect(30, 30), child_damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, DamageWhenAddedExternally) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(10, 11, 12, 13));
GetRenderSurface(root)->damage_tracker()->AddDamageNextUpdate(
gfx::Rect(15, 16, 32, 33));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::UnionRects(gfx::Rect(15, 16, 32, 33),
gfx::Rect(100 + 10, 100 + 11, 12, 13)).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
GetRenderSurface(root)->damage_tracker()->AddDamageNextUpdate(
gfx::Rect(30, 31, 14, 15));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(30, 31, 14, 15).ToString(), root_damage_rect.ToString());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageWithNoContributingLayers) {
LayerImpl* root = root_layer();
ClearDamageForAllSurfaces(root);
LayerImpl* empty_surface = AddLayer<LayerImpl>();
CopyProperties(root, empty_surface);
CreateEffectNode(empty_surface).render_surface_reason =
RenderSurfaceReason::kTest;
EmulateDrawingOneFrame(root);
DCHECK_EQ(GetRenderSurface(empty_surface), empty_surface->render_target());
const RenderSurfaceImpl* target_surface = GetRenderSurface(empty_surface);
gfx::Rect damage_rect;
EXPECT_TRUE(
target_surface->damage_tracker()->GetDamageRectIfValid(&damage_rect));
EXPECT_TRUE(damage_rect.IsEmpty());
EXPECT_FALSE(
target_surface->damage_tracker()->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageAccumulatesUntilReset) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
ClearDamageForAllSurfaces(root);
child->UnionUpdateRect(gfx::Rect(10.f, 11.f, 1.f, 2.f));
EmulateDrawingOneFrame(root);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(110, 111, 1, 2).ToString(), root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
child->UnionUpdateRect(gfx::Rect(20, 25, 1, 2));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(gfx::Rect(110, 111, 11, 16).ToString(),
root_damage_rect.ToString());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
GetRenderSurface(root)->damage_tracker()->DidDrawDamagedArea();
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(root_damage_rect.IsEmpty());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_TRUE(root_damage_rect.IsEmpty());
EXPECT_FALSE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, HugeDamageRect) {
const int kBigNumber = 900000000;
const int kRange = 5000;
for (int i = 0; i < kRange; ++i) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface();
LayerImpl* child = child_layers_[0];
SetCopyRequest(root);
gfx::Transform transform;
transform.Translate(-kBigNumber, -kBigNumber);
child->SetBounds(gfx::Size(kBigNumber + i, kBigNumber + i));
auto& transform_node = CreateTransformNode(child);
transform_node.local = transform;
transform_node.post_translation = child->offset_to_transform_parent();
child->SetOffsetToTransformParent(gfx::Vector2dF());
float device_scale_factor = 1.f;
EmulateDrawingOneFrame(root, device_scale_factor);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
gfx::Rect damage_we_care_about = gfx::Rect(i, i);
EXPECT_LE(damage_we_care_about.right(), root_damage_rect.right());
EXPECT_LE(damage_we_care_about.bottom(), root_damage_rect.bottom());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
}
TEST_F(DamageTrackerTest, DamageRectTooBig) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface(2);
LayerImpl* child1 = child_layers_[0];
LayerImpl* child2 = child_layers_[1];
SetCopyRequest(root);
child1->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 100), 0));
child1->SetBounds(gfx::Size(1, 1));
child2->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::max() - 100), 0));
child2->SetBounds(gfx::Size(1, 1));
EmulateDrawingOneFrame(root, 1.f);
gfx::Rect damage_rect;
EXPECT_FALSE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(root)->content_rect(),
GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, DamageRectTooBigWithFilter) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface(2);
LayerImpl* child1 = child_layers_[0];
LayerImpl* child2 = child_layers_[1];
SetCopyRequest(root);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(5.f));
root->SetDrawsContent(true);
SetBackdropFilter(root, filters);
child1->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 100), 0));
child1->SetBounds(gfx::Size(1, 1));
child2->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::max() - 100), 0));
child2->SetBounds(gfx::Size(1, 1));
float device_scale_factor = 1.f;
EmulateDrawingOneFrame(root, device_scale_factor);
gfx::Rect damage_rect;
EXPECT_FALSE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(root)->content_rect(),
GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, DamageRectTooBigInRenderSurface) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfacesDrawingFullyVisible();
SetCopyRequest(root);
grand_child1_->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 500), 0));
grand_child1_->SetBounds(gfx::Size(1, 1));
grand_child1_->SetDrawsContent(true);
grand_child2_->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::max() - 500), 0));
grand_child2_->SetBounds(gfx::Size(1, 1));
grand_child2_->SetDrawsContent(true);
UpdateDrawProperties(host_impl()->active_tree());
GetRenderSurface(child1_)->NoteAncestorPropertyChanged();
DamageTracker::UpdateDamageTracking(host_impl()->active_tree());
gfx::Rect damage_rect;
EXPECT_FALSE(
GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(child1_)->content_rect(),
GetRenderSurface(child1_)->GetDamageRect());
ASSERT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_TRUE(damage_rect.Contains(GetRenderSurface(root)->content_rect()));
EXPECT_TRUE(damage_rect.Contains(
gfx::ToEnclosingRect(GetRenderSurface(child1_)->DrawableContentRect())));
EXPECT_EQ(damage_rect, GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
root->layer_tree_impl()->ResetAllChangeTracking();
grand_child1_->AddDamageRect(gfx::Rect(grand_child1_->bounds()));
grand_child2_->AddDamageRect(gfx::Rect(grand_child1_->bounds()));
UpdateDrawProperties(host_impl()->active_tree());
DamageTracker::UpdateDamageTracking(host_impl()->active_tree());
EXPECT_FALSE(
GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(child1_)->content_rect(),
GetRenderSurface(child1_)->GetDamageRect());
ASSERT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_TRUE(damage_rect.Contains(GetRenderSurface(root)->content_rect()));
EXPECT_TRUE(damage_rect.Contains(
gfx::ToEnclosingRect(GetRenderSurface(child1_)->DrawableContentRect())));
EXPECT_EQ(damage_rect, GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, DamageRectTooBigInRenderSurfaceWithFilter) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
SetCopyRequest(root);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(5.f));
child1_->SetDrawsContent(true);
SetBackdropFilter(child1_.get(), filters);
grand_child1_->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 500), 0));
grand_child1_->SetBounds(gfx::Size(1, 1));
grand_child1_->SetDrawsContent(true);
grand_child2_->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::max() - 500), 0));
grand_child2_->SetBounds(gfx::Size(1, 1));
grand_child2_->SetDrawsContent(true);
UpdateDrawProperties(host_impl()->active_tree());
GetRenderSurface(child1_)->NoteAncestorPropertyChanged();
DamageTracker::UpdateDamageTracking(host_impl()->active_tree());
gfx::Rect damage_rect;
EXPECT_FALSE(
GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(child1_)->content_rect(),
GetRenderSurface(child1_)->GetDamageRect());
ASSERT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_TRUE(damage_rect.Contains(GetRenderSurface(root)->content_rect()));
EXPECT_TRUE(damage_rect.Contains(
gfx::ToEnclosingRect(GetRenderSurface(child1_)->DrawableContentRect())));
EXPECT_EQ(damage_rect, GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
root->layer_tree_impl()->ResetAllChangeTracking();
grand_child1_->AddDamageRect(gfx::Rect(grand_child1_->bounds()));
grand_child2_->AddDamageRect(gfx::Rect(grand_child1_->bounds()));
UpdateDrawProperties(host_impl()->active_tree());
DamageTracker::UpdateDamageTracking(host_impl()->active_tree());
EXPECT_FALSE(
GetRenderSurface(child1_)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(GetRenderSurface(child1_)->content_rect(),
GetRenderSurface(child1_)->GetDamageRect());
ASSERT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_TRUE(damage_rect.Contains(GetRenderSurface(root)->content_rect()));
EXPECT_TRUE(damage_rect.Contains(
gfx::ToEnclosingRect(GetRenderSurface(child1_)->DrawableContentRect())));
EXPECT_EQ(damage_rect, GetRenderSurface(root)->GetDamageRect());
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
EXPECT_TRUE(GetRenderSurface(child1_)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, CanUseCachedBackdropFilterResultTest) {
LayerImpl* root = CreateAndSetUpTestTreeWithFourSurfaces();
ClearDamageForAllSurfaces(root);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(2.f));
SetBackdropFilter(child1_.get(), filters);
child1_->NoteLayerPropertyChanged();
EXPECT_TRUE(GetRenderSurface(child1_)->intersects_damage_under());
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->intersects_damage_under());
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(child1_)->intersects_damage_under());
root->UnionUpdateRect(gfx::Rect(0, 0, 20, 20));
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(child1_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
root->UnionUpdateRect(gfx::Rect(260, 260, 20, 20));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(child1_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
child2_->SetOffsetToTransformParent(gfx::Vector2dF(180.f, 180.f));
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(grand_child1_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
SetBackdropFilter(child1_.get(), FilterOperations());
grand_child1_->NoteLayerPropertyChanged();
SetBackdropFilter(grand_child4_.get(), filters);
grand_child4_->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(grand_child4_)->intersects_damage_under());
EXPECT_TRUE(GetRenderSurface(grand_child1_)->intersects_damage_under());
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
grand_child1_->AddDamageRect(gfx::Rect(2, 2, 1.f, 1.f));
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
grand_child2_->AddDamageRect(gfx::Rect(0, 0, 1.f, 1.f));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
grand_child3_->AddDamageRect(gfx::Rect(0, 0, 1.f, 1.f));
EmulateDrawingOneFrame(root);
gfx::Rect damage_rect;
EXPECT_TRUE(GetRenderSurface(grand_child3_)
->damage_tracker()
->GetDamageRectIfValid(&damage_rect));
EXPECT_EQ(gfx::Rect(170, 170, 1.f, 1.f), damage_rect);
EXPECT_FALSE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
grand_child3_->SetBounds(gfx::Size(11.f, 11.f));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(grand_child3_)
->damage_tracker()
->GetDamageRectIfValid(&damage_rect));
EXPECT_EQ(gfx::Rect(170, 170, 11.f, 11.f), damage_rect);
EXPECT_TRUE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
OwnedLayerImplList layers =
host_impl()->active_tree()->DetachLayersKeepingRootLayerForTesting();
ASSERT_EQ(7u, layers.size());
ASSERT_EQ(child1_, layers[1].get());
ASSERT_EQ(grand_child1_, layers[2].get());
ASSERT_EQ(grand_child2_, layers[3].get());
ASSERT_EQ(grand_child3_, layers[4].get());
ASSERT_EQ(grand_child4_, layers[5].get());
ASSERT_EQ(child2_, layers[6].get());
host_impl()->active_tree()->AddLayer(std::move(layers[1]));
host_impl()->active_tree()->AddLayer(std::move(layers[3]));
host_impl()->active_tree()->AddLayer(std::move(layers[4]));
host_impl()->active_tree()->AddLayer(std::move(layers[5]));
host_impl()->active_tree()->AddLayer(std::move(layers[6]));
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
layers = host_impl()->active_tree()->DetachLayersKeepingRootLayerForTesting();
ASSERT_EQ(6u, layers.size());
ASSERT_EQ(child1_, layers[1].get());
ASSERT_EQ(grand_child2_, layers[2].get());
ASSERT_EQ(grand_child3_, layers[3].get());
ASSERT_EQ(grand_child4_, layers[4].get());
ASSERT_EQ(child2_, layers[5].get());
host_impl()->active_tree()->AddLayer(std::move(layers[1]));
host_impl()->active_tree()->AddLayer(std::move(layers[3]));
host_impl()->active_tree()->AddLayer(std::move(layers[4]));
host_impl()->active_tree()->AddLayer(std::move(layers[5]));
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
EmulateDrawingOneFrame(root);
EXPECT_FALSE(GetRenderSurface(grand_child4_)->intersects_damage_under());
ClearDamageForAllSurfaces(root);
SetRenderSurfaceReason(grand_child3_.get(), RenderSurfaceReason::kNone);
grand_child3_->SetDrawsContent(false);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(grand_child4_)->intersects_damage_under());
}
TEST_F(DamageTrackerTest, DamageRectOnlyVisibleContentsMoveToOutside) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface(2);
ClearDamageForAllSurfaces(root);
LayerImpl* child1 = child_layers_[0];
LayerImpl* child2 = child_layers_[1];
gfx::Rect origin_damage = child1->visible_drawable_content_rect();
origin_damage.Union(child2->visible_drawable_content_rect());
child1->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 100), 0));
child1->SetBounds(gfx::Size(1, 1));
child2->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::max() - 100), 0));
child2->SetBounds(gfx::Size(1, 1));
EmulateDrawingOneFrame(root, 1.f);
gfx::Rect damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(origin_damage, damage_rect);
EXPECT_TRUE(GetRenderSurface(root)->content_rect().Contains(damage_rect));
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, DamageRectOnlyVisibleContentsLargeTwoContents) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface(2);
ClearDamageForAllSurfaces(root);
LayerImpl* child1 = child_layers_[0];
LayerImpl* child2 = child_layers_[1];
gfx::Rect expected_damage = child1->visible_drawable_content_rect();
expected_damage.Union(child2->visible_drawable_content_rect());
expected_damage.set_x(0);
expected_damage.set_width(GetRenderSurface(root)->content_rect().width());
child1->SetOffsetToTransformParent(gfx::Vector2dF(
static_cast<float>(std::numeric_limits<int>::min() + 100), 100));
child1->SetBounds(
gfx::Size(std::numeric_limits<int>::max(), child1->bounds().height()));
child2->SetOffsetToTransformParent(gfx::Vector2dF(100, 100));
child2->SetBounds(
gfx::Size(std::numeric_limits<int>::max(), child2->bounds().height()));
EmulateDrawingOneFrame(root, 1.f);
gfx::Rect damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(expected_damage, damage_rect);
EXPECT_TRUE(GetRenderSurface(root)->content_rect().Contains(damage_rect));
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest,
DamageRectOnlyVisibleContentsHugeContentPartiallyVisible) {
LayerImpl* root = CreateAndSetUpTestTreeWithOneSurface(1);
int content_width = GetRenderSurface(root)->content_rect().width();
ClearDamageForAllSurfaces(root);
LayerImpl* child1 = child_layers_[0];
int y = child1->offset_to_transform_parent().y();
int offset = 100;
int expected_width = offset + child1->bounds().width();
child1->SetOffsetToTransformParent(
gfx::Vector2dF(std::numeric_limits<int>::min() + offset, y));
child1->SetBounds(
gfx::Size(std::numeric_limits<int>::max(), child1->bounds().height()));
EmulateDrawingOneFrame(root);
gfx::Rect expected_damage_rect1(0, y, expected_width,
child1->bounds().height());
gfx::Rect damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(expected_damage_rect1, damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
ClearDamageForAllSurfaces(root);
child1->SetOffsetToTransformParent(gfx::Vector2dF(content_width - offset, y));
child1->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
gfx::Rect expected_damage_rect2(0, y, content_width,
child1->bounds().height());
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&damage_rect));
EXPECT_EQ(expected_damage_rect2, damage_rect);
EXPECT_TRUE(GetRenderSurface(root)
->damage_tracker()
->has_damage_from_contributing_content());
}
TEST_F(DamageTrackerTest, VerifyDamageExpansionWithBackdropBlurFilters) {
LayerImpl* root = CreateAndSetUpTestTreeWithTwoSurfaces();
child1_->SetDrawsContent(true);
FilterOperations filters;
filters.Append(FilterOperation::CreateBlurFilter(2.f));
ClearDamageForAllSurfaces(root);
SetBackdropFilter(child1_.get(), filters);
child1_->NoteLayerPropertyChanged();
EmulateDrawingOneFrame(root);
ClearDamageForAllSurfaces(root);
root->UnionUpdateRect(gfx::Rect(297, 297, 2, 2));
EmulateDrawingOneFrame(root);
gfx::Rect expected_damage_rect = gfx::Rect(100, 100, 206, 208);
gfx::Rect root_damage_rect;
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(expected_damage_rect, root_damage_rect);
ClearDamageForAllSurfaces(root);
gfx::Rect damage_rect(97, 97, 2, 2);
root->UnionUpdateRect(damage_rect);
EmulateDrawingOneFrame(root);
EXPECT_TRUE(GetRenderSurface(root)->damage_tracker()->GetDamageRectIfValid(
&root_damage_rect));
EXPECT_EQ(damage_rect, root_damage_rect);
}
}
}