Flutter iOS Embedder
FlutterPlatformViews.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 #import <WebKit/WebKit.h>
8 
9 #include "flutter/display_list/effects/dl_image_filter.h"
10 #include "flutter/fml/platform/darwin/cf_utils.h"
12 
14 
15 namespace {
16 static CGRect GetCGRectFromDlRect(const flutter::DlRect& clipDlRect) {
17  return CGRectMake(clipDlRect.GetX(), //
18  clipDlRect.GetY(), //
19  clipDlRect.GetWidth(), //
20  clipDlRect.GetHeight());
21 }
22 
23 CATransform3D GetCATransform3DFromDlMatrix(const flutter::DlMatrix& matrix) {
24  CATransform3D transform = CATransform3DIdentity;
25  transform.m11 = matrix.m[0];
26  transform.m12 = matrix.m[1];
27  transform.m13 = matrix.m[2];
28  transform.m14 = matrix.m[3];
29 
30  transform.m21 = matrix.m[4];
31  transform.m22 = matrix.m[5];
32  transform.m23 = matrix.m[6];
33  transform.m24 = matrix.m[7];
34 
35  transform.m31 = matrix.m[8];
36  transform.m32 = matrix.m[9];
37  transform.m33 = matrix.m[10];
38  transform.m34 = matrix.m[11];
39 
40  transform.m41 = matrix.m[12];
41  transform.m42 = matrix.m[13];
42  transform.m43 = matrix.m[14];
43  transform.m44 = matrix.m[15];
44  return transform;
45 }
46 } // namespace
47 
48 @interface PlatformViewFilter ()
49 
50 // `YES` if the backdropFilterView has been configured at least once.
51 @property(nonatomic) BOOL backdropFilterViewConfigured;
52 @property(nonatomic) UIVisualEffectView* backdropFilterView;
53 
54 // Updates the `visualEffectView` with the current filter parameters.
55 // Also sets `self.backdropFilterView` to the updated visualEffectView.
56 - (void)updateVisualEffectView:(UIVisualEffectView*)visualEffectView;
57 
58 @end
59 
60 @implementation PlatformViewFilter
61 
62 static NSObject* _gaussianBlurFilter = nil;
63 // The index of "_UIVisualEffectBackdropView" in UIVisualEffectView's subViews.
64 static NSInteger _indexOfBackdropView = -1;
65 // The index of "_UIVisualEffectSubview" in UIVisualEffectView's subViews.
66 static NSInteger _indexOfVisualEffectSubview = -1;
67 static BOOL _preparedOnce = NO;
68 
69 - (instancetype)initWithFrame:(CGRect)frame
70  blurRadius:(CGFloat)blurRadius
71  visualEffectView:(UIVisualEffectView*)visualEffectView {
72  if (self = [super init]) {
73  _frame = frame;
74  _blurRadius = blurRadius;
75  [PlatformViewFilter prepareOnce:visualEffectView];
76  if (![PlatformViewFilter isUIVisualEffectViewImplementationValid]) {
77  FML_DLOG(ERROR) << "Apple's API for UIVisualEffectView changed. Update the implementation to "
78  "access the gaussianBlur CAFilter.";
79  return nil;
80  }
81  _backdropFilterView = visualEffectView;
82  _backdropFilterViewConfigured = NO;
83  }
84  return self;
85 }
86 
88  _preparedOnce = NO;
89  _gaussianBlurFilter = nil;
92 }
93 
94 + (void)prepareOnce:(UIVisualEffectView*)visualEffectView {
95  if (_preparedOnce) {
96  return;
97  }
98  for (NSUInteger i = 0; i < visualEffectView.subviews.count; i++) {
99  UIView* view = visualEffectView.subviews[i];
100  if ([NSStringFromClass([view class]) hasSuffix:@"BackdropView"]) {
102  for (NSObject* filter in view.layer.filters) {
103  if ([[filter valueForKey:@"name"] isEqual:@"gaussianBlur"] &&
104  [[filter valueForKey:@"inputRadius"] isKindOfClass:[NSNumber class]]) {
105  _gaussianBlurFilter = filter;
106  break;
107  }
108  }
109  } else if ([NSStringFromClass([view class]) hasSuffix:@"VisualEffectSubview"]) {
111  }
112  }
113  _preparedOnce = YES;
114 }
115 
116 + (BOOL)isUIVisualEffectViewImplementationValid {
118 }
119 
120 - (UIVisualEffectView*)backdropFilterView {
121  FML_DCHECK(_backdropFilterView);
122  if (!self.backdropFilterViewConfigured) {
123  [self updateVisualEffectView:_backdropFilterView];
124  self.backdropFilterViewConfigured = YES;
125  }
126  return _backdropFilterView;
127 }
128 
129 - (void)updateVisualEffectView:(UIVisualEffectView*)visualEffectView {
130  NSObject* gaussianBlurFilter = [_gaussianBlurFilter copy];
131  FML_DCHECK(gaussianBlurFilter);
132  UIView* backdropView = visualEffectView.subviews[_indexOfBackdropView];
133  [gaussianBlurFilter setValue:@(_blurRadius) forKey:@"inputRadius"];
134  backdropView.layer.filters = @[ gaussianBlurFilter ];
135 
136  UIView* visualEffectSubview = visualEffectView.subviews[_indexOfVisualEffectSubview];
137  visualEffectSubview.layer.backgroundColor = UIColor.clearColor.CGColor;
138  visualEffectView.frame = _frame;
139 
140  self.backdropFilterView = visualEffectView;
141 }
142 
143 @end
144 
145 @interface ChildClippingView ()
146 
147 @property(nonatomic, copy) NSArray<PlatformViewFilter*>* filters;
148 @property(nonatomic) NSMutableArray<UIVisualEffectView*>* backdropFilterSubviews;
149 
150 @end
151 
152 @implementation ChildClippingView
153 
154 // The ChildClippingView's frame is the bounding rect of the platform view. we only want touches to
155 // be hit tested and consumed by this view if they are inside the embedded platform view which could
156 // be smaller the embedded platform view is rotated.
157 - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event {
158  for (UIView* view in self.subviews) {
159  if ([view pointInside:[self convertPoint:point toView:view] withEvent:event]) {
160  return YES;
161  }
162  }
163  return NO;
164 }
165 
166 - (void)applyBlurBackdropFilters:(NSArray<PlatformViewFilter*>*)filters {
167  FML_DCHECK(self.filters.count == self.backdropFilterSubviews.count);
168  if (self.filters.count == 0 && filters.count == 0) {
169  return;
170  }
171  self.filters = filters;
172  NSUInteger index = 0;
173  for (index = 0; index < self.filters.count; index++) {
174  UIVisualEffectView* backdropFilterView;
175  PlatformViewFilter* filter = self.filters[index];
176  if (self.backdropFilterSubviews.count <= index) {
177  backdropFilterView = filter.backdropFilterView;
178  [self addSubview:backdropFilterView];
179  [self.backdropFilterSubviews addObject:backdropFilterView];
180  } else {
181  [filter updateVisualEffectView:self.backdropFilterSubviews[index]];
182  }
183  }
184  for (NSUInteger i = self.backdropFilterSubviews.count; i > index; i--) {
185  [self.backdropFilterSubviews[i - 1] removeFromSuperview];
186  [self.backdropFilterSubviews removeLastObject];
187  }
188 }
189 
190 - (NSMutableArray*)backdropFilterSubviews {
191  if (!_backdropFilterSubviews) {
192  _backdropFilterSubviews = [[NSMutableArray alloc] init];
193  }
194  return _backdropFilterSubviews;
195 }
196 
197 @end
198 
200 
201 // A `CATransform3D` matrix represnts a scale transform that revese UIScreen.scale.
202 //
203 // The transform matrix passed in clipRect/clipRRect/clipPath methods are in device coordinate
204 // space. The transfrom matrix concats `reverseScreenScale` to create a transform matrix in the iOS
205 // logical coordinates (points).
206 //
207 // See https://developer.apple.com/documentation/uikit/uiscreen/1617836-scale?language=objc for
208 // information about screen scale.
209 @property(nonatomic) CATransform3D reverseScreenScale;
210 
211 - (fml::CFRef<CGPathRef>)getTransformedPath:(CGPathRef)path matrix:(CATransform3D)matrix;
212 
213 @end
214 
215 @implementation FlutterClippingMaskView {
216  std::vector<fml::CFRef<CGPathRef>> paths_;
218  CGRect rectSoFar_;
219 }
220 
221 - (instancetype)initWithFrame:(CGRect)frame screenScale:(CGFloat)screenScale {
222  if (self = [super initWithFrame:frame]) {
223  self.backgroundColor = UIColor.clearColor;
224  _reverseScreenScale = CATransform3DMakeScale(1 / screenScale, 1 / screenScale, 1);
225  rectSoFar_ = self.bounds;
227  }
228  return self;
229 }
230 
231 + (Class)layerClass {
232  return [CAShapeLayer class];
233 }
234 
235 - (CAShapeLayer*)shapeLayer {
236  return (CAShapeLayer*)self.layer;
237 }
238 
239 - (void)reset {
240  paths_.clear();
241  rectSoFar_ = self.bounds;
243  [self shapeLayer].path = nil;
244  [self setNeedsDisplay];
245 }
246 
247 // In some scenarios, when we add this view as a maskView of the ChildClippingView, iOS added
248 // this view as a subview of the ChildClippingView.
249 // This results this view blocking touch events on the ChildClippingView.
250 // So we should always ignore any touch events sent to this view.
251 // See https://github.com/flutter/flutter/issues/66044
252 - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event {
253  return NO;
254 }
255 
256 - (void)drawRect:(CGRect)rect {
257  // It's hard to compute intersection of arbitrary non-rect paths.
258  // So we fallback to software rendering.
259  if (containsNonRectPath_ && paths_.size() > 1) {
260  CGContextRef context = UIGraphicsGetCurrentContext();
261  CGContextSaveGState(context);
262 
263  // For mask view, only the alpha channel is used.
264  CGContextSetAlpha(context, 1);
265 
266  for (size_t i = 0; i < paths_.size(); i++) {
267  CGContextAddPath(context, paths_.at(i));
268  CGContextClip(context);
269  }
270  CGContextFillRect(context, rect);
271  CGContextRestoreGState(context);
272  } else {
273  // Either a single path, or multiple rect paths.
274  // Use hardware rendering with CAShapeLayer.
275  [super drawRect:rect];
276  if (![self shapeLayer].path) {
277  if (paths_.size() == 1) {
278  // A single path, either rect or non-rect.
279  [self shapeLayer].path = paths_.at(0);
280  } else {
281  // Multiple paths, all paths must be rects.
282  CGPathRef pathSoFar = CGPathCreateWithRect(rectSoFar_, nil);
283  [self shapeLayer].path = pathSoFar;
284  CGPathRelease(pathSoFar);
285  }
286  }
287  }
288 }
289 
290 - (void)clipRect:(const flutter::DlRect&)clipDlRect matrix:(const flutter::DlMatrix&)matrix {
291  CGRect clipRect = GetCGRectFromDlRect(clipDlRect);
292  CGPathRef path = CGPathCreateWithRect(clipRect, nil);
293  // The `matrix` is based on the physical pixels, convert it to UIKit points.
294  CATransform3D matrixInPoints =
295  CATransform3DConcat(GetCATransform3DFromDlMatrix(matrix), _reverseScreenScale);
296  paths_.push_back([self getTransformedPath:path matrix:matrixInPoints]);
297  CGAffineTransform affine = [self affineWithMatrix:matrixInPoints];
298  // Make sure the rect is not rotated (only translated or scaled).
299  if (affine.b == 0 && affine.c == 0) {
300  rectSoFar_ = CGRectIntersection(rectSoFar_, CGRectApplyAffineTransform(clipRect, affine));
301  } else {
302  containsNonRectPath_ = YES;
303  }
304 }
305 
306 - (void)clipRRect:(const flutter::DlRoundRect&)clipDlRRect matrix:(const flutter::DlMatrix&)matrix {
307  if (clipDlRRect.IsEmpty()) {
308  return;
309  } else if (clipDlRRect.IsRect()) {
310  [self clipRect:clipDlRRect.GetBounds() matrix:matrix];
311  return;
312  } else {
313  CGPathRef pathRef = nullptr;
314  containsNonRectPath_ = YES;
315 
316  if (clipDlRRect.GetRadii().AreAllCornersSame()) {
317  CGRect clipRect = GetCGRectFromDlRect(clipDlRRect.GetBounds());
318  auto radii = clipDlRRect.GetRadii();
319  pathRef =
320  CGPathCreateWithRoundedRect(clipRect, radii.top_left.width, radii.top_left.height, nil);
321  } else {
322  CGMutablePathRef mutablePathRef = CGPathCreateMutable();
323  // Complex types, we manually add each corner.
324  flutter::DlRect clipDlRect = clipDlRRect.GetBounds();
325  auto left = clipDlRect.GetLeft();
326  auto top = clipDlRect.GetTop();
327  auto right = clipDlRect.GetRight();
328  auto bottom = clipDlRect.GetBottom();
329  flutter::DlRoundingRadii radii = clipDlRRect.GetRadii();
330  auto& top_left = radii.top_left;
331  auto& top_right = radii.top_right;
332  auto& bottom_left = radii.bottom_left;
333  auto& bottom_right = radii.bottom_right;
334 
335  // Start drawing RRect
336  // These calculations are off, the AddCurve methods add a Bezier curve
337  // which, for round rects should be a "magic distance" from the end
338  // point of the horizontal/vertical section to the corner.
339  // Move point to the top left corner adding the top left radii's x.
340  CGPathMoveToPoint(mutablePathRef, nil, //
341  left + top_left.width, top);
342  // Move point horizontally right to the top right corner and add the top right curve.
343  CGPathAddLineToPoint(mutablePathRef, nil, //
344  right - top_right.width, top);
345  CGPathAddCurveToPoint(mutablePathRef, nil, //
346  right, top, //
347  right, top + top_right.height, //
348  right, top + top_right.height);
349  // Move point vertically down to the bottom right corner and add the bottom right curve.
350  CGPathAddLineToPoint(mutablePathRef, nil, //
351  right, bottom - bottom_right.height);
352  CGPathAddCurveToPoint(mutablePathRef, nil, //
353  right, bottom, //
354  right - bottom_right.width, bottom, //
355  right - bottom_right.width, bottom);
356  // Move point horizontally left to the bottom left corner and add the bottom left curve.
357  CGPathAddLineToPoint(mutablePathRef, nil, //
358  left + bottom_left.width, bottom);
359  CGPathAddCurveToPoint(mutablePathRef, nil, //
360  left, bottom, //
361  left, bottom - bottom_left.height, //
362  left, bottom - bottom_left.height);
363  // Move point vertically up to the top left corner and add the top left curve.
364  CGPathAddLineToPoint(mutablePathRef, nil, //
365  left, top + top_left.height);
366  CGPathAddCurveToPoint(mutablePathRef, nil, //
367  left, top, //
368  left + top_left.width, top, //
369  left + top_left.width, top);
370  CGPathCloseSubpath(mutablePathRef);
371  pathRef = mutablePathRef;
372  }
373  // The `matrix` is based on the physical pixels, convert it to UIKit points.
374  CATransform3D matrixInPoints =
375  CATransform3DConcat(GetCATransform3DFromDlMatrix(matrix), _reverseScreenScale);
376  // TODO(cyanglaz): iOS does not seem to support hard edge on CAShapeLayer. It clearly stated
377  // that the CAShaperLayer will be drawn antialiased. Need to figure out a way to do the hard
378  // edge clipping on iOS.
379  paths_.push_back([self getTransformedPath:pathRef matrix:matrixInPoints]);
380  }
381 }
382 
383 - (void)clipPath:(const flutter::DlPath&)dlPath matrix:(const flutter::DlMatrix&)matrix {
384  containsNonRectPath_ = YES;
385  CGMutablePathRef pathRef = CGPathCreateMutable();
386  bool subpath_needs_close = false;
387  std::optional<flutter::DlPoint> pending_moveto;
388 
389  auto resolve_moveto = [&pending_moveto, &pathRef]() {
390  if (pending_moveto.has_value()) {
391  CGPathMoveToPoint(pathRef, nil, pending_moveto->x, pending_moveto->y);
392  pending_moveto.reset();
393  }
394  };
395 
396  auto& path = dlPath.GetPath();
397  for (auto it = path.begin(), end = path.end(); it != end; ++it) {
398  switch (it.type()) {
399  case impeller::Path::ComponentType::kContour: {
400  const impeller::ContourComponent* contour = it.contour();
401  FML_DCHECK(contour != nullptr);
402  if (subpath_needs_close) {
403  CGPathCloseSubpath(pathRef);
404  }
405  pending_moveto = contour->destination;
406  subpath_needs_close = contour->IsClosed();
407  break;
408  }
409  case impeller::Path::ComponentType::kLinear: {
410  const impeller::LinearPathComponent* linear = it.linear();
411  FML_DCHECK(linear != nullptr);
412  resolve_moveto();
413  CGPathAddLineToPoint(pathRef, nil, linear->p2.x, linear->p2.y);
414  break;
415  }
416  case impeller::Path::ComponentType::kQuadratic: {
417  const impeller::QuadraticPathComponent* quadratic = it.quadratic();
418  FML_DCHECK(quadratic != nullptr);
419  resolve_moveto();
420  CGPathAddQuadCurveToPoint(pathRef, nil, //
421  quadratic->cp.x, quadratic->cp.y, //
422  quadratic->p2.x, quadratic->p2.y);
423  break;
424  }
425  case impeller::Path::ComponentType::kConic: {
426  const impeller::ConicPathComponent* conic = it.conic();
427  FML_DCHECK(conic != nullptr);
428  resolve_moveto();
429  // Conic is not available in quartz, we use quad to approximate.
430  // TODO(cyanglaz): Better approximate the conic path.
431  // https://github.com/flutter/flutter/issues/35062
432  CGPathAddQuadCurveToPoint(pathRef, nil, //
433  conic->cp.x, conic->cp.y, //
434  conic->p2.x, conic->p2.y);
435  break;
436  }
437  case impeller::Path::ComponentType::kCubic: {
438  const impeller::CubicPathComponent* cubic = it.cubic();
439  FML_DCHECK(cubic != nullptr);
440  resolve_moveto();
441  CGPathAddCurveToPoint(pathRef, nil, //
442  cubic->cp1.x, cubic->cp1.y, //
443  cubic->cp2.x, cubic->cp2.y, //
444  cubic->p2.x, cubic->p2.y);
445  break;
446  }
447  }
448  }
449  if (subpath_needs_close) {
450  CGPathCloseSubpath(pathRef);
451  }
452  // The `matrix` is based on the physical pixels, convert it to UIKit points.
453  CATransform3D matrixInPoints =
454  CATransform3DConcat(GetCATransform3DFromDlMatrix(matrix), _reverseScreenScale);
455  paths_.push_back([self getTransformedPath:pathRef matrix:matrixInPoints]);
456 }
457 
458 - (CGAffineTransform)affineWithMatrix:(CATransform3D)matrix {
459  return CGAffineTransformMake(matrix.m11, matrix.m12, matrix.m21, matrix.m22, matrix.m41,
460  matrix.m42);
461 }
462 
463 - (fml::CFRef<CGPathRef>)getTransformedPath:(CGPathRef)path matrix:(CATransform3D)matrix {
464  CGAffineTransform affine = [self affineWithMatrix:matrix];
465  CGPathRef transformedPath = CGPathCreateCopyByTransformingPath(path, &affine);
466 
467  CGPathRelease(path);
468  return fml::CFRef<CGPathRef>(transformedPath);
469 }
470 
471 @end
472 
474 
475 // The maximum number of `FlutterClippingMaskView` the pool can contain.
476 // This prevents the pool to grow infinately and limits the maximum memory a pool can use.
477 @property(nonatomic) NSUInteger capacity;
478 
479 // The pool contains the views that are available to use.
480 // The number of items in the pool must not excceds `capacity`.
481 @property(nonatomic) NSMutableSet<FlutterClippingMaskView*>* pool;
482 
483 @end
484 
485 @implementation FlutterClippingMaskViewPool : NSObject
486 
487 - (instancetype)initWithCapacity:(NSInteger)capacity {
488  if (self = [super init]) {
489  // Most of cases, there are only one PlatformView in the scene.
490  // Thus init with the capacity of 1.
491  _pool = [[NSMutableSet alloc] initWithCapacity:1];
492  _capacity = capacity;
493  }
494  return self;
495 }
496 
497 - (FlutterClippingMaskView*)getMaskViewWithFrame:(CGRect)frame screenScale:(CGFloat)screenScale {
498  FML_DCHECK(self.pool.count <= self.capacity);
499  if (self.pool.count == 0) {
500  // The pool is empty, alloc a new one.
501  return [[FlutterClippingMaskView alloc] initWithFrame:frame screenScale:screenScale];
502  }
503  FlutterClippingMaskView* maskView = [self.pool anyObject];
504  maskView.frame = frame;
505  [maskView reset];
506  [self.pool removeObject:maskView];
507  return maskView;
508 }
509 
510 - (void)insertViewToPoolIfNeeded:(FlutterClippingMaskView*)maskView {
511  FML_DCHECK(![self.pool containsObject:maskView]);
512  FML_DCHECK(self.pool.count <= self.capacity);
513  if (self.pool.count == self.capacity) {
514  return;
515  }
516  [self.pool addObject:maskView];
517 }
518 
519 @end
520 
521 @implementation UIView (FirstResponder)
523  if (self.isFirstResponder) {
524  return YES;
525  }
526  for (UIView* subview in self.subviews) {
527  if (subview.flt_hasFirstResponderInViewHierarchySubtree) {
528  return YES;
529  }
530  }
531  return NO;
532 }
533 @end
534 
536 @property(nonatomic, weak, readonly) UIView* embeddedView;
537 @property(nonatomic, readonly) FlutterDelayingGestureRecognizer* delayingRecognizer;
538 @property(nonatomic, readonly) FlutterPlatformViewGestureRecognizersBlockingPolicy blockingPolicy;
539 @end
540 
542 - (instancetype)initWithEmbeddedView:(UIView*)embeddedView
543  platformViewsController:(FlutterPlatformViewsController*)platformViewsController
544  gestureRecognizersBlockingPolicy:
546  self = [super initWithFrame:embeddedView.frame];
547  if (self) {
548  self.multipleTouchEnabled = YES;
549  _embeddedView = embeddedView;
550  embeddedView.autoresizingMask =
551  (UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight);
552 
553  [self addSubview:embeddedView];
554 
555  ForwardingGestureRecognizer* forwardingRecognizer =
556  [[ForwardingGestureRecognizer alloc] initWithTarget:self
557  platformViewsController:platformViewsController];
558 
559  _delayingRecognizer =
560  [[FlutterDelayingGestureRecognizer alloc] initWithTarget:self
561  action:nil
562  forwardingRecognizer:forwardingRecognizer];
563  _blockingPolicy = blockingPolicy;
564 
565  [self addGestureRecognizer:_delayingRecognizer];
566  [self addGestureRecognizer:forwardingRecognizer];
567  }
568  return self;
569 }
570 
571 - (void)forceResetForwardingGestureRecognizerState {
572  // When iPad pencil is involved in a finger touch gesture, the gesture is not reset to "possible"
573  // state and is stuck on "failed" state, which causes subsequent touches to be blocked. As a
574  // workaround, we force reset the state by recreating the forwarding gesture recognizer. See:
575  // https://github.com/flutter/flutter/issues/136244
576  ForwardingGestureRecognizer* oldForwardingRecognizer =
577  (ForwardingGestureRecognizer*)self.delayingRecognizer.forwardingRecognizer;
578  ForwardingGestureRecognizer* newForwardingRecognizer =
579  [oldForwardingRecognizer recreateRecognizerWithTarget:self];
580  self.delayingRecognizer.forwardingRecognizer = newForwardingRecognizer;
581  [self removeGestureRecognizer:oldForwardingRecognizer];
582  [self addGestureRecognizer:newForwardingRecognizer];
583 }
584 
585 - (void)releaseGesture {
586  self.delayingRecognizer.state = UIGestureRecognizerStateFailed;
587 }
588 
589 - (BOOL)containsWebView:(UIView*)view remainingSubviewDepth:(int)remainingSubviewDepth {
590  if (remainingSubviewDepth < 0) {
591  return NO;
592  }
593  if ([view isKindOfClass:[WKWebView class]]) {
594  return YES;
595  }
596  for (UIView* subview in view.subviews) {
597  if ([self containsWebView:subview remainingSubviewDepth:remainingSubviewDepth - 1]) {
598  return YES;
599  }
600  }
601  return NO;
602 }
603 
604 - (void)blockGesture {
605  switch (_blockingPolicy) {
607  // We block all other gesture recognizers immediately in this policy.
608  self.delayingRecognizer.state = UIGestureRecognizerStateEnded;
609 
610  // On iOS 18.2, WKWebView's internal recognizer likely caches the old state of its blocking
611  // recognizers (i.e. delaying recognizer), resulting in non-tappable links. See
612  // https://github.com/flutter/flutter/issues/158961. Removing and adding back the delaying
613  // recognizer solves the problem, possibly because UIKit notifies all the recognizers related
614  // to (blocking or blocked by) this recognizer. It is not possible to inject this workaround
615  // from the web view plugin level. Right now we only observe this issue for
616  // FlutterPlatformViewGestureRecognizersBlockingPolicyEager, but we should try it if a similar
617  // issue arises for the other policy.
618  if (@available(iOS 18.2, *)) {
619  // This workaround is designed for WKWebView only. The 1P web view plugin provides a
620  // WKWebView itself as the platform view. However, some 3P plugins provide wrappers of
621  // WKWebView instead. So we perform DFS to search the view hierarchy (with a depth limit).
622  // Passing a limit of 0 means only searching for platform view itself; Pass 1 to include its
623  // children as well, and so on. We should be conservative and start with a small number. The
624  // AdMob banner has a WKWebView at depth 7.
625  if ([self containsWebView:self.embeddedView remainingSubviewDepth:1]) {
626  [self removeGestureRecognizer:self.delayingRecognizer];
627  [self addGestureRecognizer:self.delayingRecognizer];
628  }
629  }
630 
631  break;
633  if (self.delayingRecognizer.touchedEndedWithoutBlocking) {
634  // If touchesEnded of the `DelayingGesureRecognizer` has been already invoked,
635  // we want to set the state of the `DelayingGesureRecognizer` to
636  // `UIGestureRecognizerStateEnded` as soon as possible.
637  self.delayingRecognizer.state = UIGestureRecognizerStateEnded;
638  } else {
639  // If touchesEnded of the `DelayingGesureRecognizer` has not been invoked,
640  // We will set a flag to notify the `DelayingGesureRecognizer` to set the state to
641  // `UIGestureRecognizerStateEnded` when touchesEnded is called.
642  self.delayingRecognizer.shouldEndInNextTouchesEnded = YES;
643  }
644  break;
645  default:
646  break;
647  }
648 }
649 
650 // We want the intercepting view to consume the touches and not pass the touches up to the parent
651 // view. Make the touch event method not call super will not pass the touches up to the parent view.
652 // Hence we overide the touch event methods and do nothing.
653 - (void)touchesBegan:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
654 }
655 
656 - (void)touchesMoved:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
657 }
658 
659 - (void)touchesCancelled:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
660 }
661 
662 - (void)touchesEnded:(NSSet*)touches withEvent:(UIEvent*)event {
663 }
664 
666  return self.flutterAccessibilityContainer;
667 }
668 
669 @end
670 
672 
673 - (instancetype)initWithTarget:(id)target
674  action:(SEL)action
675  forwardingRecognizer:(UIGestureRecognizer*)forwardingRecognizer {
676  self = [super initWithTarget:target action:action];
677  if (self) {
678  self.delaysTouchesBegan = YES;
679  self.delaysTouchesEnded = YES;
680  self.delegate = self;
681  _shouldEndInNextTouchesEnded = NO;
682  _touchedEndedWithoutBlocking = NO;
683  _forwardingRecognizer = forwardingRecognizer;
684  }
685  return self;
686 }
687 
688 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
689  shouldBeRequiredToFailByGestureRecognizer:(UIGestureRecognizer*)otherGestureRecognizer {
690  // The forwarding gesture recognizer should always get all touch events, so it should not be
691  // required to fail by any other gesture recognizer.
692  return otherGestureRecognizer != _forwardingRecognizer && otherGestureRecognizer != self;
693 }
694 
695 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
696  shouldRequireFailureOfGestureRecognizer:(UIGestureRecognizer*)otherGestureRecognizer {
697  return otherGestureRecognizer == self;
698 }
699 
700 - (void)touchesBegan:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
701  self.touchedEndedWithoutBlocking = NO;
702  [super touchesBegan:touches withEvent:event];
703 }
704 
705 - (void)touchesEnded:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
706  if (self.shouldEndInNextTouchesEnded) {
707  self.state = UIGestureRecognizerStateEnded;
708  self.shouldEndInNextTouchesEnded = NO;
709  } else {
710  self.touchedEndedWithoutBlocking = YES;
711  }
712  [super touchesEnded:touches withEvent:event];
713 }
714 
715 - (void)touchesCancelled:(NSSet*)touches withEvent:(UIEvent*)event {
716  self.state = UIGestureRecognizerStateFailed;
717 }
718 @end
719 
721  // Weak reference to PlatformViewsController. The PlatformViewsController has
722  // a reference to the FlutterViewController, where we can dispatch pointer events to.
723  //
724  // The lifecycle of PlatformViewsController is bind to FlutterEngine, which should always
725  // outlives the FlutterViewController. And ForwardingGestureRecognizer is owned by a subview of
726  // FlutterView, so the ForwardingGestureRecognizer never out lives FlutterViewController.
727  // Therefore, `_platformViewsController` should never be nullptr.
728  __weak FlutterPlatformViewsController* _platformViewsController;
729  // Counting the pointers that has started in one touch sequence.
730  NSInteger _currentTouchPointersCount;
731  // We can't dispatch events to the framework without this back pointer.
732  // This gesture recognizer retains the `FlutterViewController` until the
733  // end of a gesture sequence, that is all the touches in touchesBegan are concluded
734  // with |touchesCancelled| or |touchesEnded|.
735  UIViewController<FlutterViewResponder>* _flutterViewController;
736 }
737 
738 - (instancetype)initWithTarget:(id)target
739  platformViewsController:(FlutterPlatformViewsController*)platformViewsController {
740  self = [super initWithTarget:target action:nil];
741  if (self) {
742  self.delegate = self;
743  FML_DCHECK(platformViewsController);
744  _platformViewsController = platformViewsController;
746  }
747  return self;
748 }
749 
750 - (ForwardingGestureRecognizer*)recreateRecognizerWithTarget:(id)target {
751  return [[ForwardingGestureRecognizer alloc] initWithTarget:target
752  platformViewsController:_platformViewsController];
753 }
754 
755 - (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent*)event {
756  FML_DCHECK(_currentTouchPointersCount >= 0);
757  if (_currentTouchPointersCount == 0) {
758  // At the start of each gesture sequence, we reset the `_flutterViewController`,
759  // so that all the touch events in the same sequence are forwarded to the same
760  // `_flutterViewController`.
761  _flutterViewController = _platformViewsController.flutterViewController;
762  }
763  [_flutterViewController touchesBegan:touches withEvent:event];
764  _currentTouchPointersCount += touches.count;
765 }
766 
767 - (void)touchesMoved:(NSSet*)touches withEvent:(UIEvent*)event {
768  [_flutterViewController touchesMoved:touches withEvent:event];
769 }
770 
771 - (void)touchesEnded:(NSSet*)touches withEvent:(UIEvent*)event {
772  [_flutterViewController touchesEnded:touches withEvent:event];
773  _currentTouchPointersCount -= touches.count;
774  // Touches in one touch sequence are sent to the touchesEnded method separately if different
775  // fingers stop touching the screen at different time. So one touchesEnded method triggering does
776  // not necessarially mean the touch sequence has ended. We Only set the state to
777  // UIGestureRecognizerStateFailed when all the touches in the current touch sequence is ended.
778  if (_currentTouchPointersCount == 0) {
779  self.state = UIGestureRecognizerStateFailed;
781  [self forceResetStateIfNeeded];
782  }
783 }
784 
785 - (void)touchesCancelled:(NSSet*)touches withEvent:(UIEvent*)event {
786  // In the event of platform view is removed, iOS generates a "stationary" change type instead of
787  // "cancelled" change type.
788  // Flutter needs all the cancelled touches to be "cancelled" change types in order to correctly
789  // handle gesture sequence.
790  // We always override the change type to "cancelled".
791  [_flutterViewController forceTouchesCancelled:touches];
792  _currentTouchPointersCount -= touches.count;
793  if (_currentTouchPointersCount == 0) {
794  self.state = UIGestureRecognizerStateFailed;
796  [self forceResetStateIfNeeded];
797  }
798 }
799 
800 - (void)forceResetStateIfNeeded {
801  __weak ForwardingGestureRecognizer* weakSelf = self;
802  dispatch_async(dispatch_get_main_queue(), ^{
803  ForwardingGestureRecognizer* strongSelf = weakSelf;
804  if (!strongSelf) {
805  return;
806  }
807  if (strongSelf.state != UIGestureRecognizerStatePossible) {
808  [(FlutterTouchInterceptingView*)strongSelf.view forceResetForwardingGestureRecognizerState];
809  }
810  });
811 }
812 
813 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
814  shouldRecognizeSimultaneouslyWithGestureRecognizer:
815  (UIGestureRecognizer*)otherGestureRecognizer {
816  return YES;
817 }
818 @end
UIView(FirstResponder)::flt_hasFirstResponderInViewHierarchySubtree
BOOL flt_hasFirstResponderInViewHierarchySubtree
Definition: FlutterPlatformViews_Internal.h:155
FlutterDelayingGestureRecognizer::forwardingRecognizer
UIGestureRecognizer * forwardingRecognizer
Definition: FlutterPlatformViews_Internal.h:174
PlatformViewFilter::backdropFilterView
UIVisualEffectView * backdropFilterView
Definition: FlutterPlatformViews_Internal.h:95
FlutterPlatformViewGestureRecognizersBlockingPolicyWaitUntilTouchesEnded
@ FlutterPlatformViewGestureRecognizersBlockingPolicyWaitUntilTouchesEnded
Definition: FlutterPlugin.h:269
rectSoFar_
CGRect rectSoFar_
Definition: FlutterPlatformViews.mm:218
ChildClippingView
Definition: FlutterPlatformViews.mm:152
-[FlutterTouchInterceptingView blockGesture]
void blockGesture()
Definition: FlutterPlatformViews.mm:604
-[FlutterClippingMaskView reset]
void reset()
Definition: FlutterPlatformViews.mm:239
-[FlutterTouchInterceptingView releaseGesture]
void releaseGesture()
Definition: FlutterPlatformViews.mm:585
FlutterClippingMaskView
Definition: FlutterPlatformViews.mm:215
_currentTouchPointersCount
NSInteger _currentTouchPointersCount
Definition: FlutterPlatformViews.mm:720
PlatformViewFilter
Definition: FlutterPlatformViews.mm:60
_flutterViewController
UIViewController< FlutterViewResponder > * _flutterViewController
Definition: FlutterPlatformViews.mm:735
-[FlutterTouchInterceptingView(Tests) accessibilityContainer]
id accessibilityContainer()
initWithFrame
instancetype initWithFrame
Definition: FlutterTextInputPlugin.h:172
+[PlatformViewFilter resetPreparation]
void resetPreparation()
Definition: FlutterPlatformViews.mm:87
PlatformViewFilter::frame
CGRect frame
Definition: FlutterPlatformViews_Internal.h:84
FlutterDelayingGestureRecognizer::shouldEndInNextTouchesEnded
BOOL shouldEndInNextTouchesEnded
Definition: FlutterPlatformViews_Internal.h:168
FlutterClippingMaskViewPool
Definition: FlutterPlatformViews.mm:485
ios_surface.h
GetCGRectFromDlRect
static CGRect GetCGRectFromDlRect(const DlRect &clipDlRect)
Definition: FlutterPlatformViewsController.mm:79
-[ChildClippingView backdropFilterSubviews]
NSMutableArray * backdropFilterSubviews()
Definition: FlutterPlatformViews.mm:190
_indexOfVisualEffectSubview
static NSInteger _indexOfVisualEffectSubview
Definition: FlutterPlatformViews.mm:66
ForwardingGestureRecognizer
Definition: FlutterPlatformViews.mm:720
-[FlutterClippingMaskView clipRect:matrix:]
void clipRect:matrix:(const flutter::DlRect &clipDlRect,[matrix] const flutter::DlMatrix &matrix)
Definition: FlutterPlatformViews.mm:290
FlutterDelayingGestureRecognizer
Definition: FlutterPlatformViews.mm:671
FlutterPlatformViewGestureRecognizersBlockingPolicyEager
@ FlutterPlatformViewGestureRecognizersBlockingPolicyEager
Definition: FlutterPlugin.h:261
flutter
Definition: accessibility_bridge.h:26
containsNonRectPath_
BOOL containsNonRectPath_
Definition: FlutterPlatformViews.mm:215
FlutterPlatformViews_Internal.h
fml
Definition: profiler_metrics_ios.mm:41
FlutterPlatformViewGestureRecognizersBlockingPolicy
FlutterPlatformViewGestureRecognizersBlockingPolicy
Definition: FlutterPlugin.h:252
GetCATransform3DFromDlMatrix
static CATransform3D GetCATransform3DFromDlMatrix(const DlMatrix &matrix)
Definition: FlutterPlatformViewsController.mm:46
PlatformViewFilter::blurRadius
CGFloat blurRadius
Definition: FlutterPlatformViews_Internal.h:89
UIView(FirstResponder)
Definition: FlutterPlatformViews.mm:521
FlutterPlatformViewsController
Definition: FlutterPlatformViewsController.h:30
FlutterTouchInterceptingView
Definition: FlutterPlatformViews.mm:541
_gaussianBlurFilter
static NSObject * _gaussianBlurFilter
Definition: FlutterPlatformViews.mm:62
-[FlutterTouchInterceptingView embeddedView]
UIView * embeddedView()
_indexOfBackdropView
static NSInteger _indexOfBackdropView
Definition: FlutterPlatformViews.mm:64
_preparedOnce
static BOOL _preparedOnce
Definition: FlutterPlatformViews.mm:67
FLUTTER_ASSERT_ARC
Definition: FlutterChannelKeyResponder.mm:13