| // Copyright 2015 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 "flutter/flow/layers/layer_tree.h" |
| |
| #include "flutter/flow/layers/layer.h" |
| #include "flutter/glue/trace_event.h" |
| #include "third_party/skia/include/core/SkPictureRecorder.h" |
| |
| namespace flow { |
| |
| LayerTree::LayerTree() |
| : frame_size_{}, |
| rasterizer_tracing_threshold_(0), |
| checkerboard_raster_cache_images_(false), |
| checkerboard_offscreen_layers_(false) {} |
| |
| LayerTree::~LayerTree() = default; |
| |
| void LayerTree::Preroll(CompositorContext::ScopedFrame& frame, |
| bool ignore_raster_cache) { |
| TRACE_EVENT0("flutter", "LayerTree::Preroll"); |
| SkColorSpace* color_space = |
| frame.canvas() ? frame.canvas()->imageInfo().colorSpace() : nullptr; |
| frame.context().raster_cache().SetCheckboardCacheImages( |
| checkerboard_raster_cache_images_); |
| Layer::PrerollContext context = { |
| ignore_raster_cache ? nullptr : &frame.context().raster_cache(), |
| frame.gr_context(), |
| color_space, |
| SkRect::MakeEmpty(), |
| }; |
| |
| root_layer_->Preroll(&context, SkMatrix::I()); |
| } |
| |
| #if defined(OS_FUCHSIA) |
| void LayerTree::UpdateScene(SceneUpdateContext& context, |
| scenic_lib::ContainerNode& container) { |
| TRACE_EVENT0("flutter", "LayerTree::UpdateScene"); |
| const auto& metrics = context.metrics(); |
| SceneUpdateContext::Transform transform(context, // context |
| 1.0f / metrics->scale_x, // X |
| 1.0f / metrics->scale_y, // Y |
| 1.0f / metrics->scale_z // Z |
| ); |
| SceneUpdateContext::Frame frame( |
| context, |
| SkRRect::MakeRect( |
| SkRect::MakeWH(frame_size_.width(), frame_size_.height())), |
| SK_ColorTRANSPARENT, 0.f); |
| if (root_layer_->needs_system_composite()) { |
| root_layer_->UpdateScene(context); |
| } |
| if (root_layer_->needs_painting()) { |
| frame.AddPaintedLayer(root_layer_.get()); |
| } |
| container.AddChild(transform.entity_node()); |
| } |
| #endif |
| |
| void LayerTree::Paint(CompositorContext::ScopedFrame& frame) const { |
| TRACE_EVENT0("flutter", "LayerTree::Paint"); |
| Layer::PaintContext context = { |
| *frame.canvas(), // |
| frame.context().frame_time(), // |
| frame.context().engine_time(), // |
| frame.context().texture_registry(), // |
| checkerboard_offscreen_layers_ // |
| }; |
| |
| if (root_layer_->needs_painting()) |
| root_layer_->Paint(context); |
| } |
| |
| sk_sp<SkPicture> LayerTree::Flatten(const SkRect& bounds) { |
| TRACE_EVENT0("flutter", "LayerTree::Flatten"); |
| |
| SkPictureRecorder recorder; |
| auto canvas = recorder.beginRecording(bounds); |
| |
| if (!canvas) { |
| return nullptr; |
| } |
| |
| Layer::PrerollContext preroll_context{ |
| nullptr, // raster_cache (don't consult the cache) |
| nullptr, // gr_context (used for the raster cache) |
| nullptr, // SkColorSpace* dst_color_space |
| SkRect::MakeEmpty(), // SkRect child_paint_bounds |
| }; |
| |
| const Stopwatch unused_stopwatch; |
| TextureRegistry unused_texture_registry; |
| |
| Layer::PaintContext paint_context = { |
| *canvas, // canvas |
| unused_stopwatch, // frame time (dont care) |
| unused_stopwatch, // engine time (dont care) |
| unused_texture_registry, // texture registry (not supported) |
| false // checkerboard offscreen layers |
| }; |
| |
| // Even if we don't have a root layer, we still need to create an empty |
| // picture. |
| if (root_layer_) { |
| root_layer_->Preroll(&preroll_context, SkMatrix::I()); |
| // The needs painting flag may be set after the preroll. So check it after. |
| if (root_layer_->needs_painting()) { |
| root_layer_->Paint(paint_context); |
| } |
| } |
| |
| return recorder.finishRecordingAsPicture(); |
| } |
| |
| } // namespace flow |