Flutter iOS Embedder
overlay_layer_pool.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 
10 
11 namespace flutter {
12 
13 OverlayLayer::OverlayLayer(UIView* overlay_view,
14  UIView* overlay_view_wrapper,
15  std::unique_ptr<IOSSurface> ios_surface,
16  std::unique_ptr<Surface> surface)
17  : overlay_view(overlay_view),
18  overlay_view_wrapper(overlay_view_wrapper),
19  ios_surface(std::move(ios_surface)),
20  surface(std::move(surface)){};
21 
22 void OverlayLayer::UpdateViewState(UIView* flutter_view,
23  SkRect rect,
24  int64_t view_id,
25  int64_t overlay_id) {
26  auto screenScale = ((FlutterView*)flutter_view).screen.scale;
27  // Set the size of the overlay view wrapper.
28  // This wrapper view masks the overlay view.
29  overlay_view_wrapper.frame = CGRectMake(rect.x() / screenScale, rect.y() / screenScale,
30  rect.width() / screenScale, rect.height() / screenScale);
31  // Set a unique view identifier, so the overlay_view_wrapper can be identified in XCUITests.
32  overlay_view_wrapper.accessibilityIdentifier =
33  [NSString stringWithFormat:@"platform_view[%lld].overlay[%lld]", view_id, overlay_id];
34 
35  // Set the size of the overlay view.
36  // This size is equal to the device screen size.
37  overlay_view.frame = [flutter_view convertRect:flutter_view.bounds toView:overlay_view_wrapper];
38  // Set a unique view identifier, so the overlay_view can be identified in XCUITests.
39  overlay_view.accessibilityIdentifier =
40  [NSString stringWithFormat:@"platform_view[%lld].overlay_view[%lld]", view_id, overlay_id];
41 }
42 
43 // OverlayLayerPool
44 ////////////////////////////////////////////////////////
45 
46 std::shared_ptr<OverlayLayer> OverlayLayerPool::GetNextLayer() {
47  std::shared_ptr<OverlayLayer> result;
48  if (available_layer_index_ < layers_.size()) {
49  result = layers_[available_layer_index_];
50  available_layer_index_++;
51  }
52 
53  return result;
54 }
55 
56 void OverlayLayerPool::CreateLayer(const std::shared_ptr<IOSContext>& ios_context,
57  MTLPixelFormat pixel_format,
58  CGFloat screenScale) {
59  FML_DCHECK([[NSThread currentThread] isMainThread]);
60  std::shared_ptr<OverlayLayer> layer;
61  UIView* overlay_view;
62  UIView* overlay_view_wrapper;
63 
64  overlay_view = [[FlutterOverlayView alloc] initWithContentsScale:screenScale
65  pixelFormat:pixel_format];
66  overlay_view_wrapper = [[FlutterOverlayView alloc] initWithContentsScale:screenScale
67  pixelFormat:pixel_format];
68 
69  CALayer* ca_layer = overlay_view.layer;
70  std::unique_ptr<IOSSurface> ios_surface = IOSSurface::Create(ios_context, ca_layer);
71  std::unique_ptr<Surface> surface = ios_surface->CreateGPUSurface();
72 
73  layer = std::make_shared<OverlayLayer>(overlay_view, overlay_view_wrapper, std::move(ios_surface),
74  std::move(surface));
75  // The overlay view wrapper masks the overlay view.
76  // This is required to keep the backing surface size unchanged between frames.
77  //
78  // Otherwise, changing the size of the overlay would require a new surface,
79  // which can be very expensive.
80  //
81  // This is the case of an animation in which the overlay size is changing in every frame.
82  //
83  // +------------------------+
84  // | overlay_view |
85  // | +--------------+ | +--------------+
86  // | | wrapper | | == mask => | overlay_view |
87  // | +--------------+ | +--------------+
88  // +------------------------+
89  layer->overlay_view_wrapper.clipsToBounds = YES;
90  [layer->overlay_view_wrapper addSubview:layer->overlay_view];
91 
92  layers_.push_back(layer);
93 }
94 
96  available_layer_index_ = 0;
97 }
98 
99 std::vector<std::shared_ptr<OverlayLayer>> OverlayLayerPool::RemoveUnusedLayers() {
100  std::vector<std::shared_ptr<OverlayLayer>> results;
101  for (size_t i = available_layer_index_; i < layers_.size(); i++) {
102  results.push_back(layers_[i]);
103  }
104  // Leave at least one overlay layer, to work around cases where scrolling
105  // platform views under an app bar continually adds and removes an
106  // overlay layer. This logic could be removed if https://github.com/flutter/flutter/issues/150646
107  // is fixed.
108  static constexpr size_t kLeakLayerCount = 1;
109  size_t erase_offset = std::max(available_layer_index_, kLeakLayerCount);
110  if (erase_offset < layers_.size()) {
111  layers_.erase(layers_.begin() + erase_offset, layers_.end());
112  }
113  return results;
114 }
115 
116 size_t OverlayLayerPool::size() const {
117  return layers_.size();
118 }
119 
120 } // namespace flutter
flutter::OverlayLayer::OverlayLayer
OverlayLayer(UIView *overlay_view, UIView *overlay_view_wrapper, std::unique_ptr< IOSSurface > ios_surface, std::unique_ptr< Surface > surface)
Definition: overlay_layer_pool.mm:13
flutter::OverlayLayerPool::GetNextLayer
std::shared_ptr< OverlayLayer > GetNextLayer()
Gets a layer from the pool if available.
Definition: overlay_layer_pool.mm:46
flutter::OverlayLayerPool::size
size_t size() const
The count of layers currently in the pool.
Definition: overlay_layer_pool.mm:116
flutter::OverlayLayer::overlay_view_wrapper
UIView * overlay_view_wrapper
Definition: overlay_layer_pool.h:31
ios_surface.h
flutter::IOSSurface::Create
static std::unique_ptr< IOSSurface > Create(std::shared_ptr< IOSContext > context, CALayer *layer)
Definition: ios_surface.mm:17
flutter
Definition: accessibility_bridge.h:26
FlutterOverlayView.h
flutter::OverlayLayerPool::RecycleLayers
void RecycleLayers()
Marks the layers in the pool as available for reuse.
Definition: overlay_layer_pool.mm:95
flutter::OverlayLayerPool::RemoveUnusedLayers
std::vector< std::shared_ptr< OverlayLayer > > RemoveUnusedLayers()
Removes unused layers from the pool. Returns the unused layers.
Definition: overlay_layer_pool.mm:99
FlutterOverlayView
Definition: FlutterOverlayView.h:22
overlay_layer_pool.h
flutter::OverlayLayer::overlay_view
UIView * overlay_view
Definition: overlay_layer_pool.h:30
FlutterView
Definition: FlutterView.h:32
FlutterView.h
flutter::OverlayLayerPool::CreateLayer
void CreateLayer(const std::shared_ptr< IOSContext > &ios_context, MTLPixelFormat pixel_format, CGFloat screenScale)
Create a new overlay layer.
Definition: overlay_layer_pool.mm:56
flutter::OverlayLayer::UpdateViewState
void UpdateViewState(UIView *flutter_view, SkRect rect, int64_t view_id, int64_t overlay_id)
Definition: overlay_layer_pool.mm:22