Flutter iOS Embedder
FlutterClippingMaskView Class Reference

#import <FlutterPlatformViews_Internal.h>

Inheritance diagram for FlutterClippingMaskView:

Instance Methods

(instancetype) - initWithFrame:screenScale:
 
(void) - reset
 
(void) - clipRect:matrix:
 
(void) - clipRRect:matrix:
 
(void) - clipPath:matrix:
 

Detailed Description

Definition at line 215 of file FlutterPlatformViews.mm.

Method Documentation

◆ clipPath:matrix:

- (void) clipPath: (const flutter::DlPath&)  path
matrix: (const flutter::DlMatrix&)  matrix 

Definition at line 383 of file FlutterPlatformViews.mm.

383  :(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 }

References containsNonRectPath_, and GetCATransform3DFromDlMatrix().

◆ clipRect:matrix:

- (void) clipRect: (const flutter::DlRect&)  clipDlRect
matrix: (const flutter::DlMatrix&)  matrix 

Definition at line 290 of file FlutterPlatformViews.mm.

290  :(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 }

References containsNonRectPath_, GetCATransform3DFromDlMatrix(), GetCGRectFromDlRect(), and rectSoFar_.

Referenced by clipRRect:matrix:.

◆ clipRRect:matrix:

- (void) clipRRect: (const flutter::DlRoundRect&)  clipDlRRect
matrix: (const flutter::DlMatrix&)  matrix 

Definition at line 306 of file FlutterPlatformViews.mm.

306  :(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 }

References clipRect:matrix:, containsNonRectPath_, GetCATransform3DFromDlMatrix(), and GetCGRectFromDlRect().

◆ initWithFrame:screenScale:

- (instancetype) initWithFrame: (CGRect)  frame
screenScale: (CGFloat)  screenScale 

Definition at line 221 of file FlutterPlatformViews.mm.

221  :(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 }

References containsNonRectPath_, initWithFrame, and rectSoFar_.

◆ reset

- (void) reset

Definition at line 239 of file FlutterPlatformViews.mm.

239  {
240  paths_.clear();
241  rectSoFar_ = self.bounds;
243  [self shapeLayer].path = nil;
244  [self setNeedsDisplay];
245 }

References containsNonRectPath_, and rectSoFar_.


The documentation for this class was generated from the following files:
rectSoFar_
CGRect rectSoFar_
Definition: FlutterPlatformViews.mm:218
initWithFrame
instancetype initWithFrame
Definition: FlutterTextInputPlugin.h:172
GetCGRectFromDlRect
static CGRect GetCGRectFromDlRect(const DlRect &clipDlRect)
Definition: FlutterPlatformViewsController.mm:79
containsNonRectPath_
BOOL containsNonRectPath_
Definition: FlutterPlatformViews.mm:215
GetCATransform3DFromDlMatrix
static CATransform3D GetCATransform3DFromDlMatrix(const DlMatrix &matrix)
Definition: FlutterPlatformViewsController.mm:46