7 #import <WebKit/WebKit.h>
9 #include "flutter/display_list/effects/dl_image_filter.h"
10 #include "flutter/fml/platform/darwin/cf_utils.h"
17 return CGRectMake(clipDlRect.GetX(),
19 clipDlRect.GetWidth(),
20 clipDlRect.GetHeight());
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];
30 transform.m21 = matrix.m[4];
31 transform.m22 = matrix.m[5];
32 transform.m23 = matrix.m[6];
33 transform.m24 = matrix.m[7];
35 transform.m31 = matrix.m[8];
36 transform.m32 = matrix.m[9];
37 transform.m33 = matrix.m[10];
38 transform.m34 = matrix.m[11];
40 transform.m41 = matrix.m[12];
41 transform.m42 = matrix.m[13];
42 transform.m43 = matrix.m[14];
43 transform.m44 = matrix.m[15];
51 @property(nonatomic) BOOL backdropFilterViewConfigured;
56 - (void)updateVisualEffectView:(UIVisualEffectView*)visualEffectView;
70 blurRadius:(CGFloat)blurRadius
71 visualEffectView:(UIVisualEffectView*)visualEffectView {
72 if (
self = [super init]) {
77 FML_DLOG(ERROR) <<
"Apple's API for UIVisualEffectView changed. Update the implementation to "
78 "access the gaussianBlur CAFilter.";
81 _backdropFilterView = visualEffectView;
82 _backdropFilterViewConfigured = NO;
94 + (void)prepareOnce:(UIVisualEffectView*)visualEffectView {
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;
109 }
else if ([NSStringFromClass([view
class]) hasSuffix:
@"VisualEffectSubview"]) {
116 + (BOOL)isUIVisualEffectViewImplementationValid {
121 FML_DCHECK(_backdropFilterView);
122 if (!
self.backdropFilterViewConfigured) {
123 [
self updateVisualEffectView:_backdropFilterView];
124 self.backdropFilterViewConfigured = YES;
126 return _backdropFilterView;
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 ];
136 UIView* visualEffectSubview = visualEffectView.subviews[_indexOfVisualEffectSubview];
137 visualEffectSubview.layer.backgroundColor = UIColor.clearColor.CGColor;
138 visualEffectView.frame = _frame;
140 self.backdropFilterView = visualEffectView;
147 @property(nonatomic, copy) NSArray<PlatformViewFilter*>* filters;
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]) {
167 FML_DCHECK(
self.filters.count ==
self.backdropFilterSubviews.count);
168 if (
self.filters.count == 0 && filters.count == 0) {
171 self.filters = filters;
172 NSUInteger index = 0;
173 for (index = 0; index <
self.filters.count; index++) {
174 UIVisualEffectView* backdropFilterView;
178 [
self addSubview:backdropFilterView];
179 [
self.backdropFilterSubviews addObject:backdropFilterView];
181 [filter updateVisualEffectView:self.backdropFilterSubviews[index]];
185 [
self.backdropFilterSubviews[i - 1] removeFromSuperview];
186 [
self.backdropFilterSubviews removeLastObject];
191 if (!_backdropFilterSubviews) {
192 _backdropFilterSubviews = [[NSMutableArray alloc] init];
194 return _backdropFilterSubviews;
209 @property(nonatomic) CATransform3D reverseScreenScale;
211 - (
fml::CFRef<CGPathRef>)getTransformedPath:(CGPathRef)path matrix:(CATransform3D)matrix;
216 std::vector<fml::CFRef<CGPathRef>> paths_;
221 - (instancetype)
initWithFrame:(CGRect)frame screenScale:(CGFloat)screenScale {
223 self.backgroundColor = UIColor.clearColor;
224 _reverseScreenScale = CATransform3DMakeScale(1 / screenScale, 1 / screenScale, 1);
231 + (Class)layerClass {
232 return [CAShapeLayer class];
235 - (CAShapeLayer*)shapeLayer {
236 return (CAShapeLayer*)
self.layer;
243 [
self shapeLayer].path = nil;
244 [
self setNeedsDisplay];
252 - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event {
256 - (void)drawRect:(CGRect)rect {
260 CGContextRef context = UIGraphicsGetCurrentContext();
261 CGContextSaveGState(context);
264 CGContextSetAlpha(context, 1);
266 for (
size_t i = 0; i < paths_.size(); i++) {
267 CGContextAddPath(context, paths_.at(i));
268 CGContextClip(context);
270 CGContextFillRect(context, rect);
271 CGContextRestoreGState(context);
275 [
super drawRect:rect];
276 if (![
self shapeLayer].path) {
277 if (paths_.size() == 1) {
279 [
self shapeLayer].path = paths_.at(0);
282 CGPathRef pathSoFar = CGPathCreateWithRect(
rectSoFar_, nil);
283 [
self shapeLayer].path = pathSoFar;
284 CGPathRelease(pathSoFar);
290 - (void)clipRect:(const
flutter::DlRect&)clipDlRect matrix:(const
flutter::DlMatrix&)matrix {
292 CGPathRef path = CGPathCreateWithRect(clipRect, nil);
294 CATransform3D matrixInPoints =
296 paths_.push_back([
self getTransformedPath:path matrix:matrixInPoints]);
297 CGAffineTransform affine = [
self affineWithMatrix:matrixInPoints];
299 if (affine.b == 0 && affine.c == 0) {
306 - (void)clipRRect:(const
flutter::DlRoundRect&)clipDlRRect matrix:(const
flutter::DlMatrix&)matrix {
307 if (clipDlRRect.IsEmpty()) {
309 }
else if (clipDlRRect.IsRect()) {
313 CGPathRef pathRef =
nullptr;
316 if (clipDlRRect.GetRadii().AreAllCornersSame()) {
318 auto radii = clipDlRRect.GetRadii();
320 CGPathCreateWithRoundedRect(clipRect, radii.top_left.width, radii.top_left.height, nil);
322 CGMutablePathRef mutablePathRef = CGPathCreateMutable();
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;
340 CGPathMoveToPoint(mutablePathRef, nil,
341 left + top_left.width, top);
343 CGPathAddLineToPoint(mutablePathRef, nil,
344 right - top_right.width, top);
345 CGPathAddCurveToPoint(mutablePathRef, nil,
347 right, top + top_right.height,
348 right, top + top_right.height);
350 CGPathAddLineToPoint(mutablePathRef, nil,
351 right, bottom - bottom_right.height);
352 CGPathAddCurveToPoint(mutablePathRef, nil,
354 right - bottom_right.width, bottom,
355 right - bottom_right.width, bottom);
357 CGPathAddLineToPoint(mutablePathRef, nil,
358 left + bottom_left.width, bottom);
359 CGPathAddCurveToPoint(mutablePathRef, nil,
361 left, bottom - bottom_left.height,
362 left, bottom - bottom_left.height);
364 CGPathAddLineToPoint(mutablePathRef, nil,
365 left, top + top_left.height);
366 CGPathAddCurveToPoint(mutablePathRef, nil,
368 left + top_left.width, top,
369 left + top_left.width, top);
370 CGPathCloseSubpath(mutablePathRef);
371 pathRef = mutablePathRef;
374 CATransform3D matrixInPoints =
379 paths_.push_back([
self getTransformedPath:pathRef matrix:matrixInPoints]);
383 - (void)clipPath:(const
flutter::DlPath&)dlPath matrix:(const
flutter::DlMatrix&)matrix {
385 CGMutablePathRef pathRef = CGPathCreateMutable();
386 bool subpath_needs_close =
false;
387 std::optional<flutter::DlPoint> pending_moveto;
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();
396 auto& path = dlPath.GetPath();
397 for (
auto it = path.begin(), end = path.end(); it != end; ++it) {
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);
405 pending_moveto = contour->destination;
406 subpath_needs_close = contour->IsClosed();
409 case impeller::Path::ComponentType::kLinear: {
410 const impeller::LinearPathComponent* linear = it.linear();
411 FML_DCHECK(linear !=
nullptr);
413 CGPathAddLineToPoint(pathRef, nil, linear->p2.x, linear->p2.y);
416 case impeller::Path::ComponentType::kQuadratic: {
417 const impeller::QuadraticPathComponent* quadratic = it.quadratic();
418 FML_DCHECK(quadratic !=
nullptr);
420 CGPathAddQuadCurveToPoint(pathRef, nil,
421 quadratic->cp.x, quadratic->cp.y,
422 quadratic->p2.x, quadratic->p2.y);
425 case impeller::Path::ComponentType::kConic: {
426 const impeller::ConicPathComponent* conic = it.conic();
427 FML_DCHECK(conic !=
nullptr);
432 CGPathAddQuadCurveToPoint(pathRef, nil,
433 conic->cp.x, conic->cp.y,
434 conic->p2.x, conic->p2.y);
437 case impeller::Path::ComponentType::kCubic: {
438 const impeller::CubicPathComponent* cubic = it.cubic();
439 FML_DCHECK(cubic !=
nullptr);
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);
449 if (subpath_needs_close) {
450 CGPathCloseSubpath(pathRef);
453 CATransform3D matrixInPoints =
455 paths_.push_back([
self getTransformedPath:pathRef matrix:matrixInPoints]);
458 - (CGAffineTransform)affineWithMatrix:(CATransform3D)matrix {
459 return CGAffineTransformMake(matrix.m11, matrix.m12, matrix.m21, matrix.m22, matrix.m41,
463 - (
fml::CFRef<CGPathRef>)getTransformedPath:(CGPathRef)path matrix:(CATransform3D)matrix {
464 CGAffineTransform affine = [
self affineWithMatrix:matrix];
465 CGPathRef transformedPath = CGPathCreateCopyByTransformingPath(path, &affine);
468 return fml::CFRef<CGPathRef>(transformedPath);
477 @property(nonatomic) NSUInteger capacity;
481 @property(nonatomic) NSMutableSet<FlutterClippingMaskView*>* pool;
487 - (instancetype)initWithCapacity:(NSInteger)capacity {
488 if (
self = [super init]) {
491 _pool = [[NSMutableSet alloc] initWithCapacity:1];
492 _capacity = capacity;
498 FML_DCHECK(
self.pool.count <=
self.capacity);
499 if (
self.pool.count == 0) {
504 maskView.frame = frame;
506 [
self.pool removeObject:maskView];
511 FML_DCHECK(![
self.pool containsObject:maskView]);
512 FML_DCHECK(
self.pool.count <=
self.capacity);
513 if (
self.pool.count ==
self.capacity) {
516 [
self.pool addObject:maskView];
523 if (
self.isFirstResponder) {
526 for (UIView* subview in
self.subviews) {
527 if (subview.flt_hasFirstResponderInViewHierarchySubtree) {
542 - (instancetype)initWithEmbeddedView:(UIView*)embeddedView
544 gestureRecognizersBlockingPolicy:
546 self = [
super initWithFrame:embeddedView.frame];
548 self.multipleTouchEnabled = YES;
551 (UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight);
553 [
self addSubview:embeddedView];
557 platformViewsController:platformViewsController];
559 _delayingRecognizer =
562 forwardingRecognizer:forwardingRecognizer];
563 _blockingPolicy = blockingPolicy;
565 [
self addGestureRecognizer:_delayingRecognizer];
566 [
self addGestureRecognizer:forwardingRecognizer];
571 - (void)forceResetForwardingGestureRecognizerState {
579 [oldForwardingRecognizer recreateRecognizerWithTarget:
self];
580 self.delayingRecognizer.forwardingRecognizer = newForwardingRecognizer;
581 [
self removeGestureRecognizer:oldForwardingRecognizer];
582 [
self addGestureRecognizer:newForwardingRecognizer];
586 self.delayingRecognizer.state = UIGestureRecognizerStateFailed;
589 - (BOOL)containsWebView:(UIView*)view remainingSubviewDepth:(
int)remainingSubviewDepth {
590 if (remainingSubviewDepth < 0) {
593 if ([view isKindOfClass:[WKWebView
class]]) {
596 for (UIView* subview in view.subviews) {
597 if ([
self containsWebView:subview remainingSubviewDepth:remainingSubviewDepth - 1]) {
605 switch (_blockingPolicy) {
608 self.delayingRecognizer.state = UIGestureRecognizerStateEnded;
618 if (@available(iOS 18.2, *)) {
625 if ([
self containsWebView:
self.
embeddedView remainingSubviewDepth:1]) {
626 [
self removeGestureRecognizer:self.delayingRecognizer];
627 [
self addGestureRecognizer:self.delayingRecognizer];
633 if (
self.delayingRecognizer.touchedEndedWithoutBlocking) {
637 self.delayingRecognizer.state = UIGestureRecognizerStateEnded;
642 self.delayingRecognizer.shouldEndInNextTouchesEnded = YES;
653 - (void)touchesBegan:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
656 - (void)touchesMoved:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
659 - (void)touchesCancelled:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
662 - (void)touchesEnded:(NSSet*)touches withEvent:(UIEvent*)event {
666 return self.flutterAccessibilityContainer;
673 - (instancetype)initWithTarget:(
id)target
675 forwardingRecognizer:(UIGestureRecognizer*)forwardingRecognizer {
676 self = [
super initWithTarget:target action:action];
678 self.delaysTouchesBegan = YES;
679 self.delaysTouchesEnded = YES;
680 self.delegate =
self;
681 _shouldEndInNextTouchesEnded = NO;
682 _touchedEndedWithoutBlocking = NO;
688 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
689 shouldBeRequiredToFailByGestureRecognizer:(UIGestureRecognizer*)otherGestureRecognizer {
692 return otherGestureRecognizer != _forwardingRecognizer && otherGestureRecognizer !=
self;
695 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
696 shouldRequireFailureOfGestureRecognizer:(UIGestureRecognizer*)otherGestureRecognizer {
697 return otherGestureRecognizer ==
self;
700 - (void)touchesBegan:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
701 self.touchedEndedWithoutBlocking = NO;
702 [
super touchesBegan:touches withEvent:event];
705 - (void)touchesEnded:(NSSet<UITouch*>*)touches withEvent:(UIEvent*)event {
707 self.state = UIGestureRecognizerStateEnded;
708 self.shouldEndInNextTouchesEnded = NO;
710 self.touchedEndedWithoutBlocking = YES;
712 [
super touchesEnded:touches withEvent:event];
715 - (void)touchesCancelled:(NSSet*)touches withEvent:(UIEvent*)event {
716 self.state = UIGestureRecognizerStateFailed;
738 - (instancetype)initWithTarget:(
id)target
740 self = [
super initWithTarget:target action:nil];
742 self.delegate =
self;
743 FML_DCHECK(platformViewsController);
744 _platformViewsController = platformViewsController;
752 platformViewsController:_platformViewsController];
755 - (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent*)event {
763 [_flutterViewController touchesBegan:touches withEvent:event];
767 - (void)touchesMoved:(NSSet*)touches withEvent:(UIEvent*)event {
768 [_flutterViewController touchesMoved:touches withEvent:event];
771 - (void)touchesEnded:(NSSet*)touches withEvent:(UIEvent*)event {
772 [_flutterViewController touchesEnded:touches withEvent:event];
779 self.state = UIGestureRecognizerStateFailed;
781 [
self forceResetStateIfNeeded];
785 - (void)touchesCancelled:(NSSet*)touches withEvent:(UIEvent*)event {
791 [_flutterViewController forceTouchesCancelled:touches];
794 self.state = UIGestureRecognizerStateFailed;
796 [
self forceResetStateIfNeeded];
800 - (void)forceResetStateIfNeeded {
802 dispatch_async(dispatch_get_main_queue(), ^{
807 if (strongSelf.state != UIGestureRecognizerStatePossible) {
808 [(FlutterTouchInterceptingView*)strongSelf.view forceResetForwardingGestureRecognizerState];
813 - (BOOL)gestureRecognizer:(UIGestureRecognizer*)gestureRecognizer
814 shouldRecognizeSimultaneouslyWithGestureRecognizer:
815 (UIGestureRecognizer*)otherGestureRecognizer {