Flutter Engine
 
Loading...
Searching...
No Matches
external_view_embedder_2.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
7#include "flow/view_slicer.h"
10#include "fml/make_copyable.h"
12
13namespace flutter {
14
16 const AndroidContext& android_context,
17 std::shared_ptr<PlatformViewAndroidJNI> jni_facade,
18 std::shared_ptr<AndroidSurfaceFactory> surface_factory,
19 const TaskRunners& task_runners)
21 android_context_(android_context),
22 jni_facade_(std::move(jni_facade)),
23 surface_factory_(std::move(surface_factory)),
24 surface_pool_(
25 std::make_unique<SurfacePool>(/*use_new_surface_methods=*/true)),
26 task_runners_(task_runners) {}
27
28// |ExternalViewEmbedder|
30 int64_t view_id,
31 std::unique_ptr<EmbeddedViewParams> params) {
32 TRACE_EVENT0("flutter",
33 "AndroidExternalViewEmbedder2::PrerollCompositeEmbeddedView");
34
35 DlRect view_bounds = DlRect::MakeSize(frame_size_);
36 std::unique_ptr<EmbedderViewSlice> view;
37 view = std::make_unique<DisplayListEmbedderViewSlice>(view_bounds);
38 slices_.insert_or_assign(view_id, std::move(view));
39
40 composition_order_.push_back(view_id);
41 // Update params only if they changed.
42 if (view_params_.count(view_id) == 1 &&
43 view_params_.at(view_id) == *params.get()) {
44 return;
45 }
46 view_params_.insert_or_assign(view_id, EmbeddedViewParams(*params.get()));
47}
48
49// |ExternalViewEmbedder|
51 if (slices_.count(view_id) == 1) {
52 return slices_.at(view_id)->canvas();
53 }
54 return nullptr;
55}
56
58 int64_t view_id,
59 const std::unordered_map<int64_t, EmbeddedViewParams>& view_params) {
60 const EmbeddedViewParams& params = view_params.at(view_id);
61 // https://github.com/flutter/flutter/issues/59821
63}
64
65// |ExternalViewEmbedder|
67 int64_t flutter_view_id,
68 GrDirectContext* context,
69 const std::shared_ptr<impeller::AiksContext>& aiks_context,
70 std::unique_ptr<SurfaceFrame> frame) {
71 TRACE_EVENT0("flutter", "AndroidExternalViewEmbedder2::SubmitFlutterView");
72
73 if (!FrameHasPlatformLayers()) {
74 frame->Submit();
76 [this, jni_facade = jni_facade_,
77 views_visible_last_frame = views_visible_last_frame_]() {
78 // This pointer is guaranteed to not be dangling as long as
79 // DestroySurfaces is called before the embedder is deleted. See
80 // https://github.com/flutter/flutter/pull/176742#discussion_r2415229396.
81 this->HideOverlayLayerIfNeeded();
82 for (int64_t view_id : views_visible_last_frame) {
83 jni_facade->hidePlatformView2(view_id);
84 }
85
86 jni_facade->applyTransaction();
87 }));
88 views_visible_last_frame_.clear();
89 return;
90 }
91
92 std::unordered_map<int64_t, DlRect> view_rects;
93 for (auto platform_id : composition_order_) {
94 view_rects[platform_id] = GetViewRect(platform_id, view_params_);
95 }
96
97 std::unordered_map<int64_t, DlRect> overlay_layers =
98 SliceViews(frame->Canvas(), //
99 composition_order_, //
100 slices_, //
101 view_rects //
102 );
103
104 // If there is no overlay Surface, initialize one on the platform thread. This
105 // will only be done once per application launch, as the singular overlay
106 // surface is never released.
107 if (!surface_pool_->HasLayers()) {
108 std::shared_ptr<fml::CountDownLatch> latch =
109 std::make_shared<fml::CountDownLatch>(1u);
110 task_runners_.GetPlatformTaskRunner()->PostTask(
111 fml::MakeCopyable([&, latch]() {
112 surface_pool_->GetLayer(context, android_context_, jni_facade_,
113 surface_factory_);
114 latch->CountDown();
115 }));
116 latch->Wait();
117 }
118 surface_pool_->ResetLayers();
119
120 // Create Overlay frame. If overlay surface creation failed,
121 // all this work must be skipped.
122 std::unique_ptr<SurfaceFrame> overlay_frame;
123 if (surface_pool_->HasLayers()) {
124 for (size_t i = 0; i < composition_order_.size(); i++) {
125 int64_t view_id = composition_order_[i];
126 std::unordered_map<int64_t, DlRect>::const_iterator overlay =
127 overlay_layers.find(view_id);
128
129 if (overlay == overlay_layers.end()) {
130 continue;
131 }
132 if (overlay_frame == nullptr) {
133 std::shared_ptr<OverlayLayer> layer = surface_pool_->GetLayer(
134 context, android_context_, jni_facade_, surface_factory_);
135 overlay_frame = layer->surface->AcquireFrame(frame_size_);
136 overlay_frame->Canvas()->Clear(flutter::DlColor::kTransparent());
137 }
138
139 DlCanvas* overlay_canvas = overlay_frame->Canvas();
140 int restore_count = overlay_canvas->GetSaveCount();
141 overlay_canvas->Save();
142 overlay_canvas->ClipRect(overlay->second);
143
144 // For all following platform views that would cover this overlay,
145 // emulate the effect by adding a difference clip. This makes the
146 // overlays appear as if they are under the platform view, when in
147 // reality there is only a single layer.
148 for (size_t j = i + 1; j < composition_order_.size(); j++) {
149 DlRect view_rect = GetViewRect(composition_order_[j], view_params_);
150 overlay_canvas->ClipRect(view_rect, DlClipOp::kDifference);
151 }
152
153 slices_[view_id]->render_into(overlay_canvas);
154 overlay_canvas->RestoreToCount(restore_count);
155 }
156 }
157 bool overlay_layer_has_content_this_frame_;
158 if (overlay_frame != nullptr) {
159 overlay_frame->set_submit_info({.frame_boundary = false});
160 overlay_frame->Submit();
161 overlay_layer_has_content_this_frame_ = true;
162 } else {
163 overlay_layer_has_content_this_frame_ = false;
164 }
165
166 frame->Submit();
168 [&, composition_order = composition_order_, view_params = view_params_,
169 jni_facade = jni_facade_, device_pixel_ratio = device_pixel_ratio_,
170 slices = std::move(slices_),
171 views_visible_last_frame = views_visible_last_frame_,
172 overlay_layer_has_content_this_frame_]() mutable -> void {
173 jni_facade->swapTransaction();
174
175 if (overlay_layer_has_content_this_frame_) {
176 ShowOverlayLayerIfNeeded();
177 } else {
178 HideOverlayLayerIfNeeded();
179 }
180
181 for (int64_t view_id : composition_order) {
182 DlRect view_rect = GetViewRect(view_id, view_params);
183 const EmbeddedViewParams& params = view_params.at(view_id);
184 jni_facade->onDisplayPlatformView2(
185 view_id, //
186 view_rect.GetX(), //
187 view_rect.GetY(), //
188 view_rect.GetWidth(), //
189 view_rect.GetHeight(), //
190 params.sizePoints().width * device_pixel_ratio,
191 params.sizePoints().height * device_pixel_ratio,
192 params.mutatorsStack() //
193 );
194 // Remove from views visible last frame, so we can hide the rest.
195 views_visible_last_frame.erase(view_id);
196 }
197 // Hide views that were visible last frame, but not in this frame.
198 for (int64_t view_id : views_visible_last_frame) {
199 jni_facade->hidePlatformView2(view_id);
200 }
201
202 jni_facade_->onEndFrame2();
203 }));
204
205 views_visible_last_frame_.clear();
206 views_visible_last_frame_.insert(composition_order_.begin(),
207 composition_order_.end());
208}
209
210// |ExternalViewEmbedder|
215
216bool AndroidExternalViewEmbedder2::FrameHasPlatformLayers() {
217 return !composition_order_.empty();
218}
219
220// |ExternalViewEmbedder|
222 // On Android, the root surface is created from the on-screen render target.
223 return nullptr;
224}
225
226void AndroidExternalViewEmbedder2::Reset() {
227 composition_order_.clear();
228 slices_.clear();
229}
230
231// |ExternalViewEmbedder|
233 GrDirectContext* context,
234 const fml::RefPtr<fml::RasterThreadMerger>& raster_thread_merger) {}
235
236// |ExternalViewEmbedder|
238 DlISize frame_size,
239 double device_pixel_ratio) {
240 Reset();
241
242 // The surface size changed. Therefore, destroy existing surfaces as
243 // the existing surfaces in the pool can't be recycled.
244 if (frame_size_ != frame_size) {
245 DestroySurfaces();
246 }
247 surface_pool_->SetFrameSize(frame_size);
248
249 frame_size_ = frame_size;
250 device_pixel_ratio_ = device_pixel_ratio;
251}
252
253// |ExternalViewEmbedder|
257
258// |ExternalViewEmbedder|
260 bool should_resubmit_frame,
261 const fml::RefPtr<fml::RasterThreadMerger>& raster_thread_merger) {}
262
263// |ExternalViewEmbedder|
267
268// |ExternalViewEmbedder|
270 DestroySurfaces();
271}
272
273// |ExternalViewEmbedder|
274void AndroidExternalViewEmbedder2::DestroySurfaces() {
275 if (!surface_pool_->HasLayers()) {
276 return;
277 }
280 [&]() {
281 surface_pool_->DestroyLayers(jni_facade_);
282 latch.Signal();
283 });
284 latch.Wait();
285 overlay_layer_is_shown_.store(false);
286}
287
288void AndroidExternalViewEmbedder2::ShowOverlayLayerIfNeeded() {
289 if (!overlay_layer_is_shown_.load()) {
290 jni_facade_->showOverlaySurface2();
291 overlay_layer_is_shown_.store(true);
292 }
293}
294
295void AndroidExternalViewEmbedder2::HideOverlayLayerIfNeeded() {
296 if (overlay_layer_is_shown_.load()) {
297 jni_facade_->hideOverlaySurface2();
298 overlay_layer_is_shown_.store(false);
299 }
300}
301
302} // namespace flutter
Holds state that is shared across Android surfaces.
DlCanvas * CompositeEmbeddedView(int64_t view_id) override
void SubmitFlutterView(int64_t flutter_view_id, GrDirectContext *context, const std::shared_ptr< impeller::AiksContext > &aiks_context, std::unique_ptr< SurfaceFrame > frame) override
PostPrerollResult PostPrerollAction(const fml::RefPtr< fml::RasterThreadMerger > &raster_thread_merger) override
void EndFrame(bool should_resubmit_frame, const fml::RefPtr< fml::RasterThreadMerger > &raster_thread_merger) override
AndroidExternalViewEmbedder2(const AndroidContext &android_context, std::shared_ptr< PlatformViewAndroidJNI > jni_facade, std::shared_ptr< AndroidSurfaceFactory > surface_factory, const TaskRunners &task_runners)
static DlRect GetViewRect(int64_t view_id, const std::unordered_map< int64_t, EmbeddedViewParams > &view_params)
void PrerollCompositeEmbeddedView(int64_t view_id, std::unique_ptr< flutter::EmbeddedViewParams > params) override
void BeginFrame(GrDirectContext *context, const fml::RefPtr< fml::RasterThreadMerger > &raster_thread_merger) override
void PrepareFlutterView(DlISize frame_size, double device_pixel_ratio) override
Developer-facing API for rendering anything within the engine.
Definition dl_canvas.h:32
virtual void ClipRect(const DlRect &rect, DlClipOp clip_op=DlClipOp::kIntersect, bool is_aa=false)=0
virtual int GetSaveCount() const =0
virtual void RestoreToCount(int restore_count)=0
virtual void Save()=0
const DlRect & finalBoundingRect() const
fml::RefPtr< fml::TaskRunner > GetPlatformTaskRunner() const
static void RunNowOrPostTask(const fml::RefPtr< fml::TaskRunner > &runner, const fml::closure &task)
virtual void PostTask(const fml::closure &task) override
const EmbeddedViewParams * params
TaskRunners task_runners_
FlView * view
G_BEGIN_DECLS FlutterViewId view_id
std::unordered_map< int64_t, DlRect > SliceViews(DlCanvas *background_canvas, const std::vector< int64_t > &composition_order, const std::unordered_map< int64_t, std::unique_ptr< EmbedderViewSlice > > &slices, const std::unordered_map< int64_t, DlRect > &view_rects)
Compute the required overlay layers and clip the view slices according to the size and position of th...
internal::CopyableLambda< T > MakeCopyable(T lambda)
Definition ref_ptr.h:261
static constexpr DlColor kTransparent()
Definition dl_color.h:68
constexpr Type GetY() const
Returns the Y coordinate of the upper left corner, equivalent to |GetOrigin().y|.
Definition rect.h:337
constexpr Type GetHeight() const
Returns the height of the rectangle, equivalent to |GetSize().height|.
Definition rect.h:347
constexpr Type GetX() const
Returns the X coordinate of the upper left corner, equivalent to |GetOrigin().x|.
Definition rect.h:333
static constexpr TRect MakeSize(const TSize< U > &size)
Definition rect.h:150
constexpr Type GetWidth() const
Returns the width of the rectangle, equivalent to |GetSize().width|.
Definition rect.h:341
#define TRACE_EVENT0(category_group, name)