Skip to content
This repository was archived by the owner on Feb 25, 2025. It is now read-only.

Commit f763b5b

Browse files
christopherfujinojmagmanTim SneathChris Yangjason-simmons
authored
Flutter 1.22.0-12.2.pre engine cherrypicks (#21366)
* Update 1.22 engine to use Dart 2.10.0-110.5.beta * Dark mode friendly iOS debugging message (#21277) * Update FlutterViewController.mm (#21362) * Tweaked the label here a little for style ("apps", not "application" -- plural and shorter); and "launching" rather than "re-launching"? * Fix iOS platform view's mask view blocking touch events. (#21286) * Disconnect the view's AndroidKeyProcessor when detaching from the engine (#21307) * Remove extraneous window inset call on IME animation (#21213) * Retain the WindowInsetsAnimation callback if code shrinking is enabled (#21330) Co-authored-by: Jenn Magder <[email protected]> Co-authored-by: Tim Sneath <[email protected]> Co-authored-by: Chris Yang <[email protected]> Co-authored-by: Jason Simmons <[email protected]> Co-authored-by: Gary Qian <[email protected]>
1 parent 4654fc6 commit f763b5b

File tree

12 files changed

+162
-24
lines changed

12 files changed

+162
-24
lines changed

DEPS

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ vars = {
3434
# Dart is: https://github.com/dart-lang/sdk/blob/master/DEPS.
3535
# You can use //tools/dart/create_updated_flutter_deps.py to produce
3636
# updated revision list of existing dependencies.
37-
'dart_revision': '52130c19ca593b185ea9cf72b26b1d02455551ef',
37+
'dart_revision': '4215dca724fb80de592f51a6cdba51e7638d1723',
3838

3939
# WARNING: DO NOT EDIT MANUALLY
4040
# The lines between blank lines above and below are generated by a script. See create_updated_flutter_deps.py

ci/licenses_golden/licenses_third_party

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
Signature: a1bbcd05a2657658be7c5f38e0d366f4
1+
Signature: 52ed6d65d7e96daef749ee003a3463a0
22

33
UNUSED LICENSES:
44

shell/platform/android/io/flutter/embedding/android/AndroidKeyProcessor.java

+9
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,15 @@ public AndroidKeyProcessor(
7070
this.keyEventChannel.setEventResponseHandler(eventResponder);
7171
}
7272

73+
/**
74+
* Detaches the key processor from the Flutter engine.
75+
*
76+
* <p>The AndroidKeyProcessor instance should not be used after calling this.
77+
*/
78+
public void destroy() {
79+
keyEventChannel.setEventResponseHandler(null);
80+
}
81+
7382
/**
7483
* Called when a key up event is received by the {@link FlutterView}.
7584
*

shell/platform/android/io/flutter/embedding/android/FlutterView.java

+2
Original file line numberDiff line numberDiff line change
@@ -992,6 +992,8 @@ public void detachFromFlutterEngine() {
992992
textInputPlugin.getInputMethodManager().restartInput(this);
993993
textInputPlugin.destroy();
994994

995+
androidKeyProcessor.destroy();
996+
995997
if (mouseCursorPlugin != null) {
996998
mouseCursorPlugin.destroy();
997999
}

shell/platform/android/io/flutter/plugin/editing/TextInputPlugin.java

+28-12
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
import android.view.inputmethod.InputConnection;
2929
import android.view.inputmethod.InputMethodManager;
3030
import android.view.inputmethod.InputMethodSubtype;
31+
import androidx.annotation.Keep;
3132
import androidx.annotation.NonNull;
3233
import androidx.annotation.Nullable;
3334
import androidx.annotation.RequiresApi;
@@ -192,14 +193,25 @@ public void sendAppPrivateCommand(String action, Bundle data) {
192193
@TargetApi(30)
193194
@RequiresApi(30)
194195
@SuppressLint({"NewApi", "Override"})
196+
@Keep
195197
class ImeSyncDeferringInsetsCallback extends WindowInsetsAnimation.Callback
196198
implements View.OnApplyWindowInsetsListener {
197199
private int overlayInsetTypes;
198200
private int deferredInsetTypes;
199201

200202
private View view;
201203
private WindowInsets lastWindowInsets;
202-
private boolean started = false;
204+
// True when an animation that matches deferredInsetTypes is active.
205+
//
206+
// While this is active, this class will capture the initial window inset
207+
// sent into lastWindowInsets by flagging needsSave to true, and will hold
208+
// onto the intitial inset until the animation is completed, when it will
209+
// re-dispatch the inset change.
210+
private boolean animating = false;
211+
// When an animation begins, android sends a WindowInset with the final
212+
// state of the animation. When needsSave is true, we know to capture this
213+
// initial WindowInset.
214+
private boolean needsSave = false;
203215

204216
ImeSyncDeferringInsetsCallback(
205217
@NonNull View view, int overlayInsetTypes, int deferredInsetTypes) {
@@ -212,34 +224,38 @@ class ImeSyncDeferringInsetsCallback extends WindowInsetsAnimation.Callback
212224
@Override
213225
public WindowInsets onApplyWindowInsets(View view, WindowInsets windowInsets) {
214226
this.view = view;
215-
if (started) {
227+
if (needsSave) {
228+
// Store the view and insets for us in onEnd() below. This captured inset
229+
// is not part of the animation and instead, represents the final state
230+
// of the inset after the animation is completed. Thus, we defer the processing
231+
// of this WindowInset until the animation completes.
232+
lastWindowInsets = windowInsets;
233+
needsSave = false;
234+
}
235+
if (animating) {
216236
// While animation is running, we consume the insets to prevent disrupting
217237
// the animation, which skips this implementation and calls the view's
218238
// onApplyWindowInsets directly to avoid being consumed here.
219239
return WindowInsets.CONSUMED;
220240
}
221241

222-
// Store the view and insets for us in onEnd() below
223-
lastWindowInsets = windowInsets;
224-
225242
// If no animation is happening, pass the insets on to the view's own
226243
// inset handling.
227244
return view.onApplyWindowInsets(windowInsets);
228245
}
229246

230247
@Override
231-
public WindowInsetsAnimation.Bounds onStart(
232-
WindowInsetsAnimation animation, WindowInsetsAnimation.Bounds bounds) {
248+
public void onPrepare(WindowInsetsAnimation animation) {
233249
if ((animation.getTypeMask() & deferredInsetTypes) != 0) {
234-
started = true;
250+
animating = true;
251+
needsSave = true;
235252
}
236-
return bounds;
237253
}
238254

239255
@Override
240256
public WindowInsets onProgress(
241257
WindowInsets insets, List<WindowInsetsAnimation> runningAnimations) {
242-
if (!started) {
258+
if (!animating || needsSave) {
243259
return insets;
244260
}
245261
boolean matching = false;
@@ -280,10 +296,10 @@ public WindowInsets onProgress(
280296

281297
@Override
282298
public void onEnd(WindowInsetsAnimation animation) {
283-
if (started && (animation.getTypeMask() & deferredInsetTypes) != 0) {
299+
if (animating && (animation.getTypeMask() & deferredInsetTypes) != 0) {
284300
// If we deferred the IME insets and an IME animation has finished, we need to reset
285301
// the flags
286-
started = false;
302+
animating = false;
287303

288304
// And finally dispatch the deferred insets to the view now.
289305
// Ideally we would just call view.requestApplyInsets() and let the normal dispatch

shell/platform/android/test/io/flutter/embedding/android/AndroidKeyProcessorTest.java

+18
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,9 @@
22

33
import static junit.framework.TestCase.assertEquals;
44
import static org.mockito.Mockito.any;
5+
import static org.mockito.Mockito.isNull;
56
import static org.mockito.Mockito.mock;
7+
import static org.mockito.Mockito.notNull;
68
import static org.mockito.Mockito.times;
79
import static org.mockito.Mockito.verify;
810
import static org.mockito.Mockito.when;
@@ -56,6 +58,22 @@ public void respondsTrueWhenHandlingNewEvents() {
5658
verify(fakeView, times(0)).dispatchKeyEvent(any(KeyEvent.class));
5759
}
5860

61+
@Test
62+
public void destroyTest() {
63+
FlutterEngine flutterEngine = mockFlutterEngine();
64+
KeyEventChannel fakeKeyEventChannel = flutterEngine.getKeyEventChannel();
65+
View fakeView = mock(View.class);
66+
67+
AndroidKeyProcessor processor =
68+
new AndroidKeyProcessor(fakeView, fakeKeyEventChannel, mock(TextInputPlugin.class));
69+
70+
verify(fakeKeyEventChannel, times(1))
71+
.setEventResponseHandler(notNull(KeyEventChannel.EventResponseHandler.class));
72+
processor.destroy();
73+
verify(fakeKeyEventChannel, times(1))
74+
.setEventResponseHandler(isNull(KeyEventChannel.EventResponseHandler.class));
75+
}
76+
5977
public void synthesizesEventsWhenKeyDownNotHandled() {
6078
FlutterEngine flutterEngine = mockFlutterEngine();
6179
KeyEventChannel fakeKeyEventChannel = flutterEngine.getKeyEventChannel();

shell/platform/android/test/io/flutter/plugin/editing/TextInputPluginTest.java

+9-1
Original file line numberDiff line numberDiff line change
@@ -669,6 +669,8 @@ public void ime_windowInsetsSync() {
669669
WindowInsets.Builder builder = new WindowInsets.Builder();
670670
WindowInsets noneInsets = builder.build();
671671

672+
// imeInsets0, 1, and 2 contain unique IME bottom insets, and are used
673+
// to distinguish which insets were sent at each stage.
672674
builder.setInsets(WindowInsets.Type.ime(), Insets.of(0, 0, 0, 100));
673675
builder.setInsets(WindowInsets.Type.navigationBars(), Insets.of(10, 10, 10, 40));
674676
WindowInsets imeInsets0 = builder.build();
@@ -677,6 +679,10 @@ public void ime_windowInsetsSync() {
677679
builder.setInsets(WindowInsets.Type.navigationBars(), Insets.of(10, 10, 10, 40));
678680
WindowInsets imeInsets1 = builder.build();
679681

682+
builder.setInsets(WindowInsets.Type.ime(), Insets.of(0, 0, 0, 50));
683+
builder.setInsets(WindowInsets.Type.navigationBars(), Insets.of(10, 10, 10, 40));
684+
WindowInsets imeInsets2 = builder.build();
685+
680686
builder.setInsets(WindowInsets.Type.ime(), Insets.of(0, 0, 0, 200));
681687
builder.setInsets(WindowInsets.Type.navigationBars(), Insets.of(10, 10, 10, 0));
682688
WindowInsets deferredInsets = builder.build();
@@ -696,6 +702,8 @@ public void ime_windowInsetsSync() {
696702
imeSyncCallback.onPrepare(animation);
697703
imeSyncCallback.onApplyWindowInsets(testView, deferredInsets);
698704
imeSyncCallback.onStart(animation, null);
705+
// Only the final state call is saved, extra calls are passed on.
706+
imeSyncCallback.onApplyWindowInsets(testView, imeInsets2);
699707

700708
verify(flutterRenderer).setViewportMetrics(viewportMetricsCaptor.capture());
701709
// No change, as deferredInset is stored to be passed in onEnd()
@@ -723,7 +731,7 @@ public void ime_windowInsetsSync() {
723731
imeSyncCallback.onEnd(animation);
724732

725733
verify(flutterRenderer).setViewportMetrics(viewportMetricsCaptor.capture());
726-
// Values should be of deferredInsets
734+
// Values should be of deferredInsets, not imeInsets2
727735
assertEquals(0, viewportMetricsCaptor.getValue().paddingBottom);
728736
assertEquals(10, viewportMetricsCaptor.getValue().paddingTop);
729737
assertEquals(200, viewportMetricsCaptor.getValue().viewInsetBottom);

shell/platform/darwin/ios/framework/Source/FlutterPlatformViews_Internal.mm

+9
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,15 @@ - (instancetype)initWithFrame:(CGRect)frame {
9090
return self;
9191
}
9292

93+
// In some scenarios, when we add this view as a maskView of the ChildClippingView, iOS added
94+
// this view as a subview of the ChildClippingView.
95+
// This results this view blocking touch events on the ChildClippingView.
96+
// So we should always ignore any touch events sent to this view.
97+
// See https://github.com/flutter/flutter/issues/66044
98+
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event {
99+
return NO;
100+
}
101+
93102
- (void)drawRect:(CGRect)rect {
94103
CGContextRef context = UIGraphicsGetCurrentContext();
95104
CGContextSaveGState(context);

shell/platform/darwin/ios/framework/Source/FlutterViewController.mm

+7-3
Original file line numberDiff line numberDiff line change
@@ -326,7 +326,11 @@ - (void)pushRoute:(NSString*)route {
326326
auto placeholder = [[[UIView alloc] init] autorelease];
327327

328328
placeholder.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
329-
placeholder.backgroundColor = UIColor.whiteColor;
329+
if (@available(iOS 13.0, *)) {
330+
placeholder.backgroundColor = UIColor.systemBackgroundColor;
331+
} else {
332+
placeholder.backgroundColor = UIColor.whiteColor;
333+
}
330334
placeholder.autoresizesSubviews = YES;
331335

332336
// Only add the label when we know we have failed to enable tracing (and it was necessary).
@@ -339,9 +343,9 @@ - (void)pushRoute:(NSString*)route {
339343
messageLabel.autoresizingMask =
340344
UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
341345
messageLabel.text =
342-
@"In iOS 14+, Flutter application in debug mode can only be launched from Flutter tooling, "
346+
@"In iOS 14+, debug mode Flutter apps can only be launched from Flutter tooling, "
343347
@"IDEs with Flutter plugins or from Xcode.\n\nAlternatively, build in profile or release "
344-
@"modes to enable re-launching from the home screen.";
348+
@"modes to enable launching from the home screen.";
345349
[placeholder addSubview:messageLabel];
346350
}
347351

testing/scenario_app/ios/Scenarios/Scenarios/AppDelegate.m

+3-2
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,9 @@ @implementation AppDelegate
2323
- (BOOL)application:(UIApplication*)application
2424
didFinishLaunchingWithOptions:(NSDictionary*)launchOptions {
2525
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
26-
26+
if ([[[NSProcessInfo processInfo] arguments] containsObject:@"--maskview-blocking"]) {
27+
self.window.tintColor = UIColor.systemPinkColor;
28+
}
2729
NSDictionary<NSString*, NSString*>* launchArgsMap = @{
2830
// The Platform view golden test args should match `PlatformViewGoldenTestManager`.
2931
@"--locale-initialization" : @"locale_initialization",
@@ -58,7 +60,6 @@ - (BOOL)application:(UIApplication*)application
5860
*stop = YES;
5961
}
6062
}];
61-
6263
if (flutterViewControllerTestName) {
6364
[self setupFlutterViewControllerTest:flutterViewControllerTestName];
6465
} else if ([[[NSProcessInfo processInfo] arguments] containsObject:@"--screen-before-flutter"]) {

testing/scenario_app/ios/Scenarios/ScenariosUITests/PlatformViewGestureRecognizerTests.m

+45
Original file line numberDiff line numberDiff line change
@@ -110,9 +110,54 @@ - (void)testAccept {
110110
[[XCTNSPredicateExpectation alloc] initWithPredicate:predicate object:platformView];
111111

112112
[platformView tap];
113+
114+
[self waitForExpectations:@[ expection ] timeout:kSecondsToWaitForPlatformView];
115+
XCTAssertEqualObjects(platformView.label,
116+
@"-gestureTouchesBegan-gestureTouchesEnded-platformViewTapped");
117+
}
118+
119+
- (void)testGestureWithMaskViewBlockingPlatformView {
120+
XCUIApplication* app = [[XCUIApplication alloc] init];
121+
app.launchArguments = @[ @"--gesture-accept", @"--maskview-blocking" ];
122+
[app launch];
123+
124+
NSPredicate* predicateToFindPlatformView =
125+
[NSPredicate predicateWithBlock:^BOOL(id _Nullable evaluatedObject,
126+
NSDictionary<NSString*, id>* _Nullable bindings) {
127+
XCUIElement* element = evaluatedObject;
128+
return [element.identifier hasPrefix:@"platform_view"];
129+
}];
130+
XCUIElement* platformView = [app.textViews elementMatchingPredicate:predicateToFindPlatformView];
131+
if (![platformView waitForExistenceWithTimeout:kSecondsToWaitForPlatformView]) {
132+
NSLog(@"%@", app.debugDescription);
133+
XCTFail(@"Failed due to not able to find any platformView with %@ seconds",
134+
@(kSecondsToWaitForPlatformView));
135+
}
136+
137+
XCTAssertNotNil(platformView);
138+
XCTAssertEqualObjects(platformView.label, @"");
139+
140+
NSPredicate* predicate = [NSPredicate
141+
predicateWithFormat:@"label == %@",
142+
@"-gestureTouchesBegan-gestureTouchesEnded-platformViewTapped"];
143+
XCTNSPredicateExpectation* expection =
144+
[[XCTNSPredicateExpectation alloc] initWithPredicate:predicate object:platformView];
145+
146+
XCUICoordinate* coordinate =
147+
[self getNormalizedCoordinate:app
148+
point:CGVectorMake(platformView.frame.origin.x + 10,
149+
platformView.frame.origin.y + 10)];
150+
[coordinate tap];
151+
113152
[self waitForExpectations:@[ expection ] timeout:kSecondsToWaitForPlatformView];
114153
XCTAssertEqualObjects(platformView.label,
115154
@"-gestureTouchesBegan-gestureTouchesEnded-platformViewTapped");
116155
}
117156

157+
- (XCUICoordinate*)getNormalizedCoordinate:(XCUIApplication*)app point:(CGVector)vector {
158+
XCUICoordinate* appZero = [app coordinateWithNormalizedOffset:CGVectorMake(0, 0)];
159+
XCUICoordinate* coordinate = [appZero coordinateWithOffset:vector];
160+
return coordinate;
161+
}
162+
118163
@end

testing/scenario_app/lib/src/platform_view.dart

+30-4
Original file line numberDiff line numberDiff line change
@@ -336,9 +336,9 @@ class MultiPlatformViewBackgroundForegroundScenario extends Scenario with _BaseP
336336
MultiPlatformViewBackgroundForegroundScenario(Window window, {this.firstId, this.secondId})
337337
: assert(window != null),
338338
super(window) {
339+
_nextFrame = _firstFrame;
339340
createPlatformView(window, 'platform view 1', firstId);
340341
createPlatformView(window, 'platform view 2', secondId);
341-
_nextFrame = _firstFrame;
342342
}
343343

344344
/// The platform view identifier to use for the first platform view.
@@ -532,6 +532,8 @@ class PlatformViewForTouchIOSScenario extends Scenario
532532

533533
int _viewId;
534534
bool _accept;
535+
536+
VoidCallback _nextFrame;
535537
/// Creates the PlatformView scenario.
536538
///
537539
/// The [window] parameter must not be null.
@@ -545,14 +547,24 @@ class PlatformViewForTouchIOSScenario extends Scenario
545547
} else {
546548
createPlatformView(window, text, id);
547549
}
550+
_nextFrame = _firstFrame;
548551
}
549552

550553
@override
551554
void onBeginFrame(Duration duration) {
552-
final SceneBuilder builder = SceneBuilder();
555+
_nextFrame();
556+
}
553557

554-
builder.pushOffset(0, 0);
555-
finishBuilderByAddingPlatformViewAndPicture(builder, _viewId);
558+
@override
559+
void onDrawFrame() {
560+
// Some iOS gesture recognizers bugs are introduced in the second frame (with a different platform view rect) after laying out the platform view.
561+
// So in this test, we load 2 frames to ensure that we cover those cases.
562+
// See https://github.com/flutter/flutter/issues/66044
563+
if (_nextFrame == _firstFrame) {
564+
_nextFrame = _secondFrame;
565+
window.scheduleFrame();
566+
}
567+
super.onDrawFrame();
556568
}
557569

558570
@override
@@ -585,6 +597,20 @@ class PlatformViewForTouchIOSScenario extends Scenario
585597
}
586598

587599
}
600+
601+
void _firstFrame() {
602+
final SceneBuilder builder = SceneBuilder();
603+
604+
builder.pushOffset(0, 0);
605+
finishBuilderByAddingPlatformViewAndPicture(builder, _viewId);
606+
}
607+
608+
void _secondFrame() {
609+
final SceneBuilder builder = SceneBuilder();
610+
611+
builder.pushOffset(5, 5);
612+
finishBuilderByAddingPlatformViewAndPicture(builder, _viewId);
613+
}
588614
}
589615

590616
mixin _BasePlatformViewScenarioMixin on Scenario {

0 commit comments

Comments
 (0)