Flutter iOS Embedder
FlutterPlatformViewsController.mm
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 
6 
7 #include "flutter/display_list/effects/image_filters/dl_blur_image_filter.h"
8 #include "flutter/display_list/utils/dl_matrix_clip_tracker.h"
9 #include "flutter/flow/surface_frame.h"
10 #include "flutter/flow/view_slicer.h"
11 #include "flutter/fml/make_copyable.h"
12 #include "flutter/fml/synchronization/count_down_latch.h"
18 
19 using flutter::DlMatrix;
20 using flutter::DlRect;
21 using flutter::DlRoundRect;
22 
23 static constexpr NSUInteger kFlutterClippingMaskViewPoolCapacity = 5;
24 
25 struct LayerData {
26  SkRect rect;
27  int64_t view_id;
28  int64_t overlay_id;
29  std::shared_ptr<flutter::OverlayLayer> layer;
30 };
31 using LayersMap = std::unordered_map<int64_t, LayerData>;
32 
33 /// Each of the following structs stores part of the platform view hierarchy according to its
34 /// ID.
35 ///
36 /// This data must only be accessed on the platform thread.
38  NSObject<FlutterPlatformView>* view;
40  UIView* root_view;
41 };
42 
43 // Converts a SkMatrix to CATransform3D.
44 //
45 // Certain fields are ignored in CATransform3D since SkMatrix is 3x3 and CATransform3D is 4x4.
46 static CATransform3D GetCATransform3DFromDlMatrix(const DlMatrix& matrix) {
47  CATransform3D transform = CATransform3DIdentity;
48  transform.m11 = matrix.m[0];
49  transform.m12 = matrix.m[1];
50  transform.m13 = matrix.m[2];
51  transform.m14 = matrix.m[3];
52 
53  transform.m21 = matrix.m[4];
54  transform.m22 = matrix.m[5];
55  transform.m23 = matrix.m[6];
56  transform.m24 = matrix.m[7];
57 
58  transform.m31 = matrix.m[8];
59  transform.m32 = matrix.m[9];
60  transform.m33 = matrix.m[10];
61  transform.m34 = matrix.m[11];
62 
63  transform.m41 = matrix.m[12];
64  transform.m42 = matrix.m[13];
65  transform.m43 = matrix.m[14];
66  transform.m44 = matrix.m[15];
67  return transform;
68 }
69 
70 // Reset the anchor of `layer` to match the transform operation from flow.
71 //
72 // The position of the `layer` should be unchanged after resetting the anchor.
73 static void ResetAnchor(CALayer* layer) {
74  // Flow uses (0, 0) to apply transform matrix so we need to match that in Quartz.
75  layer.anchorPoint = CGPointZero;
76  layer.position = CGPointZero;
77 }
78 
79 static CGRect GetCGRectFromDlRect(const DlRect& clipDlRect) {
80  return CGRectMake(clipDlRect.GetLeft(), //
81  clipDlRect.GetTop(), //
82  clipDlRect.GetWidth(), //
83  clipDlRect.GetHeight());
84 }
85 
87 
88 // The pool of reusable view layers. The pool allows to recycle layer in each frame.
89 @property(nonatomic, readonly) flutter::OverlayLayerPool* layerPool;
90 
91 // The platform view's |EmbedderViewSlice| keyed off the view id, which contains any subsequent
92 // operation until the next platform view or the end of the last leaf node in the layer tree.
93 //
94 // The Slices are deleted by the PlatformViewsController.reset().
95 @property(nonatomic, readonly)
96  std::unordered_map<int64_t, std::unique_ptr<flutter::EmbedderViewSlice>>& slices;
97 
98 @property(nonatomic, readonly) FlutterClippingMaskViewPool* maskViewPool;
99 
100 @property(nonatomic, readonly)
101  std::unordered_map<std::string, NSObject<FlutterPlatformViewFactory>*>& factories;
102 
103 // The FlutterPlatformViewGestureRecognizersBlockingPolicy for each type of platform view.
104 @property(nonatomic, readonly)
105  std::unordered_map<std::string, FlutterPlatformViewGestureRecognizersBlockingPolicy>&
106  gestureRecognizersBlockingPolicies;
107 
108 /// The size of the current onscreen surface in physical pixels.
109 @property(nonatomic, assign) SkISize frameSize;
110 
111 /// The task runner for posting tasks to the platform thread.
112 @property(nonatomic, readonly) const fml::RefPtr<fml::TaskRunner>& platformTaskRunner;
113 
114 /// This data must only be accessed on the platform thread.
115 @property(nonatomic, readonly) std::unordered_map<int64_t, PlatformViewData>& platformViews;
116 
117 /// The composition parameters for each platform view.
118 ///
119 /// This state is only modified on the raster thread.
120 @property(nonatomic, readonly)
121  std::unordered_map<int64_t, flutter::EmbeddedViewParams>& currentCompositionParams;
122 
123 /// Method channel `OnDispose` calls adds the views to be disposed to this set to be disposed on
124 /// the next frame.
125 ///
126 /// This state is modified on both the platform and raster thread.
127 @property(nonatomic, readonly) std::unordered_set<int64_t>& viewsToDispose;
128 
129 /// view IDs in composition order.
130 ///
131 /// This state is only modified on the raster thread.
132 @property(nonatomic, readonly) std::vector<int64_t>& compositionOrder;
133 
134 /// platform view IDs visited during layer tree composition.
135 ///
136 /// This state is only modified on the raster thread.
137 @property(nonatomic, readonly) std::vector<int64_t>& visitedPlatformViews;
138 
139 /// Only composite platform views in this set.
140 ///
141 /// This state is only modified on the raster thread.
142 @property(nonatomic, readonly) std::unordered_set<int64_t>& viewsToRecomposite;
143 
144 /// @brief The composition order from the previous thread.
145 ///
146 /// Only accessed from the platform thread.
147 @property(nonatomic, readonly) std::vector<int64_t>& previousCompositionOrder;
148 
149 /// Whether the previous frame had any platform views in active composition order.
150 ///
151 /// This state is tracked so that the first frame after removing the last platform view
152 /// runs through the platform view rendering code path, giving us a chance to remove the
153 /// platform view from the UIView hierarchy.
154 ///
155 /// Only accessed from the raster thread.
156 @property(nonatomic, assign) BOOL hadPlatformViews;
157 
158 /// Whether blurred backdrop filters can be applied.
159 ///
160 /// Defaults to YES, but becomes NO if blurred backdrop filters cannot be applied.
161 @property(nonatomic, assign) BOOL canApplyBlurBackdrop;
162 
163 /// Populate any missing overlay layers.
164 ///
165 /// This requires posting a task to the platform thread and blocking on its completion.
166 - (void)createMissingOverlays:(size_t)requiredOverlayLayers
167  withIosContext:(const std::shared_ptr<flutter::IOSContext>&)iosContext;
168 
169 /// Update the buffers and mutate the platform views in CATransaction on the platform thread.
170 - (void)performSubmit:(const LayersMap&)platformViewLayers
171  currentCompositionParams:
172  (std::unordered_map<int64_t, flutter::EmbeddedViewParams>&)currentCompositionParams
173  viewsToRecomposite:(const std::unordered_set<int64_t>&)viewsToRecomposite
174  compositionOrder:(const std::vector<int64_t>&)compositionOrder
175  unusedLayers:
176  (const std::vector<std::shared_ptr<flutter::OverlayLayer>>&)unusedLayers
177  surfaceFrames:
178  (const std::vector<std::unique_ptr<flutter::SurfaceFrame>>&)surfaceFrames;
179 
180 - (void)onCreate:(FlutterMethodCall*)call result:(FlutterResult)result;
181 - (void)onDispose:(FlutterMethodCall*)call result:(FlutterResult)result;
182 - (void)onAcceptGesture:(FlutterMethodCall*)call result:(FlutterResult)result;
183 - (void)onRejectGesture:(FlutterMethodCall*)call result:(FlutterResult)result;
184 
185 - (void)clipViewSetMaskView:(UIView*)clipView;
186 
187 // Applies the mutators in the mutatorsStack to the UIView chain that was constructed by
188 // `ReconstructClipViewsChain`
189 //
190 // Clips are applied to the `embeddedView`'s super view(|ChildClippingView|) using a
191 // |FlutterClippingMaskView|. Transforms are applied to `embeddedView`
192 //
193 // The `boundingRect` is the final bounding rect of the PlatformView
194 // (EmbeddedViewParams::finalBoundingRect). If a clip mutator's rect contains the final bounding
195 // rect of the PlatformView, the clip mutator is not applied for performance optimization.
196 //
197 // This method is only called when the `embeddedView` needs to be re-composited at the current
198 // frame. See: `compositeView:withParams:` for details.
199 - (void)applyMutators:(const flutter::MutatorsStack&)mutatorsStack
200  embeddedView:(UIView*)embeddedView
201  boundingRect:(const SkRect&)boundingRect;
202 
203 // Appends the overlay views and platform view and sets their z index based on the composition
204 // order.
205 - (void)bringLayersIntoView:(const LayersMap&)layerMap
206  withCompositionOrder:(const std::vector<int64_t>&)compositionOrder;
207 
208 - (std::shared_ptr<flutter::OverlayLayer>)nextLayerInPool;
209 
210 /// Runs on the platform thread.
211 - (void)createLayerWithIosContext:(const std::shared_ptr<flutter::IOSContext>&)iosContext
212  pixelFormat:(MTLPixelFormat)pixelFormat
213  screenScale:(CGFloat)screenScale;
214 
215 /// Removes overlay views and platform views that aren't needed in the current frame.
216 /// Must run on the platform thread.
217 - (void)removeUnusedLayers:(const std::vector<std::shared_ptr<flutter::OverlayLayer>>&)unusedLayers
218  withCompositionOrder:(const std::vector<int64_t>&)compositionOrder;
219 
220 /// Computes and returns all views to be disposed on the platform thread, removes them from
221 /// self.platformViews, self.viewsToRecomposite, and self.currentCompositionParams. Any views that
222 /// still require compositing are not returned, but instead added to `viewsToDelayDispose` for
223 /// disposal on the next call.
224 - (std::vector<UIView*>)computeViewsToDispose;
225 
226 /// Resets the state of the frame.
227 - (void)resetFrameState;
228 @end
229 
231  // TODO(cbracken): Replace with Obj-C types and use @property declarations to automatically
232  // synthesize the ivars.
233  //
234  // These ivars are required because we're transitioning the previous C++ implementation to Obj-C.
235  // We require ivars to declare the concrete types and then wrap with @property declarations that
236  // return a reference to the ivar, allowing for use like `self.layerPool` and
237  // `self.slices[viewId] = x`.
238  std::unique_ptr<flutter::OverlayLayerPool> _layerPool;
239  std::unordered_map<int64_t, std::unique_ptr<flutter::EmbedderViewSlice>> _slices;
240  std::unordered_map<std::string, NSObject<FlutterPlatformViewFactory>*> _factories;
241  std::unordered_map<std::string, FlutterPlatformViewGestureRecognizersBlockingPolicy>
243  fml::RefPtr<fml::TaskRunner> _platformTaskRunner;
244  std::unordered_map<int64_t, PlatformViewData> _platformViews;
245  std::unordered_map<int64_t, flutter::EmbeddedViewParams> _currentCompositionParams;
246  std::unordered_set<int64_t> _viewsToDispose;
247  std::vector<int64_t> _compositionOrder;
248  std::vector<int64_t> _visitedPlatformViews;
249  std::unordered_set<int64_t> _viewsToRecomposite;
250  std::vector<int64_t> _previousCompositionOrder;
251 }
252 
253 - (id)init {
254  if (self = [super init]) {
255  _layerPool = std::make_unique<flutter::OverlayLayerPool>();
256  _maskViewPool =
257  [[FlutterClippingMaskViewPool alloc] initWithCapacity:kFlutterClippingMaskViewPoolCapacity];
258  _hadPlatformViews = NO;
259  _canApplyBlurBackdrop = YES;
260  }
261  return self;
262 }
263 
264 - (const fml::RefPtr<fml::TaskRunner>&)taskRunner {
265  return _platformTaskRunner;
266 }
267 
268 - (void)setTaskRunner:(const fml::RefPtr<fml::TaskRunner>&)platformTaskRunner {
269  _platformTaskRunner = platformTaskRunner;
270 }
271 
272 - (void)onMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
273  if ([[call method] isEqualToString:@"create"]) {
274  [self onCreate:call result:result];
275  } else if ([[call method] isEqualToString:@"dispose"]) {
276  [self onDispose:call result:result];
277  } else if ([[call method] isEqualToString:@"acceptGesture"]) {
278  [self onAcceptGesture:call result:result];
279  } else if ([[call method] isEqualToString:@"rejectGesture"]) {
280  [self onRejectGesture:call result:result];
281  } else {
283  }
284 }
285 
286 - (void)onCreate:(FlutterMethodCall*)call result:(FlutterResult)result {
287  NSDictionary<NSString*, id>* args = [call arguments];
288 
289  int64_t viewId = [args[@"id"] longLongValue];
290  NSString* viewTypeString = args[@"viewType"];
291  std::string viewType(viewTypeString.UTF8String);
292 
293  if (self.platformViews.count(viewId) != 0) {
294  result([FlutterError errorWithCode:@"recreating_view"
295  message:@"trying to create an already created view"
296  details:[NSString stringWithFormat:@"view id: '%lld'", viewId]]);
297  return;
298  }
299 
300  NSObject<FlutterPlatformViewFactory>* factory = self.factories[viewType];
301  if (factory == nil) {
302  result([FlutterError
303  errorWithCode:@"unregistered_view_type"
304  message:[NSString stringWithFormat:@"A UIKitView widget is trying to create a "
305  @"PlatformView with an unregistered type: < %@ >",
306  viewTypeString]
307  details:@"If you are the author of the PlatformView, make sure `registerViewFactory` "
308  @"is invoked.\n"
309  @"See: "
310  @"https://docs.flutter.dev/development/platform-integration/"
311  @"platform-views#on-the-platform-side-1 for more details.\n"
312  @"If you are not the author of the PlatformView, make sure to call "
313  @"`GeneratedPluginRegistrant.register`."]);
314  return;
315  }
316 
317  id params = nil;
318  if ([factory respondsToSelector:@selector(createArgsCodec)]) {
319  NSObject<FlutterMessageCodec>* codec = [factory createArgsCodec];
320  if (codec != nil && args[@"params"] != nil) {
321  FlutterStandardTypedData* paramsData = args[@"params"];
322  params = [codec decode:paramsData.data];
323  }
324  }
325 
326  NSObject<FlutterPlatformView>* embeddedView = [factory createWithFrame:CGRectZero
327  viewIdentifier:viewId
328  arguments:params];
329  UIView* platformView = [embeddedView view];
330  // Set a unique view identifier, so the platform view can be identified in unit tests.
331  platformView.accessibilityIdentifier = [NSString stringWithFormat:@"platform_view[%lld]", viewId];
332 
334  initWithEmbeddedView:platformView
335  platformViewsController:self
336  gestureRecognizersBlockingPolicy:self.gestureRecognizersBlockingPolicies[viewType]];
337 
338  ChildClippingView* clippingView = [[ChildClippingView alloc] initWithFrame:CGRectZero];
339  [clippingView addSubview:touchInterceptor];
340 
341  self.platformViews.emplace(viewId, PlatformViewData{
342  .view = embeddedView, //
343  .touch_interceptor = touchInterceptor, //
344  .root_view = clippingView //
345  });
346 
347  result(nil);
348 }
349 
350 - (void)onDispose:(FlutterMethodCall*)call result:(FlutterResult)result {
351  NSNumber* arg = [call arguments];
352  int64_t viewId = [arg longLongValue];
353 
354  if (self.platformViews.count(viewId) == 0) {
355  result([FlutterError errorWithCode:@"unknown_view"
356  message:@"trying to dispose an unknown"
357  details:[NSString stringWithFormat:@"view id: '%lld'", viewId]]);
358  return;
359  }
360  // We wait for next submitFrame to dispose views.
361  self.viewsToDispose.insert(viewId);
362  result(nil);
363 }
364 
365 - (void)onAcceptGesture:(FlutterMethodCall*)call result:(FlutterResult)result {
366  NSDictionary<NSString*, id>* args = [call arguments];
367  int64_t viewId = [args[@"id"] longLongValue];
368 
369  if (self.platformViews.count(viewId) == 0) {
370  result([FlutterError errorWithCode:@"unknown_view"
371  message:@"trying to set gesture state for an unknown view"
372  details:[NSString stringWithFormat:@"view id: '%lld'", viewId]]);
373  return;
374  }
375 
376  FlutterTouchInterceptingView* view = self.platformViews[viewId].touch_interceptor;
377  [view releaseGesture];
378 
379  result(nil);
380 }
381 
382 - (void)onRejectGesture:(FlutterMethodCall*)call result:(FlutterResult)result {
383  NSDictionary<NSString*, id>* args = [call arguments];
384  int64_t viewId = [args[@"id"] longLongValue];
385 
386  if (self.platformViews.count(viewId) == 0) {
387  result([FlutterError errorWithCode:@"unknown_view"
388  message:@"trying to set gesture state for an unknown view"
389  details:[NSString stringWithFormat:@"view id: '%lld'", viewId]]);
390  return;
391  }
392 
393  FlutterTouchInterceptingView* view = self.platformViews[viewId].touch_interceptor;
394  [view blockGesture];
395 
396  result(nil);
397 }
398 
399 - (void)registerViewFactory:(NSObject<FlutterPlatformViewFactory>*)factory
400  withId:(NSString*)factoryId
401  gestureRecognizersBlockingPolicy:
402  (FlutterPlatformViewGestureRecognizersBlockingPolicy)gestureRecognizerBlockingPolicy {
403  std::string idString([factoryId UTF8String]);
404  FML_CHECK(self.factories.count(idString) == 0);
405  self.factories[idString] = factory;
406  self.gestureRecognizersBlockingPolicies[idString] = gestureRecognizerBlockingPolicy;
407 }
408 
409 - (void)beginFrameWithSize:(SkISize)frameSize {
410  [self resetFrameState];
411  self.frameSize = frameSize;
412 }
413 
414 - (void)cancelFrame {
415  [self resetFrameState];
416 }
417 
418 - (flutter::PostPrerollResult)postPrerollActionWithThreadMerger:
419  (const fml::RefPtr<fml::RasterThreadMerger>&)rasterThreadMerger {
420  return flutter::PostPrerollResult::kSuccess;
421 }
422 
423 - (void)endFrameWithResubmit:(BOOL)shouldResubmitFrame
424  threadMerger:(const fml::RefPtr<fml::RasterThreadMerger>&)rasterThreadMerger {
425 }
426 
427 - (void)pushFilterToVisitedPlatformViews:(const std::shared_ptr<flutter::DlImageFilter>&)filter
428  withRect:(const SkRect&)filterRect {
429  for (int64_t id : self.visitedPlatformViews) {
430  flutter::EmbeddedViewParams params = self.currentCompositionParams[id];
431  params.PushImageFilter(filter, filterRect);
432  self.currentCompositionParams[id] = params;
433  }
434 }
435 
436 - (void)prerollCompositeEmbeddedView:(int64_t)viewId
437  withParams:(std::unique_ptr<flutter::EmbeddedViewParams>)params {
438  SkRect viewBounds = SkRect::Make(self.frameSize);
439  std::unique_ptr<flutter::EmbedderViewSlice> view;
440  view = std::make_unique<flutter::DisplayListEmbedderViewSlice>(viewBounds);
441  self.slices.insert_or_assign(viewId, std::move(view));
442 
443  self.compositionOrder.push_back(viewId);
444 
445  if (self.currentCompositionParams.count(viewId) == 1 &&
446  self.currentCompositionParams[viewId] == *params.get()) {
447  // Do nothing if the params didn't change.
448  return;
449  }
450  self.currentCompositionParams[viewId] = flutter::EmbeddedViewParams(*params.get());
451  self.viewsToRecomposite.insert(viewId);
452 }
453 
454 - (size_t)embeddedViewCount {
455  return self.compositionOrder.size();
456 }
457 
458 - (UIView*)platformViewForId:(int64_t)viewId {
459  return [self flutterTouchInterceptingViewForId:viewId].embeddedView;
460 }
461 
462 - (FlutterTouchInterceptingView*)flutterTouchInterceptingViewForId:(int64_t)viewId {
463  if (self.platformViews.empty()) {
464  return nil;
465  }
466  return self.platformViews[viewId].touch_interceptor;
467 }
468 
469 - (long)firstResponderPlatformViewId {
470  for (auto const& [id, platformViewData] : self.platformViews) {
471  UIView* rootView = platformViewData.root_view;
472  if (rootView.flt_hasFirstResponderInViewHierarchySubtree) {
473  return id;
474  }
475  }
476  return -1;
477 }
478 
479 - (void)clipViewSetMaskView:(UIView*)clipView {
480  FML_DCHECK([[NSThread currentThread] isMainThread]);
481  if (clipView.maskView) {
482  return;
483  }
484  CGRect frame =
485  CGRectMake(-clipView.frame.origin.x, -clipView.frame.origin.y,
486  CGRectGetWidth(self.flutterView.bounds), CGRectGetHeight(self.flutterView.bounds));
487  clipView.maskView = [self.maskViewPool
488  getMaskViewWithFrame:frame
489  screenScale:[self.flutterViewController flutterScreenIfViewLoaded].scale];
490 }
491 
492 - (void)applyMutators:(const flutter::MutatorsStack&)mutatorsStack
493  embeddedView:(UIView*)embeddedView
494  boundingRect:(const SkRect&)boundingRect {
495  if (self.flutterView == nil) {
496  return;
497  }
498 
499  ResetAnchor(embeddedView.layer);
500  ChildClippingView* clipView = (ChildClippingView*)embeddedView.superview;
501 
502  DlMatrix transformMatrix;
503  const DlRect& dlBoundingRect = flutter::ToDlRect(boundingRect);
504  NSMutableArray* blurFilters = [[NSMutableArray alloc] init];
505  FML_DCHECK(!clipView.maskView ||
506  [clipView.maskView isKindOfClass:[FlutterClippingMaskView class]]);
507  if (clipView.maskView) {
508  [self.maskViewPool insertViewToPoolIfNeeded:(FlutterClippingMaskView*)(clipView.maskView)];
509  clipView.maskView = nil;
510  }
511  CGFloat screenScale = [self.flutterViewController flutterScreenIfViewLoaded].scale;
512  auto iter = mutatorsStack.Begin();
513  while (iter != mutatorsStack.End()) {
514  switch ((*iter)->GetType()) {
515  case flutter::MutatorType::kTransform: {
516  transformMatrix = transformMatrix * (*iter)->GetMatrix();
517  break;
518  }
519  case flutter::MutatorType::kClipRect: {
520  if (flutter::DisplayListMatrixClipState::TransformedRectCoversBounds(
521  (*iter)->GetRect(), transformMatrix, dlBoundingRect)) {
522  break;
523  }
524  [self clipViewSetMaskView:clipView];
525  [(FlutterClippingMaskView*)clipView.maskView clipRect:(*iter)->GetRect()
526  matrix:transformMatrix];
527  break;
528  }
529  case flutter::MutatorType::kClipRRect: {
530  if (flutter::DisplayListMatrixClipState::TransformedRRectCoversBounds(
531  (*iter)->GetRRect(), transformMatrix, dlBoundingRect)) {
532  break;
533  }
534  [self clipViewSetMaskView:clipView];
535  [(FlutterClippingMaskView*)clipView.maskView clipRRect:(*iter)->GetRRect()
536  matrix:transformMatrix];
537  break;
538  }
539  case flutter::MutatorType::kClipRSE: {
540  if (flutter::DisplayListMatrixClipState::TransformedRoundSuperellipseCoversBounds(
541  (*iter)->GetRSE(), transformMatrix, dlBoundingRect)) {
542  break;
543  }
544  [self clipViewSetMaskView:clipView];
545  [(FlutterClippingMaskView*)clipView.maskView clipRRect:(*iter)->GetRRect()
546  matrix:transformMatrix];
547  break;
548  }
549  case flutter::MutatorType::kClipPath: {
550  // TODO(cyanglaz): Find a way to pre-determine if path contains the PlatformView boudning
551  // rect. See `ClipRRectContainsPlatformViewBoundingRect`.
552  // https://github.com/flutter/flutter/issues/118650
553  [self clipViewSetMaskView:clipView];
554  [(FlutterClippingMaskView*)clipView.maskView clipPath:(*iter)->GetPath()
555  matrix:transformMatrix];
556  break;
557  }
558  case flutter::MutatorType::kOpacity:
559  embeddedView.alpha = (*iter)->GetAlphaFloat() * embeddedView.alpha;
560  break;
561  case flutter::MutatorType::kBackdropFilter: {
562  // Only support DlBlurImageFilter for BackdropFilter.
563  if (!self.canApplyBlurBackdrop || !(*iter)->GetFilterMutation().GetFilter().asBlur()) {
564  break;
565  }
566  CGRect filterRect = GetCGRectFromDlRect((*iter)->GetFilterMutation().GetFilterRect());
567  // `filterRect` is in global coordinates. We need to convert to local space.
568  filterRect = CGRectApplyAffineTransform(
569  filterRect, CGAffineTransformMakeScale(1 / screenScale, 1 / screenScale));
570  // `filterRect` reprents the rect that should be filtered inside the `_flutterView`.
571  // The `PlatformViewFilter` needs the frame inside the `clipView` that needs to be
572  // filtered.
573  if (CGRectIsNull(CGRectIntersection(filterRect, clipView.frame))) {
574  break;
575  }
576  CGRect intersection = CGRectIntersection(filterRect, clipView.frame);
577  CGRect frameInClipView = [self.flutterView convertRect:intersection toView:clipView];
578  // sigma_x is arbitrarily chosen as the radius value because Quartz sets
579  // sigma_x and sigma_y equal to each other. DlBlurImageFilter's Tile Mode
580  // is not supported in Quartz's gaussianBlur CAFilter, so it is not used
581  // to blur the PlatformView.
582  CGFloat blurRadius = (*iter)->GetFilterMutation().GetFilter().asBlur()->sigma_x();
583  UIVisualEffectView* visualEffectView = [[UIVisualEffectView alloc]
584  initWithEffect:[UIBlurEffect effectWithStyle:UIBlurEffectStyleLight]];
585  PlatformViewFilter* filter = [[PlatformViewFilter alloc] initWithFrame:frameInClipView
586  blurRadius:blurRadius
587  visualEffectView:visualEffectView];
588  if (!filter) {
589  self.canApplyBlurBackdrop = NO;
590  } else {
591  [blurFilters addObject:filter];
592  }
593  break;
594  }
595  }
596  ++iter;
597  }
598 
599  if (self.canApplyBlurBackdrop) {
600  [clipView applyBlurBackdropFilters:blurFilters];
601  }
602 
603  // The UIKit frame is set based on the logical resolution (points) instead of physical.
604  // (https://developer.apple.com/library/archive/documentation/DeviceInformation/Reference/iOSDeviceCompatibility/Displays/Displays.html).
605  // However, flow is based on the physical resolution. For example, 1000 pixels in flow equals
606  // 500 points in UIKit for devices that has screenScale of 2. We need to scale the transformMatrix
607  // down to the logical resoltion before applying it to the layer of PlatformView.
608  flutter::DlScalar pointScale = 1.0 / screenScale;
609  transformMatrix = DlMatrix::MakeScale({pointScale, pointScale, 1}) * transformMatrix;
610 
611  // Reverse the offset of the clipView.
612  // The clipView's frame includes the final translate of the final transform matrix.
613  // Thus, this translate needs to be reversed so the platform view can layout at the correct
614  // offset.
615  //
616  // Note that the transforms are not applied to the clipping paths because clipping paths happen on
617  // the mask view, whose origin is always (0,0) to the _flutterView.
618  impeller::Vector3 origin = impeller::Vector3(clipView.frame.origin.x, clipView.frame.origin.y);
619  transformMatrix = DlMatrix::MakeTranslation(-origin) * transformMatrix;
620 
621  embeddedView.layer.transform = GetCATransform3DFromDlMatrix(transformMatrix);
622 }
623 
624 - (void)compositeView:(int64_t)viewId withParams:(const flutter::EmbeddedViewParams&)params {
625  // TODO(https://github.com/flutter/flutter/issues/109700)
626  CGRect frame = CGRectMake(0, 0, params.sizePoints().width(), params.sizePoints().height());
627  FlutterTouchInterceptingView* touchInterceptor = self.platformViews[viewId].touch_interceptor;
628  touchInterceptor.layer.transform = CATransform3DIdentity;
629  touchInterceptor.frame = frame;
630  touchInterceptor.alpha = 1;
631 
632  const flutter::MutatorsStack& mutatorStack = params.mutatorsStack();
633  UIView* clippingView = self.platformViews[viewId].root_view;
634  // The frame of the clipping view should be the final bounding rect.
635  // Because the translate matrix in the Mutator Stack also includes the offset,
636  // when we apply the transforms matrix in |applyMutators:embeddedView:boundingRect|, we need
637  // to remember to do a reverse translate.
638  const SkRect& rect = params.finalBoundingRect();
639  CGFloat screenScale = [self.flutterViewController flutterScreenIfViewLoaded].scale;
640  clippingView.frame = CGRectMake(rect.x() / screenScale, rect.y() / screenScale,
641  rect.width() / screenScale, rect.height() / screenScale);
642  [self applyMutators:mutatorStack embeddedView:touchInterceptor boundingRect:rect];
643 }
644 
645 - (flutter::DlCanvas*)compositeEmbeddedViewWithId:(int64_t)viewId {
646  FML_DCHECK(self.slices.find(viewId) != self.slices.end());
647  return self.slices[viewId]->canvas();
648 }
649 
650 - (void)reset {
651  // Reset will only be called from the raster thread or a merged raster/platform thread.
652  // _platformViews must only be modified on the platform thread, and any operations that
653  // read or modify platform views should occur there.
654  fml::TaskRunner::RunNowOrPostTask(self.platformTaskRunner, [self]() {
655  for (int64_t viewId : self.compositionOrder) {
656  [self.platformViews[viewId].root_view removeFromSuperview];
657  }
658  self.platformViews.clear();
659  self.previousCompositionOrder.clear();
660  });
661 
662  self.compositionOrder.clear();
663  self.slices.clear();
664  self.currentCompositionParams.clear();
665  self.viewsToRecomposite.clear();
666  self.layerPool->RecycleLayers();
667  self.visitedPlatformViews.clear();
668 }
669 
670 - (BOOL)submitFrame:(std::unique_ptr<flutter::SurfaceFrame>)background_frame
671  withIosContext:(const std::shared_ptr<flutter::IOSContext>&)iosContext {
672  TRACE_EVENT0("flutter", "PlatformViewsController::SubmitFrame");
673 
674  // No platform views to render; we're done.
675  if (self.flutterView == nil || (self.compositionOrder.empty() && !self.hadPlatformViews)) {
676  self.hadPlatformViews = NO;
677  return background_frame->Submit();
678  }
679  self.hadPlatformViews = !self.compositionOrder.empty();
680 
681  bool didEncode = true;
682  LayersMap platformViewLayers;
683  std::vector<std::unique_ptr<flutter::SurfaceFrame>> surfaceFrames;
684  surfaceFrames.reserve(self.compositionOrder.size());
685  std::unordered_map<int64_t, SkRect> viewRects;
686 
687  for (int64_t viewId : self.compositionOrder) {
688  viewRects[viewId] = self.currentCompositionParams[viewId].finalBoundingRect();
689  }
690 
691  std::unordered_map<int64_t, SkRect> overlayLayers =
692  SliceViews(background_frame->Canvas(), self.compositionOrder, self.slices, viewRects);
693 
694  size_t requiredOverlayLayers = 0;
695  for (int64_t viewId : self.compositionOrder) {
696  std::unordered_map<int64_t, SkRect>::const_iterator overlay = overlayLayers.find(viewId);
697  if (overlay == overlayLayers.end()) {
698  continue;
699  }
700  requiredOverlayLayers++;
701  }
702 
703  // If there are not sufficient overlay layers, we must construct them on the platform
704  // thread, at least until we've refactored iOS surface creation to use IOSurfaces
705  // instead of CALayers.
706  [self createMissingOverlays:requiredOverlayLayers withIosContext:iosContext];
707 
708  int64_t overlayId = 0;
709  for (int64_t viewId : self.compositionOrder) {
710  std::unordered_map<int64_t, SkRect>::const_iterator overlay = overlayLayers.find(viewId);
711  if (overlay == overlayLayers.end()) {
712  continue;
713  }
714  std::shared_ptr<flutter::OverlayLayer> layer = self.nextLayerInPool;
715  if (!layer) {
716  continue;
717  }
718 
719  std::unique_ptr<flutter::SurfaceFrame> frame = layer->surface->AcquireFrame(self.frameSize);
720  // If frame is null, AcquireFrame already printed out an error message.
721  if (!frame) {
722  continue;
723  }
724  flutter::DlCanvas* overlayCanvas = frame->Canvas();
725  int restoreCount = overlayCanvas->GetSaveCount();
726  overlayCanvas->Save();
727  overlayCanvas->ClipRect(flutter::ToDlRect(overlay->second));
728  overlayCanvas->Clear(flutter::DlColor::kTransparent());
729  self.slices[viewId]->render_into(overlayCanvas);
730  overlayCanvas->RestoreToCount(restoreCount);
731 
732  // This flutter view is never the last in a frame, since we always submit the
733  // underlay view last.
734  frame->set_submit_info({.frame_boundary = false, .present_with_transaction = true});
735  layer->did_submit_last_frame = frame->Encode();
736 
737  didEncode &= layer->did_submit_last_frame;
738  platformViewLayers[viewId] = LayerData{
739  .rect = overlay->second, //
740  .view_id = viewId, //
741  .overlay_id = overlayId, //
742  .layer = layer //
743  };
744  surfaceFrames.push_back(std::move(frame));
745  overlayId++;
746  }
747 
748  auto previousSubmitInfo = background_frame->submit_info();
749  background_frame->set_submit_info({
750  .frame_damage = previousSubmitInfo.frame_damage,
751  .buffer_damage = previousSubmitInfo.buffer_damage,
752  .present_with_transaction = true,
753  });
754  background_frame->Encode();
755  surfaceFrames.push_back(std::move(background_frame));
756 
757  // Mark all layers as available, so they can be used in the next frame.
758  std::vector<std::shared_ptr<flutter::OverlayLayer>> unusedLayers =
759  self.layerPool->RemoveUnusedLayers();
760  self.layerPool->RecycleLayers();
761 
762  auto task = [self, //
763  platformViewLayers = std::move(platformViewLayers), //
764  currentCompositionParams = self.currentCompositionParams, //
765  viewsToRecomposite = self.viewsToRecomposite, //
766  compositionOrder = self.compositionOrder, //
767  unusedLayers = std::move(unusedLayers), //
768  surfaceFrames = std::move(surfaceFrames) //
769  ]() mutable {
770  [self performSubmit:platformViewLayers
771  currentCompositionParams:currentCompositionParams
772  viewsToRecomposite:viewsToRecomposite
773  compositionOrder:compositionOrder
774  unusedLayers:unusedLayers
775  surfaceFrames:surfaceFrames];
776  };
777 
778  fml::TaskRunner::RunNowOrPostTask(self.platformTaskRunner, fml::MakeCopyable(std::move(task)));
779 
780  return didEncode;
781 }
782 
783 - (void)createMissingOverlays:(size_t)requiredOverlayLayers
784  withIosContext:(const std::shared_ptr<flutter::IOSContext>&)iosContext {
785  TRACE_EVENT0("flutter", "PlatformViewsController::CreateMissingLayers");
786 
787  if (requiredOverlayLayers <= self.layerPool->size()) {
788  return;
789  }
790  auto missingLayerCount = requiredOverlayLayers - self.layerPool->size();
791 
792  // If the raster thread isn't merged, create layers on the platform thread and block until
793  // complete.
794  auto latch = std::make_shared<fml::CountDownLatch>(1u);
795  fml::TaskRunner::RunNowOrPostTask(
796  self.platformTaskRunner, [self, missingLayerCount, iosContext, latch]() {
797  for (auto i = 0u; i < missingLayerCount; i++) {
798  [self createLayerWithIosContext:iosContext
799  pixelFormat:((FlutterView*)self.flutterView).pixelFormat
800  screenScale:((FlutterView*)self.flutterView).screen.scale];
801  }
802  latch->CountDown();
803  });
804  if (![[NSThread currentThread] isMainThread]) {
805  latch->Wait();
806  }
807 }
808 
809 - (void)performSubmit:(const LayersMap&)platformViewLayers
810  currentCompositionParams:
811  (std::unordered_map<int64_t, flutter::EmbeddedViewParams>&)currentCompositionParams
812  viewsToRecomposite:(const std::unordered_set<int64_t>&)viewsToRecomposite
813  compositionOrder:(const std::vector<int64_t>&)compositionOrder
814  unusedLayers:
815  (const std::vector<std::shared_ptr<flutter::OverlayLayer>>&)unusedLayers
816  surfaceFrames:
817  (const std::vector<std::unique_ptr<flutter::SurfaceFrame>>&)surfaceFrames {
818  TRACE_EVENT0("flutter", "PlatformViewsController::PerformSubmit");
819  FML_DCHECK([[NSThread currentThread] isMainThread]);
820 
821  [CATransaction begin];
822 
823  // Configure Flutter overlay views.
824  for (const auto& [viewId, layerData] : platformViewLayers) {
825  layerData.layer->UpdateViewState(self.flutterView, //
826  layerData.rect, //
827  layerData.view_id, //
828  layerData.overlay_id //
829  );
830  }
831 
832  // Dispose unused Flutter Views.
833  for (auto& view : [self computeViewsToDispose]) {
834  [view removeFromSuperview];
835  }
836 
837  // Composite Platform Views.
838  for (int64_t viewId : viewsToRecomposite) {
839  [self compositeView:viewId withParams:currentCompositionParams[viewId]];
840  }
841 
842  // Present callbacks.
843  for (const auto& frame : surfaceFrames) {
844  frame->Submit();
845  }
846 
847  // If a layer was allocated in the previous frame, but it's not used in the current frame,
848  // then it can be removed from the scene.
849  [self removeUnusedLayers:unusedLayers withCompositionOrder:compositionOrder];
850 
851  // Organize the layers by their z indexes.
852  [self bringLayersIntoView:platformViewLayers withCompositionOrder:compositionOrder];
853 
854  [CATransaction commit];
855 }
856 
857 - (void)bringLayersIntoView:(const LayersMap&)layerMap
858  withCompositionOrder:(const std::vector<int64_t>&)compositionOrder {
859  FML_DCHECK(self.flutterView);
860  UIView* flutterView = self.flutterView;
861 
862  self.previousCompositionOrder.clear();
863  NSMutableArray* desiredPlatformSubviews = [NSMutableArray array];
864  for (int64_t platformViewId : compositionOrder) {
865  self.previousCompositionOrder.push_back(platformViewId);
866  UIView* platformViewRoot = self.platformViews[platformViewId].root_view;
867  if (platformViewRoot != nil) {
868  [desiredPlatformSubviews addObject:platformViewRoot];
869  }
870 
871  auto maybeLayerData = layerMap.find(platformViewId);
872  if (maybeLayerData != layerMap.end()) {
873  auto view = maybeLayerData->second.layer->overlay_view_wrapper;
874  if (view != nil) {
875  [desiredPlatformSubviews addObject:view];
876  }
877  }
878  }
879 
880  NSSet* desiredPlatformSubviewsSet = [NSSet setWithArray:desiredPlatformSubviews];
881  NSArray* existingPlatformSubviews = [flutterView.subviews
882  filteredArrayUsingPredicate:[NSPredicate
883  predicateWithBlock:^BOOL(id object, NSDictionary* bindings) {
884  return [desiredPlatformSubviewsSet containsObject:object];
885  }]];
886 
887  // Manipulate view hierarchy only if needed, to address a performance issue where
888  // this method is called even when view hierarchy stays the same.
889  // See: https://github.com/flutter/flutter/issues/121833
890  // TODO(hellohuanlin): investigate if it is possible to skip unnecessary bringLayersIntoView.
891  if (![desiredPlatformSubviews isEqualToArray:existingPlatformSubviews]) {
892  for (UIView* subview in desiredPlatformSubviews) {
893  // `addSubview` will automatically reorder subview if it is already added.
894  [flutterView addSubview:subview];
895  }
896  }
897 }
898 
899 - (std::shared_ptr<flutter::OverlayLayer>)nextLayerInPool {
900  return self.layerPool->GetNextLayer();
901 }
902 
903 - (void)createLayerWithIosContext:(const std::shared_ptr<flutter::IOSContext>&)iosContext
904  pixelFormat:(MTLPixelFormat)pixelFormat
905  screenScale:(CGFloat)screenScale {
906  self.layerPool->CreateLayer(iosContext, pixelFormat, screenScale);
907 }
908 
909 - (void)removeUnusedLayers:(const std::vector<std::shared_ptr<flutter::OverlayLayer>>&)unusedLayers
910  withCompositionOrder:(const std::vector<int64_t>&)compositionOrder {
911  for (const std::shared_ptr<flutter::OverlayLayer>& layer : unusedLayers) {
912  [layer->overlay_view_wrapper removeFromSuperview];
913  }
914 
915  std::unordered_set<int64_t> compositionOrderSet;
916  for (int64_t viewId : compositionOrder) {
917  compositionOrderSet.insert(viewId);
918  }
919  // Remove unused platform views.
920  for (int64_t viewId : self.previousCompositionOrder) {
921  if (compositionOrderSet.find(viewId) == compositionOrderSet.end()) {
922  UIView* platformViewRoot = self.platformViews[viewId].root_view;
923  [platformViewRoot removeFromSuperview];
924  }
925  }
926 }
927 
928 - (std::vector<UIView*>)computeViewsToDispose {
929  std::vector<UIView*> views;
930  if (self.viewsToDispose.empty()) {
931  return views;
932  }
933 
934  std::unordered_set<int64_t> viewsToComposite(self.compositionOrder.begin(),
935  self.compositionOrder.end());
936  std::unordered_set<int64_t> viewsToDelayDispose;
937  for (int64_t viewId : self.viewsToDispose) {
938  if (viewsToComposite.count(viewId)) {
939  viewsToDelayDispose.insert(viewId);
940  continue;
941  }
942  UIView* rootView = self.platformViews[viewId].root_view;
943  views.push_back(rootView);
944  self.currentCompositionParams.erase(viewId);
945  self.viewsToRecomposite.erase(viewId);
946  self.platformViews.erase(viewId);
947  }
948  self.viewsToDispose = std::move(viewsToDelayDispose);
949  return views;
950 }
951 
952 - (void)resetFrameState {
953  self.slices.clear();
954  self.compositionOrder.clear();
955  self.visitedPlatformViews.clear();
956 }
957 
958 - (void)pushVisitedPlatformViewId:(int64_t)viewId {
959  self.visitedPlatformViews.push_back(viewId);
960 }
961 
962 - (const flutter::EmbeddedViewParams&)compositionParamsForView:(int64_t)viewId {
963  return self.currentCompositionParams.find(viewId)->second;
964 }
965 
966 #pragma mark - Properties
967 
968 - (flutter::OverlayLayerPool*)layerPool {
969  return _layerPool.get();
970 }
971 
972 - (std::unordered_map<int64_t, std::unique_ptr<flutter::EmbedderViewSlice>>&)slices {
973  return _slices;
974 }
975 
976 - (std::unordered_map<std::string, NSObject<FlutterPlatformViewFactory>*>&)factories {
977  return _factories;
978 }
979 - (std::unordered_map<std::string, FlutterPlatformViewGestureRecognizersBlockingPolicy>&)
980  gestureRecognizersBlockingPolicies {
982 }
983 
984 - (std::unordered_map<int64_t, PlatformViewData>&)platformViews {
985  return _platformViews;
986 }
987 
988 - (std::unordered_map<int64_t, flutter::EmbeddedViewParams>&)currentCompositionParams {
990 }
991 
992 - (std::unordered_set<int64_t>&)viewsToDispose {
993  return _viewsToDispose;
994 }
995 
996 - (std::vector<int64_t>&)compositionOrder {
997  return _compositionOrder;
998 }
999 
1000 - (std::vector<int64_t>&)visitedPlatformViews {
1001  return _visitedPlatformViews;
1002 }
1003 
1004 - (std::unordered_set<int64_t>&)viewsToRecomposite {
1005  return _viewsToRecomposite;
1006 }
1007 
1008 - (std::vector<int64_t>&)previousCompositionOrder {
1010 }
1011 
1012 @end
self
return self
Definition: FlutterTextureRegistryRelay.mm:19
_compositionOrder
std::vector< int64_t > _compositionOrder
Definition: FlutterPlatformViewsController.mm:247
_gestureRecognizersBlockingPolicies
std::unordered_map< std::string, FlutterPlatformViewGestureRecognizersBlockingPolicy > _gestureRecognizersBlockingPolicies
Definition: FlutterPlatformViewsController.mm:242
FlutterPlatformViewsController.h
ChildClippingView
Definition: FlutterPlatformViews.mm:152
ResetAnchor
static void ResetAnchor(CALayer *layer)
Definition: FlutterPlatformViewsController.mm:73
-[FlutterTouchInterceptingView blockGesture]
void blockGesture()
Definition: FlutterPlatformViews.mm:604
PlatformViewData::touch_interceptor
FlutterTouchInterceptingView * touch_interceptor
Definition: FlutterPlatformViewsController.mm:39
FlutterMethodNotImplemented
FLUTTER_DARWIN_EXPORT NSObject const * FlutterMethodNotImplemented
-[FlutterTouchInterceptingView releaseGesture]
void releaseGesture()
Definition: FlutterPlatformViews.mm:585
FlutterClippingMaskView
Definition: FlutterPlatformViews.mm:215
_viewsToRecomposite
std::unordered_set< int64_t > _viewsToRecomposite
Definition: FlutterPlatformViewsController.mm:249
_platformViews
std::unordered_map< int64_t, PlatformViewData > _platformViews
Definition: FlutterPlatformViewsController.mm:244
PlatformViewFilter
Definition: FlutterPlatformViews.mm:60
FlutterError
Definition: FlutterCodecs.h:246
LayersMap
std::unordered_map< int64_t, LayerData > LayersMap
Definition: FlutterPlatformViewsController.mm:31
_factories
std::unordered_map< std::string, NSObject< FlutterPlatformViewFactory > * > _factories
Definition: FlutterPlatformViewsController.mm:240
-[ChildClippingView applyBlurBackdropFilters:]
void applyBlurBackdropFilters:(NSArray< PlatformViewFilter * > *filters)
Definition: FlutterPlatformViews.mm:166
PlatformViewData
Definition: FlutterPlatformViewsController.mm:37
LayerData::layer
std::shared_ptr< flutter::OverlayLayer > layer
Definition: FlutterPlatformViewsController.mm:29
_previousCompositionOrder
std::vector< int64_t > _previousCompositionOrder
Definition: FlutterPlatformViewsController.mm:250
LayerData::overlay_id
int64_t overlay_id
Definition: FlutterPlatformViewsController.mm:28
PlatformViewData::view
NSObject< FlutterPlatformView > * view
Definition: FlutterPlatformViewsController.mm:38
_slices
std::unordered_map< int64_t, std::unique_ptr< flutter::EmbedderViewSlice > > _slices
Definition: FlutterPlatformViewsController.mm:230
FlutterClippingMaskViewPool
Definition: FlutterPlatformViews.mm:485
ios_surface.h
GetCGRectFromDlRect
static CGRect GetCGRectFromDlRect(const DlRect &clipDlRect)
Definition: FlutterPlatformViewsController.mm:79
LayerData
Definition: FlutterPlatformViewsController.mm:25
FlutterMethodCall
Definition: FlutterCodecs.h:220
-[FlutterPlatformViewsController(Testing) previousCompositionOrder]
std::vector< int64_t > & previousCompositionOrder()
flutter
Definition: accessibility_bridge.h:26
FlutterOverlayView.h
_viewsToDispose
std::unordered_set< int64_t > _viewsToDispose
Definition: FlutterPlatformViewsController.mm:246
FlutterResult
void(^ FlutterResult)(id _Nullable result)
Definition: FlutterChannels.h:194
fml
Definition: profiler_metrics_ios.mm:41
UIViewController+FlutterScreenAndSceneIfLoaded.h
FlutterPlatformViewGestureRecognizersBlockingPolicy
FlutterPlatformViewGestureRecognizersBlockingPolicy
Definition: FlutterPlugin.h:252
_visitedPlatformViews
std::vector< int64_t > _visitedPlatformViews
Definition: FlutterPlatformViewsController.mm:248
PlatformViewData::root_view
UIView * root_view
Definition: FlutterPlatformViewsController.mm:40
GetCATransform3DFromDlMatrix
static CATransform3D GetCATransform3DFromDlMatrix(const DlMatrix &matrix)
Definition: FlutterPlatformViewsController.mm:46
FlutterPlatformViewFactory-p
Definition: FlutterPlatformViews.h:26
overlay_layer_pool.h
FlutterPlatformViewsController
Definition: FlutterPlatformViewsController.h:30
_currentCompositionParams
std::unordered_map< int64_t, flutter::EmbeddedViewParams > _currentCompositionParams
Definition: FlutterPlatformViewsController.mm:245
LayerData::view_id
int64_t view_id
Definition: FlutterPlatformViewsController.mm:27
FlutterStandardTypedData
Definition: FlutterCodecs.h:300
FlutterTouchInterceptingView
Definition: FlutterPlatformViews.mm:541
kFlutterClippingMaskViewPoolCapacity
static constexpr NSUInteger kFlutterClippingMaskViewPoolCapacity
Definition: FlutterPlatformViewsController.mm:23
LayerData::rect
SkRect rect
Definition: FlutterPlatformViewsController.mm:26
FlutterView.h
_platformTaskRunner
fml::RefPtr< fml::TaskRunner > _platformTaskRunner
Definition: FlutterPlatformViewsController.mm:243
FlutterMethodCall::arguments
id arguments
Definition: FlutterCodecs.h:238
flutter::OverlayLayerPool
Storage for Overlay layers across frames.
Definition: overlay_layer_pool.h:47