|  | // Copyright 2013 The Flutter 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 "flutter/flow/layers/layer_tree.h" | 
|  |  | 
|  | #include "flutter/flow/compositor_context.h" | 
|  | #include "flutter/flow/layers/container_layer.h" | 
|  | #include "flutter/flow/testing/mock_layer.h" | 
|  | #include "flutter/fml/macros.h" | 
|  | #include "flutter/testing/canvas_test.h" | 
|  | #include "flutter/testing/mock_canvas.h" | 
|  | #include "gtest/gtest.h" | 
|  |  | 
|  | namespace flutter { | 
|  | namespace testing { | 
|  |  | 
|  | class LayerTreeTest : public CanvasTest { | 
|  | public: | 
|  | LayerTreeTest() | 
|  | : layer_tree_(SkISize::Make(64, 64), 100.0f, 1.0f), | 
|  | compositor_context_(fml::kDefaultFrameBudget), | 
|  | root_transform_(SkMatrix::Translate(1.0f, 1.0f)), | 
|  | scoped_frame_(compositor_context_.AcquireFrame(nullptr, | 
|  | &mock_canvas(), | 
|  | nullptr, | 
|  | root_transform_, | 
|  | false, | 
|  | true, | 
|  | nullptr)) {} | 
|  |  | 
|  | LayerTree& layer_tree() { return layer_tree_; } | 
|  | CompositorContext::ScopedFrame& frame() { return *scoped_frame_.get(); } | 
|  | const SkMatrix& root_transform() { return root_transform_; } | 
|  |  | 
|  | private: | 
|  | LayerTree layer_tree_; | 
|  | CompositorContext compositor_context_; | 
|  | SkMatrix root_transform_; | 
|  | std::unique_ptr<CompositorContext::ScopedFrame> scoped_frame_; | 
|  | }; | 
|  |  | 
|  | TEST_F(LayerTreeTest, PaintingEmptyLayerDies) { | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  |  | 
|  | layer_tree().set_root_layer(layer); | 
|  | layer_tree().Preroll(frame()); | 
|  | EXPECT_EQ(layer->paint_bounds(), SkRect::MakeEmpty()); | 
|  | EXPECT_FALSE(layer->needs_painting()); | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | } | 
|  |  | 
|  | TEST_F(LayerTreeTest, PaintBeforePreollDies) { | 
|  | const SkRect child_bounds = SkRect::MakeLTRB(5.0f, 6.0f, 20.5f, 21.5f); | 
|  | SkPath child_path; | 
|  | child_path.addRect(child_bounds); | 
|  | auto mock_layer = std::make_shared<MockLayer>(child_path); | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  | layer->Add(mock_layer); | 
|  |  | 
|  | layer_tree().set_root_layer(layer); | 
|  | EXPECT_EQ(mock_layer->paint_bounds(), kEmptyRect); | 
|  | EXPECT_EQ(layer->paint_bounds(), kEmptyRect); | 
|  | EXPECT_FALSE(mock_layer->needs_painting()); | 
|  | EXPECT_FALSE(layer->needs_painting()); | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | EXPECT_EQ(mock_canvas().draw_calls(), std::vector<MockCanvas::DrawCall>()); | 
|  | } | 
|  |  | 
|  | TEST_F(LayerTreeTest, Simple) { | 
|  | const SkRect child_bounds = SkRect::MakeLTRB(5.0f, 6.0f, 20.5f, 21.5f); | 
|  | const SkPath child_path = SkPath().addRect(child_bounds); | 
|  | const SkPaint child_paint = SkPaint(SkColors::kCyan); | 
|  | auto mock_layer = std::make_shared<MockLayer>(child_path, child_paint); | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  | layer->Add(mock_layer); | 
|  |  | 
|  | layer_tree().set_root_layer(layer); | 
|  | layer_tree().Preroll(frame()); | 
|  | EXPECT_EQ(mock_layer->paint_bounds(), child_bounds); | 
|  | EXPECT_EQ(layer->paint_bounds(), mock_layer->paint_bounds()); | 
|  | EXPECT_TRUE(mock_layer->needs_painting()); | 
|  | EXPECT_TRUE(layer->needs_painting()); | 
|  | EXPECT_EQ(mock_layer->parent_matrix(), root_transform()); | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | EXPECT_EQ(mock_canvas().draw_calls(), | 
|  | std::vector({MockCanvas::DrawCall{ | 
|  | 0, MockCanvas::DrawPathData{child_path, child_paint}}})); | 
|  | } | 
|  |  | 
|  | TEST_F(LayerTreeTest, Multiple) { | 
|  | const SkPath child_path1 = SkPath().addRect(5.0f, 6.0f, 20.5f, 21.5f); | 
|  | const SkPath child_path2 = SkPath().addRect(8.0f, 2.0f, 16.5f, 14.5f); | 
|  | const SkPaint child_paint1(SkColors::kGray); | 
|  | const SkPaint child_paint2(SkColors::kGreen); | 
|  | auto mock_layer1 = std::make_shared<MockLayer>( | 
|  | child_path1, child_paint1, true /* fake_has_platform_view */); | 
|  | auto mock_layer2 = std::make_shared<MockLayer>(child_path2, child_paint2); | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  | layer->Add(mock_layer1); | 
|  | layer->Add(mock_layer2); | 
|  |  | 
|  | SkRect expected_total_bounds = child_path1.getBounds(); | 
|  | expected_total_bounds.join(child_path2.getBounds()); | 
|  | layer_tree().set_root_layer(layer); | 
|  | layer_tree().Preroll(frame()); | 
|  | EXPECT_EQ(mock_layer1->paint_bounds(), child_path1.getBounds()); | 
|  | EXPECT_EQ(mock_layer2->paint_bounds(), child_path2.getBounds()); | 
|  | EXPECT_EQ(layer->paint_bounds(), expected_total_bounds); | 
|  | EXPECT_TRUE(mock_layer1->needs_painting()); | 
|  | EXPECT_TRUE(mock_layer2->needs_painting()); | 
|  | EXPECT_TRUE(layer->needs_painting()); | 
|  | EXPECT_FALSE(mock_layer1->needs_system_composite()); | 
|  | EXPECT_FALSE(mock_layer2->needs_system_composite()); | 
|  | EXPECT_FALSE(layer->needs_system_composite()); | 
|  | EXPECT_EQ(mock_layer1->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer2->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer1->parent_cull_rect(), kGiantRect); | 
|  | EXPECT_EQ(mock_layer2->parent_cull_rect(), | 
|  | kGiantRect);  // Siblings are independent | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | EXPECT_EQ( | 
|  | mock_canvas().draw_calls(), | 
|  | std::vector({MockCanvas::DrawCall{ | 
|  | 0, MockCanvas::DrawPathData{child_path1, child_paint1}}, | 
|  | MockCanvas::DrawCall{0, MockCanvas::DrawPathData{ | 
|  | child_path2, child_paint2}}})); | 
|  | } | 
|  |  | 
|  | TEST_F(LayerTreeTest, MultipleWithEmpty) { | 
|  | const SkPath child_path1 = SkPath().addRect(5.0f, 6.0f, 20.5f, 21.5f); | 
|  | const SkPaint child_paint1(SkColors::kGray); | 
|  | const SkPaint child_paint2(SkColors::kGreen); | 
|  | auto mock_layer1 = std::make_shared<MockLayer>(child_path1, child_paint1); | 
|  | auto mock_layer2 = std::make_shared<MockLayer>(SkPath(), child_paint2); | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  | layer->Add(mock_layer1); | 
|  | layer->Add(mock_layer2); | 
|  |  | 
|  | layer_tree().set_root_layer(layer); | 
|  | layer_tree().Preroll(frame()); | 
|  | EXPECT_EQ(mock_layer1->paint_bounds(), child_path1.getBounds()); | 
|  | EXPECT_EQ(mock_layer2->paint_bounds(), SkPath().getBounds()); | 
|  | EXPECT_EQ(layer->paint_bounds(), child_path1.getBounds()); | 
|  | EXPECT_TRUE(mock_layer1->needs_painting()); | 
|  | EXPECT_FALSE(mock_layer2->needs_painting()); | 
|  | EXPECT_TRUE(layer->needs_painting()); | 
|  | EXPECT_FALSE(mock_layer1->needs_system_composite()); | 
|  | EXPECT_FALSE(mock_layer2->needs_system_composite()); | 
|  | EXPECT_FALSE(layer->needs_system_composite()); | 
|  | EXPECT_EQ(mock_layer1->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer2->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer1->parent_cull_rect(), kGiantRect); | 
|  | EXPECT_EQ(mock_layer2->parent_cull_rect(), kGiantRect); | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | EXPECT_EQ(mock_canvas().draw_calls(), | 
|  | std::vector({MockCanvas::DrawCall{ | 
|  | 0, MockCanvas::DrawPathData{child_path1, child_paint1}}})); | 
|  | } | 
|  |  | 
|  | TEST_F(LayerTreeTest, NeedsSystemComposite) { | 
|  | const SkPath child_path1 = SkPath().addRect(5.0f, 6.0f, 20.5f, 21.5f); | 
|  | const SkPath child_path2 = SkPath().addRect(8.0f, 2.0f, 16.5f, 14.5f); | 
|  | const SkPaint child_paint1(SkColors::kGray); | 
|  | const SkPaint child_paint2(SkColors::kGreen); | 
|  | auto mock_layer1 = std::make_shared<MockLayer>( | 
|  | child_path1, child_paint1, false /* fake_has_platform_view */, | 
|  | true /* fake_needs_system_composite */); | 
|  | auto mock_layer2 = std::make_shared<MockLayer>(child_path2, child_paint2); | 
|  | auto layer = std::make_shared<ContainerLayer>(); | 
|  | layer->Add(mock_layer1); | 
|  | layer->Add(mock_layer2); | 
|  |  | 
|  | SkRect expected_total_bounds = child_path1.getBounds(); | 
|  | expected_total_bounds.join(child_path2.getBounds()); | 
|  | layer_tree().set_root_layer(layer); | 
|  | layer_tree().Preroll(frame()); | 
|  | EXPECT_EQ(mock_layer1->paint_bounds(), child_path1.getBounds()); | 
|  | EXPECT_EQ(mock_layer2->paint_bounds(), child_path2.getBounds()); | 
|  | EXPECT_EQ(layer->paint_bounds(), expected_total_bounds); | 
|  | EXPECT_TRUE(mock_layer1->needs_painting()); | 
|  | EXPECT_TRUE(mock_layer2->needs_painting()); | 
|  | EXPECT_TRUE(layer->needs_painting()); | 
|  | EXPECT_TRUE(mock_layer1->needs_system_composite()); | 
|  | EXPECT_FALSE(mock_layer2->needs_system_composite()); | 
|  | EXPECT_TRUE(layer->needs_system_composite()); | 
|  | EXPECT_EQ(mock_layer1->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer2->parent_matrix(), root_transform()); | 
|  | EXPECT_EQ(mock_layer1->parent_cull_rect(), kGiantRect); | 
|  | EXPECT_EQ(mock_layer2->parent_cull_rect(), kGiantRect); | 
|  |  | 
|  | layer_tree().Paint(frame()); | 
|  | EXPECT_EQ( | 
|  | mock_canvas().draw_calls(), | 
|  | std::vector({MockCanvas::DrawCall{ | 
|  | 0, MockCanvas::DrawPathData{child_path1, child_paint1}}, | 
|  | MockCanvas::DrawCall{0, MockCanvas::DrawPathData{ | 
|  | child_path2, child_paint2}}})); | 
|  | } | 
|  |  | 
|  | }  // namespace testing | 
|  | }  // namespace flutter |