Flutter iOS Embedder
FlutterTextInputPluginTest.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 
8 
9 #import <OCMock/OCMock.h>
10 #import <XCTest/XCTest.h>
11 
16 
18 
19 @interface FlutterEngine ()
21 @end
22 
23 @interface FlutterTextInputView ()
24 @property(nonatomic, copy) NSString* autofillId;
25 - (void)setEditableTransform:(NSArray*)matrix;
26 - (void)setTextInputClient:(int)client;
27 - (void)setTextInputState:(NSDictionary*)state;
28 - (void)setMarkedRect:(CGRect)markedRect;
29 - (void)updateEditingState;
30 - (BOOL)isVisibleToAutofill;
31 - (id<FlutterTextInputDelegate>)textInputDelegate;
32 - (void)configureWithDictionary:(NSDictionary*)configuration;
33 - (void)handleSearchWebAction;
34 - (void)handleLookUpAction;
35 - (void)handleShareAction;
36 @end
37 
39 @property(nonatomic, assign) UIAccessibilityNotifications receivedNotification;
40 @property(nonatomic, assign) id receivedNotificationTarget;
41 @property(nonatomic, assign) BOOL isAccessibilityFocused;
42 
43 - (void)postAccessibilityNotification:(UIAccessibilityNotifications)notification target:(id)target;
44 
45 @end
46 
47 @implementation FlutterTextInputViewSpy {
48 }
49 
50 - (void)postAccessibilityNotification:(UIAccessibilityNotifications)notification target:(id)target {
51  self.receivedNotification = notification;
52  self.receivedNotificationTarget = target;
53 }
54 
55 - (BOOL)accessibilityElementIsFocused {
56  return _isAccessibilityFocused;
57 }
58 
59 @end
60 
62 @property(nonatomic, strong) UITextField* textField;
63 @end
64 
65 @interface FlutterTextInputPlugin ()
66 @property(nonatomic, assign) FlutterTextInputView* activeView;
67 @property(nonatomic, readonly) UIView* inputHider;
68 @property(nonatomic, readonly) UIView* keyboardViewContainer;
69 @property(nonatomic, readonly) UIView* keyboardView;
70 @property(nonatomic, assign) UIView* cachedFirstResponder;
71 @property(nonatomic, readonly) CGRect keyboardRect;
72 @property(nonatomic, readonly)
73  NSMutableDictionary<NSString*, FlutterTextInputView*>* autofillContext;
74 
75 - (void)cleanUpViewHierarchy:(BOOL)includeActiveView
76  clearText:(BOOL)clearText
77  delayRemoval:(BOOL)delayRemoval;
78 - (NSArray<UIView*>*)textInputViews;
79 - (UIView*)hostView;
80 - (void)addToInputParentViewIfNeeded:(FlutterTextInputView*)inputView;
81 - (void)startLiveTextInput;
82 - (void)showKeyboardAndRemoveScreenshot;
83 
84 @end
85 
86 @interface FlutterTextInputPluginTest : XCTestCase
87 @end
88 
89 @implementation FlutterTextInputPluginTest {
90  NSDictionary* _template;
91  NSDictionary* _passwordTemplate;
92  id engine;
94 
96 }
97 
98 - (void)setUp {
99  [super setUp];
100  engine = OCMClassMock([FlutterEngine class]);
101 
102  textInputPlugin = [[FlutterTextInputPlugin alloc] initWithDelegate:engine];
103 
104  viewController = [[FlutterViewController alloc] init];
106 
107  // Clear pasteboard between tests.
108  UIPasteboard.generalPasteboard.items = @[];
109 }
110 
111 - (void)tearDown {
112  textInputPlugin = nil;
113  engine = nil;
114  [textInputPlugin.autofillContext removeAllObjects];
115  [textInputPlugin cleanUpViewHierarchy:YES clearText:YES delayRemoval:NO];
116  [[[[textInputPlugin textInputView] superview] subviews]
117  makeObjectsPerformSelector:@selector(removeFromSuperview)];
118  viewController = nil;
119  [super tearDown];
120 }
121 
122 - (void)setClientId:(int)clientId configuration:(NSDictionary*)config {
123  FlutterMethodCall* setClientCall =
124  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
125  arguments:@[ [NSNumber numberWithInt:clientId], config ]];
126  [textInputPlugin handleMethodCall:setClientCall
127  result:^(id _Nullable result){
128  }];
129 }
130 
131 - (void)setTextInputShow {
132  FlutterMethodCall* setClientCall = [FlutterMethodCall methodCallWithMethodName:@"TextInput.show"
133  arguments:@[]];
134  [textInputPlugin handleMethodCall:setClientCall
135  result:^(id _Nullable result){
136  }];
137 }
138 
139 - (void)setTextInputHide {
140  FlutterMethodCall* setClientCall = [FlutterMethodCall methodCallWithMethodName:@"TextInput.hide"
141  arguments:@[]];
142  [textInputPlugin handleMethodCall:setClientCall
143  result:^(id _Nullable result){
144  }];
145 }
146 
147 - (void)flushScheduledAsyncBlocks {
148  __block bool done = false;
149  XCTestExpectation* expectation =
150  [[XCTestExpectation alloc] initWithDescription:@"Testing on main queue"];
151  dispatch_async(dispatch_get_main_queue(), ^{
152  done = true;
153  });
154  dispatch_async(dispatch_get_main_queue(), ^{
155  XCTAssertTrue(done);
156  [expectation fulfill];
157  });
158  [self waitForExpectations:@[ expectation ] timeout:10];
159 }
160 
161 - (NSMutableDictionary*)mutableTemplateCopy {
162  if (!_template) {
163  _template = @{
164  @"inputType" : @{@"name" : @"TextInuptType.text"},
165  @"keyboardAppearance" : @"Brightness.light",
166  @"obscureText" : @NO,
167  @"inputAction" : @"TextInputAction.unspecified",
168  @"smartDashesType" : @"0",
169  @"smartQuotesType" : @"0",
170  @"autocorrect" : @YES,
171  @"enableInteractiveSelection" : @YES,
172  };
173  }
174 
175  return [_template mutableCopy];
176 }
177 
178 - (NSArray<FlutterTextInputView*>*)installedInputViews {
179  return (NSArray<FlutterTextInputView*>*)[textInputPlugin.textInputViews
180  filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"self isKindOfClass: %@",
181  [FlutterTextInputView class]]];
182 }
183 
184 - (FlutterTextRange*)getLineRangeFromTokenizer:(id<UITextInputTokenizer>)tokenizer
185  atIndex:(NSInteger)index {
186  UITextRange* range =
187  [tokenizer rangeEnclosingPosition:[FlutterTextPosition positionWithIndex:index]
188  withGranularity:UITextGranularityLine
189  inDirection:UITextLayoutDirectionRight];
190  XCTAssertTrue([range isKindOfClass:[FlutterTextRange class]]);
191  return (FlutterTextRange*)range;
192 }
193 
194 - (void)updateConfig:(NSDictionary*)config {
195  FlutterMethodCall* updateConfigCall =
196  [FlutterMethodCall methodCallWithMethodName:@"TextInput.updateConfig" arguments:config];
197  [textInputPlugin handleMethodCall:updateConfigCall
198  result:^(id _Nullable result){
199  }];
200 }
201 
202 #pragma mark - Tests
203 
204 - (void)testWillNotCrashWhenViewControllerIsNil {
205  FlutterEngine* flutterEngine = [[FlutterEngine alloc] init];
206  FlutterTextInputPlugin* inputPlugin =
207  [[FlutterTextInputPlugin alloc] initWithDelegate:(id<FlutterTextInputDelegate>)flutterEngine];
208  XCTAssertNil(inputPlugin.viewController);
209  FlutterMethodCall* methodCall = [FlutterMethodCall methodCallWithMethodName:@"TextInput.show"
210  arguments:nil];
211  XCTestExpectation* expectation = [[XCTestExpectation alloc] initWithDescription:@"result called"];
212 
213  [inputPlugin handleMethodCall:methodCall
214  result:^(id _Nullable result) {
215  XCTAssertNil(result);
216  [expectation fulfill];
217  }];
218  XCTAssertNil(inputPlugin.activeView);
219  [self waitForExpectations:@[ expectation ] timeout:1.0];
220 }
221 
222 - (void)testInvokeStartLiveTextInput {
223  FlutterMethodCall* methodCall =
224  [FlutterMethodCall methodCallWithMethodName:@"TextInput.startLiveTextInput" arguments:nil];
225  FlutterTextInputPlugin* mockPlugin = OCMPartialMock(textInputPlugin);
226  [mockPlugin handleMethodCall:methodCall
227  result:^(id _Nullable result){
228  }];
229  OCMVerify([mockPlugin startLiveTextInput]);
230 }
231 
232 - (void)testNoDanglingEnginePointer {
233  __weak FlutterTextInputPlugin* weakFlutterTextInputPlugin;
234  FlutterViewController* flutterViewController = [[FlutterViewController alloc] init];
235  __weak FlutterEngine* weakFlutterEngine;
236 
237  FlutterTextInputView* currentView;
238 
239  // The engine instance will be deallocated after the autorelease pool is drained.
240  @autoreleasepool {
241  FlutterEngine* flutterEngine = OCMClassMock([FlutterEngine class]);
242  weakFlutterEngine = flutterEngine;
243  XCTAssertNotNil(weakFlutterEngine, @"flutter engine must not be nil");
244  FlutterTextInputPlugin* flutterTextInputPlugin = [[FlutterTextInputPlugin alloc]
245  initWithDelegate:(id<FlutterTextInputDelegate>)flutterEngine];
246  weakFlutterTextInputPlugin = flutterTextInputPlugin;
247  flutterTextInputPlugin.viewController = flutterViewController;
248 
249  // Set client so the text input plugin has an active view.
250  NSDictionary* config = self.mutableTemplateCopy;
251  FlutterMethodCall* setClientCall =
252  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
253  arguments:@[ [NSNumber numberWithInt:123], config ]];
254  [flutterTextInputPlugin handleMethodCall:setClientCall
255  result:^(id _Nullable result){
256  }];
257  currentView = flutterTextInputPlugin.activeView;
258  }
259 
260  XCTAssertNil(weakFlutterEngine, @"flutter engine must be nil");
261  XCTAssertNotNil(currentView, @"current view must not be nil");
262 
263  XCTAssertNil(weakFlutterTextInputPlugin);
264  // Verify that the view can no longer access the deallocated engine/text input plugin
265  // instance.
266  XCTAssertNil(currentView.textInputDelegate);
267 }
268 
269 - (void)testSecureInput {
270  NSDictionary* config = self.mutableTemplateCopy;
271  [config setValue:@"YES" forKey:@"obscureText"];
272  [self setClientId:123 configuration:config];
273 
274  // Find all the FlutterTextInputViews we created.
275  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
276 
277  // There are no autofill and the mock framework requested a secure entry. The first and only
278  // inserted FlutterTextInputView should be a secure text entry one.
279  FlutterTextInputView* inputView = inputFields[0];
280 
281  // Verify secureTextEntry is set to the correct value.
282  XCTAssertTrue(inputView.secureTextEntry);
283 
284  // Verify keyboardType is set to the default value.
285  XCTAssertEqual(inputView.keyboardType, UIKeyboardTypeDefault);
286 
287  // We should have only ever created one FlutterTextInputView.
288  XCTAssertEqual(inputFields.count, 1ul);
289 
290  // The one FlutterTextInputView we inserted into the view hierarchy should be the text input
291  // plugin's active text input view.
292  XCTAssertEqual(inputView, textInputPlugin.textInputView);
293 
294  // Despite not given an id in configuration, inputView has
295  // an autofill id.
296  XCTAssert(inputView.autofillId.length > 0);
297 }
298 
299 - (void)testKeyboardType {
300  NSDictionary* config = self.mutableTemplateCopy;
301  [config setValue:@{@"name" : @"TextInputType.url"} forKey:@"inputType"];
302  [self setClientId:123 configuration:config];
303 
304  // Find all the FlutterTextInputViews we created.
305  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
306 
307  FlutterTextInputView* inputView = inputFields[0];
308 
309  // Verify keyboardType is set to the value specified in config.
310  XCTAssertEqual(inputView.keyboardType, UIKeyboardTypeURL);
311 }
312 
313 - (void)testKeyboardTypeWebSearch {
314  NSDictionary* config = self.mutableTemplateCopy;
315  [config setValue:@{@"name" : @"TextInputType.webSearch"} forKey:@"inputType"];
316  [self setClientId:123 configuration:config];
317 
318  // Find all the FlutterTextInputViews we created.
319  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
320 
321  FlutterTextInputView* inputView = inputFields[0];
322 
323  // Verify keyboardType is set to the value specified in config.
324  XCTAssertEqual(inputView.keyboardType, UIKeyboardTypeWebSearch);
325 }
326 
327 - (void)testKeyboardTypeTwitter {
328  NSDictionary* config = self.mutableTemplateCopy;
329  [config setValue:@{@"name" : @"TextInputType.twitter"} forKey:@"inputType"];
330  [self setClientId:123 configuration:config];
331 
332  // Find all the FlutterTextInputViews we created.
333  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
334 
335  FlutterTextInputView* inputView = inputFields[0];
336 
337  // Verify keyboardType is set to the value specified in config.
338  XCTAssertEqual(inputView.keyboardType, UIKeyboardTypeTwitter);
339 }
340 
341 - (void)testVisiblePasswordUseAlphanumeric {
342  NSDictionary* config = self.mutableTemplateCopy;
343  [config setValue:@{@"name" : @"TextInputType.visiblePassword"} forKey:@"inputType"];
344  [self setClientId:123 configuration:config];
345 
346  // Find all the FlutterTextInputViews we created.
347  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
348 
349  FlutterTextInputView* inputView = inputFields[0];
350 
351  // Verify keyboardType is set to the value specified in config.
352  XCTAssertEqual(inputView.keyboardType, UIKeyboardTypeASCIICapable);
353 }
354 
355 - (void)testSettingKeyboardTypeNoneDisablesSystemKeyboard {
356  NSDictionary* config = self.mutableTemplateCopy;
357  [config setValue:@{@"name" : @"TextInputType.none"} forKey:@"inputType"];
358  [self setClientId:123 configuration:config];
359 
360  // Verify the view's inputViewController is not nil;
361  XCTAssertNotNil(textInputPlugin.activeView.inputViewController);
362 
363  [config setValue:@{@"name" : @"TextInputType.url"} forKey:@"inputType"];
364  [self setClientId:124 configuration:config];
365  XCTAssertNotNil(textInputPlugin.activeView);
366  XCTAssertNil(textInputPlugin.activeView.inputViewController);
367 }
368 
369 - (void)testAutocorrectionPromptRectAppearsBeforeIOS17AndDoesNotAppearAfterIOS17 {
370  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
371  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
372 
373  if (@available(iOS 17.0, *)) {
374  // Auto-correction prompt is disabled in iOS 17+.
375  OCMVerify(never(), [engine flutterTextInputView:inputView
376  showAutocorrectionPromptRectForStart:0
377  end:1
378  withClient:0]);
379  } else {
380  OCMVerify([engine flutterTextInputView:inputView
381  showAutocorrectionPromptRectForStart:0
382  end:1
383  withClient:0]);
384  }
385 }
386 
387 - (void)testIgnoresSelectionChangeIfSelectionIsDisabled {
388  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
389  __block int updateCount = 0;
390  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withState:[OCMArg isNotNil]])
391  .andDo(^(NSInvocation* invocation) {
392  updateCount++;
393  });
394 
395  [inputView.text setString:@"Some initial text"];
396  XCTAssertEqual(updateCount, 0);
397 
398  FlutterTextRange* textRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)];
399  [inputView setSelectedTextRange:textRange];
400  XCTAssertEqual(updateCount, 1);
401 
402  // Disable the interactive selection.
403  NSDictionary* config = self.mutableTemplateCopy;
404  [config setValue:@(NO) forKey:@"enableInteractiveSelection"];
405  [config setValue:@(NO) forKey:@"obscureText"];
406  [config setValue:@(NO) forKey:@"enableDeltaModel"];
407  [inputView configureWithDictionary:config];
408 
409  textRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(2, 3)];
410  [inputView setSelectedTextRange:textRange];
411  // The update count does not change.
412  XCTAssertEqual(updateCount, 1);
413 }
414 
415 - (void)testAutocorrectionPromptRectDoesNotAppearDuringScribble {
416  // Auto-correction prompt is disabled in iOS 17+.
417  if (@available(iOS 17.0, *)) {
418  return;
419  }
420 
421  if (@available(iOS 14.0, *)) {
422  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
423 
424  __block int callCount = 0;
425  OCMStub([engine flutterTextInputView:inputView
426  showAutocorrectionPromptRectForStart:0
427  end:1
428  withClient:0])
429  .andDo(^(NSInvocation* invocation) {
430  callCount++;
431  });
432 
433  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
434  // showAutocorrectionPromptRectForStart fires in response to firstRectForRange
435  XCTAssertEqual(callCount, 1);
436 
437  UIScribbleInteraction* scribbleInteraction =
438  [[UIScribbleInteraction alloc] initWithDelegate:inputView];
439 
440  [inputView scribbleInteractionWillBeginWriting:scribbleInteraction];
441  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
442  // showAutocorrectionPromptRectForStart does not fire in response to setMarkedText during a
443  // scribble interaction.firstRectForRange
444  XCTAssertEqual(callCount, 1);
445 
446  [inputView scribbleInteractionDidFinishWriting:scribbleInteraction];
447  [inputView resetScribbleInteractionStatusIfEnding];
448  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
449  // showAutocorrectionPromptRectForStart fires in response to firstRectForRange.
450  XCTAssertEqual(callCount, 2);
451 
452  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusFocusing;
453  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
454  // showAutocorrectionPromptRectForStart does not fire in response to firstRectForRange during a
455  // scribble-initiated focus.
456  XCTAssertEqual(callCount, 2);
457 
458  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusFocused;
459  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
460  // showAutocorrectionPromptRectForStart does not fire in response to firstRectForRange after a
461  // scribble-initiated focus.
462  XCTAssertEqual(callCount, 2);
463 
464  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusUnfocused;
465  [inputView firstRectForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]];
466  // showAutocorrectionPromptRectForStart fires in response to firstRectForRange.
467  XCTAssertEqual(callCount, 3);
468  }
469 }
470 
471 - (void)testInputHiderOverlapWithTextWhenScribbleIsDisabledAfterIOS17AndDoesNotOverlapBeforeIOS17 {
472  FlutterTextInputPlugin* myInputPlugin =
473  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
474 
475  FlutterMethodCall* setClientCall =
476  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
477  arguments:@[ @(123), self.mutableTemplateCopy ]];
478  [myInputPlugin handleMethodCall:setClientCall
479  result:^(id _Nullable result){
480  }];
481 
482  FlutterTextInputView* mockInputView = OCMPartialMock(myInputPlugin.activeView);
483  OCMStub([mockInputView isScribbleAvailable]).andReturn(NO);
484 
485  // yOffset = 200.
486  NSArray* yOffsetMatrix = @[ @1, @0, @0, @0, @0, @1, @0, @0, @0, @0, @1, @0, @0, @200, @0, @1 ];
487 
488  FlutterMethodCall* setPlatformViewClientCall =
489  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setEditableSizeAndTransform"
490  arguments:@{@"transform" : yOffsetMatrix}];
491  [myInputPlugin handleMethodCall:setPlatformViewClientCall
492  result:^(id _Nullable result){
493  }];
494 
495  if (@available(iOS 17, *)) {
496  XCTAssert(CGRectEqualToRect(myInputPlugin.inputHider.frame, CGRectMake(0, 200, 0, 0)),
497  @"The input hider should overlap with the text on and after iOS 17");
498 
499  } else {
500  XCTAssert(CGRectEqualToRect(myInputPlugin.inputHider.frame, CGRectZero),
501  @"The input hider should be on the origin of screen on and before iOS 16.");
502  }
503 }
504 
505 - (void)testTextRangeFromPositionMatchesUITextViewBehavior {
506  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
509 
510  FlutterTextRange* flutterRange = (FlutterTextRange*)[inputView textRangeFromPosition:fromPosition
511  toPosition:toPosition];
512  NSRange range = flutterRange.range;
513 
514  XCTAssertEqual(range.location, 0ul);
515  XCTAssertEqual(range.length, 2ul);
516 }
517 
518 - (void)testTextInRange {
519  NSDictionary* config = self.mutableTemplateCopy;
520  [config setValue:@{@"name" : @"TextInputType.url"} forKey:@"inputType"];
521  [self setClientId:123 configuration:config];
522  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
523  FlutterTextInputView* inputView = inputFields[0];
524 
525  [inputView insertText:@"test"];
526 
527  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 20)];
528  NSString* substring = [inputView textInRange:range];
529  XCTAssertEqual(substring.length, 4ul);
530 
531  range = [FlutterTextRange rangeWithNSRange:NSMakeRange(10, 20)];
532  substring = [inputView textInRange:range];
533  XCTAssertEqual(substring.length, 0ul);
534 }
535 
536 - (void)testTextInRangeAcceptsNSNotFoundLocationGracefully {
537  NSDictionary* config = self.mutableTemplateCopy;
538  [self setClientId:123 configuration:config];
539  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
540  FlutterTextInputView* inputView = inputFields[0];
541 
542  [inputView insertText:@"text"];
543  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(NSNotFound, 0)];
544 
545  NSString* substring = [inputView textInRange:range];
546  XCTAssertNil(substring);
547 }
548 
549 - (void)testStandardEditActions {
550  NSDictionary* config = self.mutableTemplateCopy;
551  [self setClientId:123 configuration:config];
552  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
553  FlutterTextInputView* inputView = inputFields[0];
554 
555  [inputView insertText:@"aaaa"];
556  [inputView selectAll:nil];
557  [inputView cut:nil];
558  [inputView insertText:@"bbbb"];
559  XCTAssertTrue([inputView canPerformAction:@selector(paste:) withSender:nil]);
560  [inputView paste:nil];
561  [inputView selectAll:nil];
562  [inputView copy:nil];
563  [inputView paste:nil];
564  [inputView selectAll:nil];
565  [inputView delete:nil];
566  [inputView paste:nil];
567  [inputView paste:nil];
568 
569  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 30)];
570  NSString* substring = [inputView textInRange:range];
571  XCTAssertEqualObjects(substring, @"bbbbaaaabbbbaaaa");
572 }
573 
574 - (void)testCanPerformActionForSelectActions {
575  NSDictionary* config = self.mutableTemplateCopy;
576  [self setClientId:123 configuration:config];
577  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
578  FlutterTextInputView* inputView = inputFields[0];
579 
580  XCTAssertFalse([inputView canPerformAction:@selector(selectAll:) withSender:nil]);
581 
582  [inputView insertText:@"aaaa"];
583 
584  XCTAssertTrue([inputView canPerformAction:@selector(selectAll:) withSender:nil]);
585 }
586 
587 - (void)testDeletingBackward {
588  NSDictionary* config = self.mutableTemplateCopy;
589  [self setClientId:123 configuration:config];
590  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
591  FlutterTextInputView* inputView = inputFields[0];
592 
593  [inputView insertText:@"ឹ😀 text 🥰👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ‡ºðŸ‡³à¸”ี "];
594  [inputView deleteBackward];
595  [inputView deleteBackward];
596 
597  // Thai vowel is removed.
598  XCTAssertEqualObjects(inputView.text, @"ឹ😀 text 🥰👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ‡ºðŸ‡³à¸”");
599  [inputView deleteBackward];
600  XCTAssertEqualObjects(inputView.text, @"ឹ😀 text 🥰👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ‡ºðŸ‡³");
601  [inputView deleteBackward];
602  XCTAssertEqualObjects(inputView.text, @"ឹ😀 text 🥰👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦");
603  [inputView deleteBackward];
604  XCTAssertEqualObjects(inputView.text, @"ឹ😀 text 🥰");
605  [inputView deleteBackward];
606 
607  XCTAssertEqualObjects(inputView.text, @"ឹ😀 text ");
608  [inputView deleteBackward];
609  [inputView deleteBackward];
610  [inputView deleteBackward];
611  [inputView deleteBackward];
612  [inputView deleteBackward];
613  [inputView deleteBackward];
614 
615  XCTAssertEqualObjects(inputView.text, @"ឹ😀");
616  [inputView deleteBackward];
617  XCTAssertEqualObjects(inputView.text, @"áž¹");
618  [inputView deleteBackward];
619  XCTAssertEqualObjects(inputView.text, @"");
620 }
621 
622 // This tests the workaround to fix an iOS 16 bug
623 // See: https://github.com/flutter/flutter/issues/111494
624 - (void)testSystemOnlyAddingPartialComposedCharacter {
625  NSDictionary* config = self.mutableTemplateCopy;
626  [self setClientId:123 configuration:config];
627  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
628  FlutterTextInputView* inputView = inputFields[0];
629 
630  [inputView insertText:@"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦"];
631  [inputView deleteBackward];
632 
633  // Insert the first unichar in the emoji.
634  [inputView insertText:[@"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦" substringWithRange:NSMakeRange(0, 1)]];
635  [inputView insertText:@"ì•„"];
636 
637  XCTAssertEqualObjects(inputView.text, @"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ì•„");
638 
639  // Deleting ì•„.
640  [inputView deleteBackward];
641  // 👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ should be the current string.
642 
643  [inputView insertText:@"😀"];
644  [inputView deleteBackward];
645  // Insert the first unichar in the emoji.
646  [inputView insertText:[@"😀" substringWithRange:NSMakeRange(0, 1)]];
647  [inputView insertText:@"ì•„"];
648  XCTAssertEqualObjects(inputView.text, @"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ˜€ì•„");
649 
650  // Deleting ì•„.
651  [inputView deleteBackward];
652  // 👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ˜€ should be the current string.
653 
654  [inputView deleteBackward];
655  // Insert the first unichar in the emoji.
656  [inputView insertText:[@"😀" substringWithRange:NSMakeRange(0, 1)]];
657  [inputView insertText:@"ì•„"];
658 
659  XCTAssertEqualObjects(inputView.text, @"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦ðŸ˜€ì•„");
660 }
661 
662 - (void)testCachedComposedCharacterClearedAtKeyboardInteraction {
663  NSDictionary* config = self.mutableTemplateCopy;
664  [self setClientId:123 configuration:config];
665  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
666  FlutterTextInputView* inputView = inputFields[0];
667 
668  [inputView insertText:@"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦"];
669  [inputView deleteBackward];
670  [inputView shouldChangeTextInRange:OCMClassMock([UITextRange class]) replacementText:@""];
671 
672  // Insert the first unichar in the emoji.
673  NSString* brokenEmoji = [@"👨â€ðŸ‘©â€ðŸ‘§â€ðŸ‘¦" substringWithRange:NSMakeRange(0, 1)];
674  [inputView insertText:brokenEmoji];
675  [inputView insertText:@"ì•„"];
676 
677  NSString* finalText = [NSString stringWithFormat:@"%@ì•„", brokenEmoji];
678  XCTAssertEqualObjects(inputView.text, finalText);
679 }
680 
681 - (void)testPastingNonTextDisallowed {
682  NSDictionary* config = self.mutableTemplateCopy;
683  [self setClientId:123 configuration:config];
684  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
685  FlutterTextInputView* inputView = inputFields[0];
686 
687  UIPasteboard.generalPasteboard.color = UIColor.redColor;
688  XCTAssertNil(UIPasteboard.generalPasteboard.string);
689  XCTAssertFalse([inputView canPerformAction:@selector(paste:) withSender:nil]);
690  [inputView paste:nil];
691 
692  XCTAssertEqualObjects(inputView.text, @"");
693 }
694 
695 - (void)testNoZombies {
696  // Regression test for https://github.com/flutter/flutter/issues/62501.
697  FlutterSecureTextInputView* passwordView =
698  [[FlutterSecureTextInputView alloc] initWithOwner:textInputPlugin];
699 
700  @autoreleasepool {
701  // Initialize the lazy textField.
702  [passwordView.textField description];
703  }
704  XCTAssert([[passwordView.textField description] containsString:@"TextField"]);
705 }
706 
707 - (void)testInputViewCrash {
708  FlutterTextInputView* activeView = nil;
709  @autoreleasepool {
710  FlutterEngine* flutterEngine = [[FlutterEngine alloc] init];
711  FlutterTextInputPlugin* inputPlugin = [[FlutterTextInputPlugin alloc]
712  initWithDelegate:(id<FlutterTextInputDelegate>)flutterEngine];
713  activeView = inputPlugin.activeView;
714  }
715  [activeView updateEditingState];
716 }
717 
718 - (void)testDoNotReuseInputViews {
719  NSDictionary* config = self.mutableTemplateCopy;
720  [self setClientId:123 configuration:config];
721  FlutterTextInputView* currentView = textInputPlugin.activeView;
722  [self setClientId:456 configuration:config];
723 
724  XCTAssertNotNil(currentView);
725  XCTAssertNotNil(textInputPlugin.activeView);
726  XCTAssertNotEqual(currentView, textInputPlugin.activeView);
727 }
728 
729 - (void)ensureOnlyActiveViewCanBecomeFirstResponder {
730  for (FlutterTextInputView* inputView in self.installedInputViews) {
731  XCTAssertEqual(inputView.canBecomeFirstResponder, inputView == textInputPlugin.activeView);
732  }
733 }
734 
735 - (void)testPropagatePressEventsToViewController {
736  FlutterViewController* mockViewController = OCMPartialMock(viewController);
737  OCMStub([mockViewController pressesBegan:[OCMArg isNotNil] withEvent:[OCMArg isNotNil]]);
738  OCMStub([mockViewController pressesEnded:[OCMArg isNotNil] withEvent:[OCMArg isNotNil]]);
739 
740  textInputPlugin.viewController = mockViewController;
741 
742  NSDictionary* config = self.mutableTemplateCopy;
743  [self setClientId:123 configuration:config];
744  FlutterTextInputView* currentView = textInputPlugin.activeView;
745  [self setTextInputShow];
746 
747  [currentView pressesBegan:[NSSet setWithObjects:OCMClassMock([UIPress class]), nil]
748  withEvent:OCMClassMock([UIPressesEvent class])];
749 
750  OCMVerify(times(1), [mockViewController pressesBegan:[OCMArg isNotNil]
751  withEvent:[OCMArg isNotNil]]);
752  OCMVerify(times(0), [mockViewController pressesEnded:[OCMArg isNotNil]
753  withEvent:[OCMArg isNotNil]]);
754 
755  [currentView pressesEnded:[NSSet setWithObjects:OCMClassMock([UIPress class]), nil]
756  withEvent:OCMClassMock([UIPressesEvent class])];
757 
758  OCMVerify(times(1), [mockViewController pressesBegan:[OCMArg isNotNil]
759  withEvent:[OCMArg isNotNil]]);
760  OCMVerify(times(1), [mockViewController pressesEnded:[OCMArg isNotNil]
761  withEvent:[OCMArg isNotNil]]);
762 }
763 
764 - (void)testPropagatePressEventsToViewController2 {
765  FlutterViewController* mockViewController = OCMPartialMock(viewController);
766  OCMStub([mockViewController pressesBegan:[OCMArg isNotNil] withEvent:[OCMArg isNotNil]]);
767  OCMStub([mockViewController pressesEnded:[OCMArg isNotNil] withEvent:[OCMArg isNotNil]]);
768 
769  textInputPlugin.viewController = mockViewController;
770 
771  NSDictionary* config = self.mutableTemplateCopy;
772  [self setClientId:123 configuration:config];
773  [self setTextInputShow];
774  FlutterTextInputView* currentView = textInputPlugin.activeView;
775 
776  [currentView pressesBegan:[NSSet setWithObjects:OCMClassMock([UIPress class]), nil]
777  withEvent:OCMClassMock([UIPressesEvent class])];
778 
779  OCMVerify(times(1), [mockViewController pressesBegan:[OCMArg isNotNil]
780  withEvent:[OCMArg isNotNil]]);
781  OCMVerify(times(0), [mockViewController pressesEnded:[OCMArg isNotNil]
782  withEvent:[OCMArg isNotNil]]);
783 
784  // Switch focus to a different view.
785  [self setClientId:321 configuration:config];
786  [self setTextInputShow];
787  NSAssert(textInputPlugin.activeView, @"active view must not be nil");
788  NSAssert(textInputPlugin.activeView != currentView, @"active view must change");
789  currentView = textInputPlugin.activeView;
790  [currentView pressesEnded:[NSSet setWithObjects:OCMClassMock([UIPress class]), nil]
791  withEvent:OCMClassMock([UIPressesEvent class])];
792 
793  OCMVerify(times(1), [mockViewController pressesBegan:[OCMArg isNotNil]
794  withEvent:[OCMArg isNotNil]]);
795  OCMVerify(times(1), [mockViewController pressesEnded:[OCMArg isNotNil]
796  withEvent:[OCMArg isNotNil]]);
797 }
798 
799 - (void)testUpdateSecureTextEntry {
800  NSDictionary* config = self.mutableTemplateCopy;
801  [config setValue:@"YES" forKey:@"obscureText"];
802  [self setClientId:123 configuration:config];
803 
804  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
805  FlutterTextInputView* inputView = OCMPartialMock(inputFields[0]);
806 
807  __block int callCount = 0;
808  OCMStub([inputView reloadInputViews]).andDo(^(NSInvocation* invocation) {
809  callCount++;
810  });
811 
812  XCTAssertTrue(inputView.isSecureTextEntry);
813 
814  config = self.mutableTemplateCopy;
815  [config setValue:@"NO" forKey:@"obscureText"];
816  [self updateConfig:config];
817 
818  XCTAssertEqual(callCount, 1);
819  XCTAssertFalse(inputView.isSecureTextEntry);
820 }
821 
822 - (void)testInputActionContinueAction {
823  id mockBinaryMessenger = OCMClassMock([FlutterBinaryMessengerRelay class]);
824  FlutterEngine* testEngine = [[FlutterEngine alloc] init];
825  [testEngine setBinaryMessenger:mockBinaryMessenger];
826  [testEngine runWithEntrypoint:FlutterDefaultDartEntrypoint initialRoute:@"test"];
827 
828  FlutterTextInputPlugin* inputPlugin =
829  [[FlutterTextInputPlugin alloc] initWithDelegate:(id<FlutterTextInputDelegate>)testEngine];
830  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:inputPlugin];
831 
832  [testEngine flutterTextInputView:inputView
833  performAction:FlutterTextInputActionContinue
834  withClient:123];
835 
836  FlutterMethodCall* methodCall =
837  [FlutterMethodCall methodCallWithMethodName:@"TextInputClient.performAction"
838  arguments:@[ @(123), @"TextInputAction.continueAction" ]];
839  NSData* encodedMethodCall = [[FlutterJSONMethodCodec sharedInstance] encodeMethodCall:methodCall];
840  OCMVerify([mockBinaryMessenger sendOnChannel:@"flutter/textinput" message:encodedMethodCall]);
841 }
842 
843 - (void)testDisablingAutocorrectDisablesSpellChecking {
844  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
845 
846  // Disable the interactive selection.
847  NSDictionary* config = self.mutableTemplateCopy;
848  [inputView configureWithDictionary:config];
849 
850  XCTAssertEqual(inputView.autocorrectionType, UITextAutocorrectionTypeDefault);
851  XCTAssertEqual(inputView.spellCheckingType, UITextSpellCheckingTypeDefault);
852 
853  [config setValue:@(NO) forKey:@"autocorrect"];
854  [inputView configureWithDictionary:config];
855 
856  XCTAssertEqual(inputView.autocorrectionType, UITextAutocorrectionTypeNo);
857  XCTAssertEqual(inputView.spellCheckingType, UITextSpellCheckingTypeNo);
858 }
859 
860 - (void)testReplaceTestLocalAdjustSelectionAndMarkedTextRange {
861  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
862  [inputView setMarkedText:@"test text" selectedRange:NSMakeRange(0, 5)];
863  NSRange selectedTextRange = ((FlutterTextRange*)inputView.selectedTextRange).range;
864  const NSRange markedTextRange = ((FlutterTextRange*)inputView.markedTextRange).range;
865  XCTAssertEqual(selectedTextRange.location, 0ul);
866  XCTAssertEqual(selectedTextRange.length, 5ul);
867  XCTAssertEqual(markedTextRange.location, 0ul);
868  XCTAssertEqual(markedTextRange.length, 9ul);
869 
870  // Replaces space with space.
871  [inputView replaceRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(4, 1)] withText:@" "];
872  selectedTextRange = ((FlutterTextRange*)inputView.selectedTextRange).range;
873 
874  XCTAssertEqual(selectedTextRange.location, 5ul);
875  XCTAssertEqual(selectedTextRange.length, 0ul);
876  XCTAssertEqual(inputView.markedTextRange, nil);
877 }
878 
879 - (void)testFlutterTextInputViewOnlyRespondsToInsertionPointColorBelowIOS17 {
880  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
881  // [UITextInputTraits insertionPointColor] is non-public API, so @selector(insertionPointColor)
882  // would generate a compile-time warning.
883  SEL insertionPointColor = NSSelectorFromString(@"insertionPointColor");
884  BOOL respondsToInsertionPointColor = [inputView respondsToSelector:insertionPointColor];
885  if (@available(iOS 17, *)) {
886  XCTAssertFalse(respondsToInsertionPointColor);
887  } else {
888  XCTAssertTrue(respondsToInsertionPointColor);
889  }
890 }
891 
892 #pragma mark - TextEditingDelta tests
893 - (void)testTextEditingDeltasAreGeneratedOnTextInput {
894  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
895  inputView.enableDeltaModel = YES;
896 
897  __block int updateCount = 0;
898 
899  [inputView insertText:@"text to insert"];
900  OCMExpect(
901  [engine
902  flutterTextInputView:inputView
903  updateEditingClient:0
904  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
905  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
906  isEqualToString:@""]) &&
907  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
908  isEqualToString:@"text to insert"]) &&
909  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] == 0) &&
910  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] == 0);
911  }]])
912  .andDo(^(NSInvocation* invocation) {
913  updateCount++;
914  });
915  XCTAssertEqual(updateCount, 0);
916 
917  [self flushScheduledAsyncBlocks];
918 
919  // Update the framework exactly once.
920  XCTAssertEqual(updateCount, 1);
921 
922  [inputView deleteBackward];
923  OCMExpect([engine flutterTextInputView:inputView
924  updateEditingClient:0
925  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
926  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
927  isEqualToString:@"text to insert"]) &&
928  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
929  isEqualToString:@""]) &&
930  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"]
931  intValue] == 13) &&
932  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"]
933  intValue] == 14);
934  }]])
935  .andDo(^(NSInvocation* invocation) {
936  updateCount++;
937  });
938  [self flushScheduledAsyncBlocks];
939  XCTAssertEqual(updateCount, 2);
940 
941  inputView.selectedTextRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)];
942  OCMExpect([engine flutterTextInputView:inputView
943  updateEditingClient:0
944  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
945  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
946  isEqualToString:@"text to inser"]) &&
947  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
948  isEqualToString:@""]) &&
949  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"]
950  intValue] == -1) &&
951  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"]
952  intValue] == -1);
953  }]])
954  .andDo(^(NSInvocation* invocation) {
955  updateCount++;
956  });
957  [self flushScheduledAsyncBlocks];
958  XCTAssertEqual(updateCount, 3);
959 
960  [inputView replaceRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]
961  withText:@"replace text"];
962  OCMExpect(
963  [engine
964  flutterTextInputView:inputView
965  updateEditingClient:0
966  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
967  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
968  isEqualToString:@"text to inser"]) &&
969  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
970  isEqualToString:@"replace text"]) &&
971  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] == 0) &&
972  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] == 1);
973  }]])
974  .andDo(^(NSInvocation* invocation) {
975  updateCount++;
976  });
977  [self flushScheduledAsyncBlocks];
978  XCTAssertEqual(updateCount, 4);
979 
980  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
981  OCMExpect([engine flutterTextInputView:inputView
982  updateEditingClient:0
983  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
984  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
985  isEqualToString:@"replace textext to inser"]) &&
986  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
987  isEqualToString:@"marked text"]) &&
988  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"]
989  intValue] == 12) &&
990  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"]
991  intValue] == 12);
992  }]])
993  .andDo(^(NSInvocation* invocation) {
994  updateCount++;
995  });
996  [self flushScheduledAsyncBlocks];
997  XCTAssertEqual(updateCount, 5);
998 
999  [inputView unmarkText];
1000  OCMExpect([engine
1001  flutterTextInputView:inputView
1002  updateEditingClient:0
1003  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1004  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
1005  isEqualToString:@"replace textmarked textext to inser"]) &&
1006  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
1007  isEqualToString:@""]) &&
1008  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] ==
1009  -1) &&
1010  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] ==
1011  -1);
1012  }]])
1013  .andDo(^(NSInvocation* invocation) {
1014  updateCount++;
1015  });
1016  [self flushScheduledAsyncBlocks];
1017 
1018  XCTAssertEqual(updateCount, 6);
1019  OCMVerifyAll(engine);
1020 }
1021 
1022 - (void)testTextEditingDeltasAreBatchedAndForwardedToFramework {
1023  // Setup
1024  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1025  inputView.enableDeltaModel = YES;
1026 
1027  // Expected call.
1028  OCMExpect([engine flutterTextInputView:inputView
1029  updateEditingClient:0
1030  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1031  NSArray* deltas = state[@"deltas"];
1032  NSDictionary* firstDelta = deltas[0];
1033  NSDictionary* secondDelta = deltas[1];
1034  NSDictionary* thirdDelta = deltas[2];
1035  return [firstDelta[@"oldText"] isEqualToString:@""] &&
1036  [firstDelta[@"deltaText"] isEqualToString:@"-"] &&
1037  [firstDelta[@"deltaStart"] intValue] == 0 &&
1038  [firstDelta[@"deltaEnd"] intValue] == 0 &&
1039  [secondDelta[@"oldText"] isEqualToString:@"-"] &&
1040  [secondDelta[@"deltaText"] isEqualToString:@""] &&
1041  [secondDelta[@"deltaStart"] intValue] == 0 &&
1042  [secondDelta[@"deltaEnd"] intValue] == 1 &&
1043  [thirdDelta[@"oldText"] isEqualToString:@""] &&
1044  [thirdDelta[@"deltaText"] isEqualToString:@"—"] &&
1045  [thirdDelta[@"deltaStart"] intValue] == 0 &&
1046  [thirdDelta[@"deltaEnd"] intValue] == 0;
1047  }]]);
1048 
1049  // Simulate user input.
1050  [inputView insertText:@"-"];
1051  [inputView deleteBackward];
1052  [inputView insertText:@"—"];
1053 
1054  [self flushScheduledAsyncBlocks];
1055  OCMVerifyAll(engine);
1056 }
1057 
1058 - (void)testTextEditingDeltasAreGeneratedOnSetMarkedTextReplacement {
1059  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1060  inputView.enableDeltaModel = YES;
1061 
1062  __block int updateCount = 0;
1063  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withDelta:[OCMArg isNotNil]])
1064  .andDo(^(NSInvocation* invocation) {
1065  updateCount++;
1066  });
1067 
1068  [inputView.text setString:@"Some initial text"];
1069  XCTAssertEqual(updateCount, 0);
1070 
1071  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(13, 4)];
1072  inputView.markedTextRange = range;
1073  inputView.selectedTextRange = nil;
1074  [self flushScheduledAsyncBlocks];
1075  XCTAssertEqual(updateCount, 1);
1076 
1077  [inputView setMarkedText:@"new marked text." selectedRange:NSMakeRange(0, 1)];
1078  OCMVerify([engine
1079  flutterTextInputView:inputView
1080  updateEditingClient:0
1081  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1082  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
1083  isEqualToString:@"Some initial text"]) &&
1084  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
1085  isEqualToString:@"new marked text."]) &&
1086  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] == 13) &&
1087  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] == 17);
1088  }]]);
1089  [self flushScheduledAsyncBlocks];
1090  XCTAssertEqual(updateCount, 2);
1091 }
1092 
1093 - (void)testTextEditingDeltasAreGeneratedOnSetMarkedTextInsertion {
1094  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1095  inputView.enableDeltaModel = YES;
1096 
1097  __block int updateCount = 0;
1098  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withDelta:[OCMArg isNotNil]])
1099  .andDo(^(NSInvocation* invocation) {
1100  updateCount++;
1101  });
1102 
1103  [inputView.text setString:@"Some initial text"];
1104  [self flushScheduledAsyncBlocks];
1105  XCTAssertEqual(updateCount, 0);
1106 
1107  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(13, 4)];
1108  inputView.markedTextRange = range;
1109  inputView.selectedTextRange = nil;
1110  [self flushScheduledAsyncBlocks];
1111  XCTAssertEqual(updateCount, 1);
1112 
1113  [inputView setMarkedText:@"text." selectedRange:NSMakeRange(0, 1)];
1114  OCMVerify([engine
1115  flutterTextInputView:inputView
1116  updateEditingClient:0
1117  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1118  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
1119  isEqualToString:@"Some initial text"]) &&
1120  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
1121  isEqualToString:@"text."]) &&
1122  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] == 13) &&
1123  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] == 17);
1124  }]]);
1125  [self flushScheduledAsyncBlocks];
1126  XCTAssertEqual(updateCount, 2);
1127 }
1128 
1129 - (void)testTextEditingDeltasAreGeneratedOnSetMarkedTextDeletion {
1130  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1131  inputView.enableDeltaModel = YES;
1132 
1133  __block int updateCount = 0;
1134  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withDelta:[OCMArg isNotNil]])
1135  .andDo(^(NSInvocation* invocation) {
1136  updateCount++;
1137  });
1138 
1139  [inputView.text setString:@"Some initial text"];
1140  [self flushScheduledAsyncBlocks];
1141  XCTAssertEqual(updateCount, 0);
1142 
1143  UITextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(13, 4)];
1144  inputView.markedTextRange = range;
1145  inputView.selectedTextRange = nil;
1146  [self flushScheduledAsyncBlocks];
1147  XCTAssertEqual(updateCount, 1);
1148 
1149  [inputView setMarkedText:@"tex" selectedRange:NSMakeRange(0, 1)];
1150  OCMVerify([engine
1151  flutterTextInputView:inputView
1152  updateEditingClient:0
1153  withDelta:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1154  return ([[state[@"deltas"] objectAtIndex:0][@"oldText"]
1155  isEqualToString:@"Some initial text"]) &&
1156  ([[state[@"deltas"] objectAtIndex:0][@"deltaText"]
1157  isEqualToString:@"tex"]) &&
1158  ([[state[@"deltas"] objectAtIndex:0][@"deltaStart"] intValue] == 13) &&
1159  ([[state[@"deltas"] objectAtIndex:0][@"deltaEnd"] intValue] == 17);
1160  }]]);
1161  [self flushScheduledAsyncBlocks];
1162  XCTAssertEqual(updateCount, 2);
1163 }
1164 
1165 #pragma mark - EditingState tests
1166 
1167 - (void)testUITextInputCallsUpdateEditingStateOnce {
1168  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1169 
1170  __block int updateCount = 0;
1171  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withState:[OCMArg isNotNil]])
1172  .andDo(^(NSInvocation* invocation) {
1173  updateCount++;
1174  });
1175 
1176  [inputView insertText:@"text to insert"];
1177  // Update the framework exactly once.
1178  XCTAssertEqual(updateCount, 1);
1179 
1180  [inputView deleteBackward];
1181  XCTAssertEqual(updateCount, 2);
1182 
1183  inputView.selectedTextRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)];
1184  XCTAssertEqual(updateCount, 3);
1185 
1186  [inputView replaceRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]
1187  withText:@"replace text"];
1188  XCTAssertEqual(updateCount, 4);
1189 
1190  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1191  XCTAssertEqual(updateCount, 5);
1192 
1193  [inputView unmarkText];
1194  XCTAssertEqual(updateCount, 6);
1195 }
1196 
1197 - (void)testUITextInputCallsUpdateEditingStateWithDeltaOnce {
1198  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1199  inputView.enableDeltaModel = YES;
1200 
1201  __block int updateCount = 0;
1202  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withDelta:[OCMArg isNotNil]])
1203  .andDo(^(NSInvocation* invocation) {
1204  updateCount++;
1205  });
1206 
1207  [inputView insertText:@"text to insert"];
1208  [self flushScheduledAsyncBlocks];
1209  // Update the framework exactly once.
1210  XCTAssertEqual(updateCount, 1);
1211 
1212  [inputView deleteBackward];
1213  [self flushScheduledAsyncBlocks];
1214  XCTAssertEqual(updateCount, 2);
1215 
1216  inputView.selectedTextRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)];
1217  [self flushScheduledAsyncBlocks];
1218  XCTAssertEqual(updateCount, 3);
1219 
1220  [inputView replaceRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)]
1221  withText:@"replace text"];
1222  [self flushScheduledAsyncBlocks];
1223  XCTAssertEqual(updateCount, 4);
1224 
1225  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1226  [self flushScheduledAsyncBlocks];
1227  XCTAssertEqual(updateCount, 5);
1228 
1229  [inputView unmarkText];
1230  [self flushScheduledAsyncBlocks];
1231  XCTAssertEqual(updateCount, 6);
1232 }
1233 
1234 - (void)testTextChangesDoNotTriggerUpdateEditingClient {
1235  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1236 
1237  __block int updateCount = 0;
1238  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withState:[OCMArg isNotNil]])
1239  .andDo(^(NSInvocation* invocation) {
1240  updateCount++;
1241  });
1242 
1243  [inputView.text setString:@"BEFORE"];
1244  XCTAssertEqual(updateCount, 0);
1245 
1246  inputView.markedTextRange = nil;
1247  inputView.selectedTextRange = nil;
1248  XCTAssertEqual(updateCount, 1);
1249 
1250  // Text changes don't trigger an update.
1251  XCTAssertEqual(updateCount, 1);
1252  [inputView setTextInputState:@{@"text" : @"AFTER"}];
1253  XCTAssertEqual(updateCount, 1);
1254  [inputView setTextInputState:@{@"text" : @"AFTER"}];
1255  XCTAssertEqual(updateCount, 1);
1256 
1257  // Selection changes don't trigger an update.
1258  [inputView
1259  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @0, @"selectionExtent" : @3}];
1260  XCTAssertEqual(updateCount, 1);
1261  [inputView
1262  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @1, @"selectionExtent" : @3}];
1263  XCTAssertEqual(updateCount, 1);
1264 
1265  // Composing region changes don't trigger an update.
1266  [inputView
1267  setTextInputState:@{@"text" : @"COMPOSING", @"composingBase" : @1, @"composingExtent" : @2}];
1268  XCTAssertEqual(updateCount, 1);
1269  [inputView
1270  setTextInputState:@{@"text" : @"COMPOSING", @"composingBase" : @1, @"composingExtent" : @3}];
1271  XCTAssertEqual(updateCount, 1);
1272 }
1273 
1274 - (void)testTextChangesDoNotTriggerUpdateEditingClientWithDelta {
1275  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1276  inputView.enableDeltaModel = YES;
1277 
1278  __block int updateCount = 0;
1279  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withDelta:[OCMArg isNotNil]])
1280  .andDo(^(NSInvocation* invocation) {
1281  updateCount++;
1282  });
1283 
1284  [inputView.text setString:@"BEFORE"];
1285  [self flushScheduledAsyncBlocks];
1286  XCTAssertEqual(updateCount, 0);
1287 
1288  inputView.markedTextRange = nil;
1289  inputView.selectedTextRange = nil;
1290  [self flushScheduledAsyncBlocks];
1291  XCTAssertEqual(updateCount, 1);
1292 
1293  // Text changes don't trigger an update.
1294  XCTAssertEqual(updateCount, 1);
1295  [inputView setTextInputState:@{@"text" : @"AFTER"}];
1296  [self flushScheduledAsyncBlocks];
1297  XCTAssertEqual(updateCount, 1);
1298 
1299  [inputView setTextInputState:@{@"text" : @"AFTER"}];
1300  [self flushScheduledAsyncBlocks];
1301  XCTAssertEqual(updateCount, 1);
1302 
1303  // Selection changes don't trigger an update.
1304  [inputView
1305  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @0, @"selectionExtent" : @3}];
1306  [self flushScheduledAsyncBlocks];
1307  XCTAssertEqual(updateCount, 1);
1308 
1309  [inputView
1310  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @1, @"selectionExtent" : @3}];
1311  [self flushScheduledAsyncBlocks];
1312  XCTAssertEqual(updateCount, 1);
1313 
1314  // Composing region changes don't trigger an update.
1315  [inputView
1316  setTextInputState:@{@"text" : @"COMPOSING", @"composingBase" : @1, @"composingExtent" : @2}];
1317  [self flushScheduledAsyncBlocks];
1318  XCTAssertEqual(updateCount, 1);
1319 
1320  [inputView
1321  setTextInputState:@{@"text" : @"COMPOSING", @"composingBase" : @1, @"composingExtent" : @3}];
1322  [self flushScheduledAsyncBlocks];
1323  XCTAssertEqual(updateCount, 1);
1324 }
1325 
1326 - (void)testUITextInputAvoidUnnecessaryUndateEditingClientCalls {
1327  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1328 
1329  __block int updateCount = 0;
1330  OCMStub([engine flutterTextInputView:inputView updateEditingClient:0 withState:[OCMArg isNotNil]])
1331  .andDo(^(NSInvocation* invocation) {
1332  updateCount++;
1333  });
1334 
1335  [inputView unmarkText];
1336  // updateEditingClient shouldn't fire as the text is already unmarked.
1337  XCTAssertEqual(updateCount, 0);
1338 
1339  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1340  // updateEditingClient fires in response to setMarkedText.
1341  XCTAssertEqual(updateCount, 1);
1342 
1343  [inputView unmarkText];
1344  // updateEditingClient fires in response to unmarkText.
1345  XCTAssertEqual(updateCount, 2);
1346 }
1347 
1348 - (void)testCanCopyPasteWithScribbleEnabled {
1349  if (@available(iOS 14.0, *)) {
1350  NSDictionary* config = self.mutableTemplateCopy;
1351  [self setClientId:123 configuration:config];
1352  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
1353  FlutterTextInputView* inputView = inputFields[0];
1354 
1355  FlutterTextInputView* mockInputView = OCMPartialMock(inputView);
1356  OCMStub([mockInputView isScribbleAvailable]).andReturn(YES);
1357 
1358  [mockInputView insertText:@"aaaa"];
1359  [mockInputView selectAll:nil];
1360 
1361  XCTAssertTrue([mockInputView canPerformAction:@selector(copy:) withSender:NULL]);
1362  XCTAssertTrue([mockInputView canPerformAction:@selector(copy:) withSender:@"sender"]);
1363  XCTAssertFalse([mockInputView canPerformAction:@selector(paste:) withSender:NULL]);
1364  XCTAssertFalse([mockInputView canPerformAction:@selector(paste:) withSender:@"sender"]);
1365 
1366  [mockInputView copy:NULL];
1367  XCTAssertTrue([mockInputView canPerformAction:@selector(copy:) withSender:NULL]);
1368  XCTAssertTrue([mockInputView canPerformAction:@selector(copy:) withSender:@"sender"]);
1369  XCTAssertTrue([mockInputView canPerformAction:@selector(paste:) withSender:NULL]);
1370  XCTAssertTrue([mockInputView canPerformAction:@selector(paste:) withSender:@"sender"]);
1371  }
1372 }
1373 
1374 - (void)testSetMarkedTextDuringScribbleDoesNotTriggerUpdateEditingClient {
1375  if (@available(iOS 14.0, *)) {
1376  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1377 
1378  __block int updateCount = 0;
1379  OCMStub([engine flutterTextInputView:inputView
1380  updateEditingClient:0
1381  withState:[OCMArg isNotNil]])
1382  .andDo(^(NSInvocation* invocation) {
1383  updateCount++;
1384  });
1385 
1386  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1387  // updateEditingClient fires in response to setMarkedText.
1388  XCTAssertEqual(updateCount, 1);
1389 
1390  UIScribbleInteraction* scribbleInteraction =
1391  [[UIScribbleInteraction alloc] initWithDelegate:inputView];
1392 
1393  [inputView scribbleInteractionWillBeginWriting:scribbleInteraction];
1394  [inputView setMarkedText:@"during writing" selectedRange:NSMakeRange(1, 2)];
1395  // updateEditingClient does not fire in response to setMarkedText during a scribble interaction.
1396  XCTAssertEqual(updateCount, 1);
1397 
1398  [inputView scribbleInteractionDidFinishWriting:scribbleInteraction];
1399  [inputView resetScribbleInteractionStatusIfEnding];
1400  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1401  // updateEditingClient fires in response to setMarkedText.
1402  XCTAssertEqual(updateCount, 2);
1403 
1404  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusFocusing;
1405  [inputView setMarkedText:@"during focus" selectedRange:NSMakeRange(1, 2)];
1406  // updateEditingClient does not fire in response to setMarkedText during a scribble-initiated
1407  // focus.
1408  XCTAssertEqual(updateCount, 2);
1409 
1410  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusFocused;
1411  [inputView setMarkedText:@"after focus" selectedRange:NSMakeRange(2, 3)];
1412  // updateEditingClient does not fire in response to setMarkedText after a scribble-initiated
1413  // focus.
1414  XCTAssertEqual(updateCount, 2);
1415 
1416  inputView.scribbleFocusStatus = FlutterScribbleFocusStatusUnfocused;
1417  [inputView setMarkedText:@"marked text" selectedRange:NSMakeRange(0, 1)];
1418  // updateEditingClient fires in response to setMarkedText.
1419  XCTAssertEqual(updateCount, 3);
1420  }
1421 }
1422 
1423 - (void)testUpdateEditingClientNegativeSelection {
1424  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1425 
1426  [inputView.text setString:@"SELECTION"];
1427  inputView.markedTextRange = nil;
1428  inputView.selectedTextRange = nil;
1429 
1430  [inputView setTextInputState:@{
1431  @"text" : @"SELECTION",
1432  @"selectionBase" : @-1,
1433  @"selectionExtent" : @-1
1434  }];
1435  [inputView updateEditingState];
1436  OCMVerify([engine flutterTextInputView:inputView
1437  updateEditingClient:0
1438  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1439  return ([state[@"selectionBase"] intValue]) == 0 &&
1440  ([state[@"selectionExtent"] intValue] == 0);
1441  }]]);
1442 
1443  // Returns (0, 0) when either end goes below 0.
1444  [inputView
1445  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @-1, @"selectionExtent" : @1}];
1446  [inputView updateEditingState];
1447  OCMVerify([engine flutterTextInputView:inputView
1448  updateEditingClient:0
1449  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1450  return ([state[@"selectionBase"] intValue]) == 0 &&
1451  ([state[@"selectionExtent"] intValue] == 0);
1452  }]]);
1453 
1454  [inputView
1455  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @1, @"selectionExtent" : @-1}];
1456  [inputView updateEditingState];
1457  OCMVerify([engine flutterTextInputView:inputView
1458  updateEditingClient:0
1459  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1460  return ([state[@"selectionBase"] intValue]) == 0 &&
1461  ([state[@"selectionExtent"] intValue] == 0);
1462  }]]);
1463 }
1464 
1465 - (void)testUpdateEditingClientSelectionClamping {
1466  // Regression test for https://github.com/flutter/flutter/issues/62992.
1467  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1468 
1469  [inputView.text setString:@"SELECTION"];
1470  inputView.markedTextRange = nil;
1471  inputView.selectedTextRange = nil;
1472 
1473  [inputView
1474  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @0, @"selectionExtent" : @0}];
1475  [inputView updateEditingState];
1476  OCMVerify([engine flutterTextInputView:inputView
1477  updateEditingClient:0
1478  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1479  return ([state[@"selectionBase"] intValue]) == 0 &&
1480  ([state[@"selectionExtent"] intValue] == 0);
1481  }]]);
1482 
1483  // Needs clamping.
1484  [inputView setTextInputState:@{
1485  @"text" : @"SELECTION",
1486  @"selectionBase" : @0,
1487  @"selectionExtent" : @9999
1488  }];
1489  [inputView updateEditingState];
1490 
1491  OCMVerify([engine flutterTextInputView:inputView
1492  updateEditingClient:0
1493  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1494  return ([state[@"selectionBase"] intValue]) == 0 &&
1495  ([state[@"selectionExtent"] intValue] == 9);
1496  }]]);
1497 
1498  // No clamping needed, but in reverse direction.
1499  [inputView
1500  setTextInputState:@{@"text" : @"SELECTION", @"selectionBase" : @1, @"selectionExtent" : @0}];
1501  [inputView updateEditingState];
1502  OCMVerify([engine flutterTextInputView:inputView
1503  updateEditingClient:0
1504  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1505  return ([state[@"selectionBase"] intValue]) == 0 &&
1506  ([state[@"selectionExtent"] intValue] == 1);
1507  }]]);
1508 
1509  // Both ends need clamping.
1510  [inputView setTextInputState:@{
1511  @"text" : @"SELECTION",
1512  @"selectionBase" : @9999,
1513  @"selectionExtent" : @9999
1514  }];
1515  [inputView updateEditingState];
1516  OCMVerify([engine flutterTextInputView:inputView
1517  updateEditingClient:0
1518  withState:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
1519  return ([state[@"selectionBase"] intValue]) == 9 &&
1520  ([state[@"selectionExtent"] intValue] == 9);
1521  }]]);
1522 }
1523 
1524 - (void)testInputViewsHasNonNilInputDelegate {
1525  if (@available(iOS 13.0, *)) {
1526  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1527  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
1528 
1529  [inputView setTextInputClient:123];
1530  [inputView reloadInputViews];
1531  [inputView becomeFirstResponder];
1532  NSAssert(inputView.isFirstResponder, @"inputView is not first responder");
1533  inputView.inputDelegate = nil;
1534 
1535  FlutterTextInputView* mockInputView = OCMPartialMock(inputView);
1536  [mockInputView setTextInputState:@{
1537  @"text" : @"COMPOSING",
1538  @"composingBase" : @1,
1539  @"composingExtent" : @3
1540  }];
1541  OCMVerify([mockInputView setInputDelegate:[OCMArg isNotNil]]);
1542  [inputView removeFromSuperview];
1543  }
1544 }
1545 
1546 - (void)testInputViewsDoNotHaveUITextInteractions {
1547  if (@available(iOS 13.0, *)) {
1548  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1549  BOOL hasTextInteraction = NO;
1550  for (id interaction in inputView.interactions) {
1551  hasTextInteraction = [interaction isKindOfClass:[UITextInteraction class]];
1552  if (hasTextInteraction) {
1553  break;
1554  }
1555  }
1556  XCTAssertFalse(hasTextInteraction);
1557  }
1558 }
1559 
1560 #pragma mark - UITextInput methods - Tests
1561 
1562 - (void)testUpdateFirstRectForRange {
1563  [self setClientId:123 configuration:self.mutableTemplateCopy];
1564 
1565  FlutterTextInputView* inputView = textInputPlugin.activeView;
1566  textInputPlugin.viewController.view.frame = CGRectMake(0, 0, 0, 0);
1567 
1568  [inputView
1569  setTextInputState:@{@"text" : @"COMPOSING", @"composingBase" : @1, @"composingExtent" : @3}];
1570 
1571  CGRect kInvalidFirstRect = CGRectMake(-1, -1, 9999, 9999);
1572  FlutterTextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)];
1573  // yOffset = 200.
1574  NSArray* yOffsetMatrix = @[ @1, @0, @0, @0, @0, @1, @0, @0, @0, @0, @1, @0, @0, @200, @0, @1 ];
1575  NSArray* zeroMatrix = @[ @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0, @0 ];
1576  // This matrix can be generated by running this dart code snippet:
1577  // Matrix4.identity()..scale(3.0)..rotateZ(math.pi/2)..translate(1.0, 2.0,
1578  // 3.0);
1579  NSArray* affineMatrix = @[
1580  @(0.0), @(3.0), @(0.0), @(0.0), @(-3.0), @(0.0), @(0.0), @(0.0), @(0.0), @(0.0), @(3.0), @(0.0),
1581  @(-6.0), @(3.0), @(9.0), @(1.0)
1582  ];
1583 
1584  // Invalid since we don't have the transform or the rect.
1585  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1586 
1587  [inputView setEditableTransform:yOffsetMatrix];
1588  // Invalid since we don't have the rect.
1589  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1590 
1591  // Valid rect and transform.
1592  CGRect testRect = CGRectMake(0, 0, 100, 100);
1593  [inputView setMarkedRect:testRect];
1594 
1595  CGRect finalRect = CGRectOffset(testRect, 0, 200);
1596  XCTAssertTrue(CGRectEqualToRect(finalRect, [inputView firstRectForRange:range]));
1597  // Idempotent.
1598  XCTAssertTrue(CGRectEqualToRect(finalRect, [inputView firstRectForRange:range]));
1599 
1600  // Use an invalid matrix:
1601  [inputView setEditableTransform:zeroMatrix];
1602  // Invalid matrix is invalid.
1603  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1604  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1605 
1606  // Revert the invalid matrix change.
1607  [inputView setEditableTransform:yOffsetMatrix];
1608  [inputView setMarkedRect:testRect];
1609  XCTAssertTrue(CGRectEqualToRect(finalRect, [inputView firstRectForRange:range]));
1610 
1611  // Use an invalid rect:
1612  [inputView setMarkedRect:kInvalidFirstRect];
1613  // Invalid marked rect is invalid.
1614  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1615  XCTAssertTrue(CGRectEqualToRect(kInvalidFirstRect, [inputView firstRectForRange:range]));
1616 
1617  // Use a 3d affine transform that does 3d-scaling, z-index rotating and 3d translation.
1618  [inputView setEditableTransform:affineMatrix];
1619  [inputView setMarkedRect:testRect];
1620  XCTAssertTrue(
1621  CGRectEqualToRect(CGRectMake(-306, 3, 300, 300), [inputView firstRectForRange:range]));
1622 
1623  NSAssert(inputView.superview, @"inputView is not in the view hierarchy!");
1624  const CGPoint offset = CGPointMake(113, 119);
1625  CGRect currentFrame = inputView.frame;
1626  currentFrame.origin = offset;
1627  inputView.frame = currentFrame;
1628  // Moving the input view within the FlutterView shouldn't affect the coordinates,
1629  // since the framework sends us global coordinates.
1630  XCTAssertTrue(CGRectEqualToRect(CGRectMake(-306 - 113, 3 - 119, 300, 300),
1631  [inputView firstRectForRange:range]));
1632 }
1633 
1634 - (void)testFirstRectForRangeReturnsNoneZeroRectWhenScribbleIsEnabled {
1635  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1636  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1637 
1638  FlutterTextInputView* mockInputView = OCMPartialMock(inputView);
1639  OCMStub([mockInputView isScribbleAvailable]).andReturn(YES);
1640 
1641  [inputView setSelectionRects:@[
1642  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1643  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:1U],
1644  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1645  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1646  ]];
1647 
1648  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 3)];
1649 
1650  if (@available(iOS 17, *)) {
1651  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 300, 100),
1652  [inputView firstRectForRange:multiRectRange]));
1653  } else {
1654  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 100, 100),
1655  [inputView firstRectForRange:multiRectRange]));
1656  }
1657 }
1658 
1659 - (void)testFirstRectForRangeReturnsCorrectRectOnASingleLineLeftToRight {
1660  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1661  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1662 
1663  [inputView setSelectionRects:@[
1664  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1665  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:1U],
1666  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1667  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1668  ]];
1669  FlutterTextRange* singleRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 1)];
1670  if (@available(iOS 17, *)) {
1671  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 100, 100),
1672  [inputView firstRectForRange:singleRectRange]));
1673  } else {
1674  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:singleRectRange]));
1675  }
1676 
1677  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 3)];
1678 
1679  if (@available(iOS 17, *)) {
1680  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 300, 100),
1681  [inputView firstRectForRange:multiRectRange]));
1682  } else {
1683  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1684  }
1685 
1686  [inputView setTextInputState:@{@"text" : @"COM"}];
1687  FlutterTextRange* rangeOutsideBounds = [FlutterTextRange rangeWithNSRange:NSMakeRange(3, 1)];
1688  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:rangeOutsideBounds]));
1689 }
1690 
1691 - (void)testFirstRectForRangeReturnsCorrectRectOnASingleLineRightToLeft {
1692  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1693  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1694 
1695  [inputView setSelectionRects:@[
1696  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:0U],
1697  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:1U],
1698  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:2U],
1699  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:3U],
1700  ]];
1701  FlutterTextRange* singleRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 1)];
1702  if (@available(iOS 17, *)) {
1703  XCTAssertTrue(CGRectEqualToRect(CGRectMake(200, 0, 100, 100),
1704  [inputView firstRectForRange:singleRectRange]));
1705  } else {
1706  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:singleRectRange]));
1707  }
1708 
1709  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 3)];
1710  if (@available(iOS 17, *)) {
1711  XCTAssertTrue(CGRectEqualToRect(CGRectMake(0, 0, 300, 100),
1712  [inputView firstRectForRange:multiRectRange]));
1713  } else {
1714  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1715  }
1716 
1717  [inputView setTextInputState:@{@"text" : @"COM"}];
1718  FlutterTextRange* rangeOutsideBounds = [FlutterTextRange rangeWithNSRange:NSMakeRange(3, 1)];
1719  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:rangeOutsideBounds]));
1720 }
1721 
1722 - (void)testFirstRectForRangeReturnsCorrectRectOnMultipleLinesLeftToRight {
1723  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1724  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1725 
1726  [inputView setSelectionRects:@[
1727  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1728  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:1U],
1729  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1730  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1731  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:4U],
1732  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:5U],
1733  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:6U],
1734  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 100, 100, 100) position:7U],
1735  ]];
1736  FlutterTextRange* singleRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 1)];
1737  if (@available(iOS 17, *)) {
1738  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 100, 100),
1739  [inputView firstRectForRange:singleRectRange]));
1740  } else {
1741  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:singleRectRange]));
1742  }
1743 
1744  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1745 
1746  if (@available(iOS 17, *)) {
1747  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 300, 100),
1748  [inputView firstRectForRange:multiRectRange]));
1749  } else {
1750  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1751  }
1752 }
1753 
1754 - (void)testFirstRectForRangeReturnsCorrectRectOnMultipleLinesRightToLeft {
1755  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1756  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1757 
1758  [inputView setSelectionRects:@[
1759  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:0U],
1760  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:1U],
1761  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:2U],
1762  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:3U],
1763  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 100, 100, 100) position:4U],
1764  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:5U],
1765  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:6U],
1766  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:7U],
1767  ]];
1768  FlutterTextRange* singleRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 1)];
1769  if (@available(iOS 17, *)) {
1770  XCTAssertTrue(CGRectEqualToRect(CGRectMake(200, 0, 100, 100),
1771  [inputView firstRectForRange:singleRectRange]));
1772  } else {
1773  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:singleRectRange]));
1774  }
1775 
1776  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1777  if (@available(iOS 17, *)) {
1778  XCTAssertTrue(CGRectEqualToRect(CGRectMake(0, 0, 300, 100),
1779  [inputView firstRectForRange:multiRectRange]));
1780  } else {
1781  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1782  }
1783 }
1784 
1785 - (void)testFirstRectForRangeReturnsCorrectRectOnSingleLineWithVaryingMinYAndMaxYLeftToRight {
1786  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1787  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1788 
1789  [inputView setSelectionRects:@[
1790  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1791  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 10, 100, 80)
1792  position:1U], // shorter
1793  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, -10, 100, 120)
1794  position:2U], // taller
1795  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1796  ]];
1797 
1798  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 3)];
1799 
1800  if (@available(iOS 17, *)) {
1801  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, -10, 300, 120),
1802  [inputView firstRectForRange:multiRectRange]));
1803  } else {
1804  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1805  }
1806 }
1807 
1808 - (void)testFirstRectForRangeReturnsCorrectRectOnSingleLineWithVaryingMinYAndMaxYRightToLeft {
1809  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1810  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1811 
1812  [inputView setSelectionRects:@[
1813  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:0U],
1814  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, -10, 100, 120)
1815  position:1U], // taller
1816  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 10, 100, 80)
1817  position:2U], // shorter
1818  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:3U],
1819  ]];
1820 
1821  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 3)];
1822 
1823  if (@available(iOS 17, *)) {
1824  XCTAssertTrue(CGRectEqualToRect(CGRectMake(0, -10, 300, 120),
1825  [inputView firstRectForRange:multiRectRange]));
1826  } else {
1827  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1828  }
1829 }
1830 
1831 - (void)testFirstRectForRangeReturnsCorrectRectWithOverlappingRectsExceedingThresholdLeftToRight {
1832  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1833  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1834 
1835  [inputView setSelectionRects:@[
1836  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1837  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:1U],
1838  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1839  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1840  // y=60 exceeds threshold, so treat it as a new line.
1841  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 60, 100, 100) position:4U],
1842  ]];
1843 
1844  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1845 
1846  if (@available(iOS 17, *)) {
1847  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 300, 100),
1848  [inputView firstRectForRange:multiRectRange]));
1849  } else {
1850  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1851  }
1852 }
1853 
1854 - (void)testFirstRectForRangeReturnsCorrectRectWithOverlappingRectsExceedingThresholdRightToLeft {
1855  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1856  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1857 
1858  [inputView setSelectionRects:@[
1859  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:0U],
1860  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:1U],
1861  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:2U],
1862  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:3U],
1863  // y=60 exceeds threshold, so treat it as a new line.
1864  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 60, 100, 100) position:4U],
1865  ]];
1866 
1867  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1868 
1869  if (@available(iOS 17, *)) {
1870  XCTAssertTrue(CGRectEqualToRect(CGRectMake(0, 0, 300, 100),
1871  [inputView firstRectForRange:multiRectRange]));
1872  } else {
1873  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1874  }
1875 }
1876 
1877 - (void)testFirstRectForRangeReturnsCorrectRectWithOverlappingRectsWithinThresholdLeftToRight {
1878  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1879  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1880 
1881  [inputView setSelectionRects:@[
1882  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1883  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:1U],
1884  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1885  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:3U],
1886  // y=40 is within line threshold, so treat it as the same line
1887  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(400, 40, 100, 100) position:4U],
1888  ]];
1889 
1890  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1891 
1892  if (@available(iOS 17, *)) {
1893  XCTAssertTrue(CGRectEqualToRect(CGRectMake(100, 0, 400, 140),
1894  [inputView firstRectForRange:multiRectRange]));
1895  } else {
1896  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1897  }
1898 }
1899 
1900 - (void)testFirstRectForRangeReturnsCorrectRectWithOverlappingRectsWithinThresholdRightToLeft {
1901  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1902  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1903 
1904  [inputView setSelectionRects:@[
1905  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(400, 0, 100, 100) position:0U],
1906  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 0, 100, 100) position:1U],
1907  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1908  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100) position:3U],
1909  // y=40 is within line threshold, so treat it as the same line
1910  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 40, 100, 100) position:4U],
1911  ]];
1912 
1913  FlutterTextRange* multiRectRange = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 4)];
1914 
1915  if (@available(iOS 17, *)) {
1916  XCTAssertTrue(CGRectEqualToRect(CGRectMake(0, 0, 400, 140),
1917  [inputView firstRectForRange:multiRectRange]));
1918  } else {
1919  XCTAssertTrue(CGRectEqualToRect(CGRectZero, [inputView firstRectForRange:multiRectRange]));
1920  }
1921 }
1922 
1923 - (void)testClosestPositionToPoint {
1924  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
1925  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
1926 
1927  // Minimize the vertical distance from the center of the rects first
1928  [inputView setSelectionRects:@[
1929  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1930  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
1931  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 200, 100, 100) position:2U],
1932  ]];
1933  CGPoint point = CGPointMake(150, 150);
1934  XCTAssertEqual(2U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1935  XCTAssertEqual(UITextStorageDirectionBackward,
1936  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1937 
1938  // Then, if the point is above the bottom of the closest rects vertically, get the closest x
1939  // origin
1940  [inputView setSelectionRects:@[
1941  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1942  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
1943  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:2U],
1944  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:3U],
1945  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 200, 100, 100) position:4U],
1946  ]];
1947  point = CGPointMake(125, 150);
1948  XCTAssertEqual(2U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1949  XCTAssertEqual(UITextStorageDirectionForward,
1950  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1951 
1952  // However, if the point is below the bottom of the closest rects vertically, get the position
1953  // farthest to the right
1954  [inputView setSelectionRects:@[
1955  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1956  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
1957  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:2U],
1958  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:3U],
1959  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 300, 100, 100) position:4U],
1960  ]];
1961  point = CGPointMake(125, 201);
1962  XCTAssertEqual(4U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1963  XCTAssertEqual(UITextStorageDirectionBackward,
1964  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1965 
1966  // Also check a point at the right edge of the last selection rect
1967  [inputView setSelectionRects:@[
1968  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
1969  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
1970  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:2U],
1971  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:3U],
1972  ]];
1973  point = CGPointMake(125, 250);
1974  XCTAssertEqual(4U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1975  XCTAssertEqual(UITextStorageDirectionBackward,
1976  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1977 
1978  // Minimize vertical distance if the difference is more than 1 point.
1979  [inputView setSelectionRects:@[
1980  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 2, 100, 100) position:0U],
1981  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 2, 100, 100) position:1U],
1982  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100) position:2U],
1983  ]];
1984  point = CGPointMake(110, 50);
1985  XCTAssertEqual(2U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1986  XCTAssertEqual(UITextStorageDirectionForward,
1987  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1988 
1989  // In floating cursor mode, the vertical difference is allowed to be 10 points.
1990  // The closest horizontal position will now win.
1991  [inputView beginFloatingCursorAtPoint:CGPointZero];
1992  XCTAssertEqual(1U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).index);
1993  XCTAssertEqual(UITextStorageDirectionForward,
1994  ((FlutterTextPosition*)[inputView closestPositionToPoint:point]).affinity);
1995  [inputView endFloatingCursor];
1996 }
1997 
1998 - (void)testClosestPositionToPointRTL {
1999  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2000  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
2001 
2002  [inputView setSelectionRects:@[
2003  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 0, 100, 100)
2004  position:0U
2005  writingDirection:NSWritingDirectionRightToLeft],
2006  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 0, 100, 100)
2007  position:1U
2008  writingDirection:NSWritingDirectionRightToLeft],
2009  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100)
2010  position:2U
2011  writingDirection:NSWritingDirectionRightToLeft],
2012  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100)
2013  position:3U
2014  writingDirection:NSWritingDirectionRightToLeft],
2015  ]];
2016  FlutterTextPosition* position =
2017  (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(275, 50)];
2018  XCTAssertEqual(0U, position.index);
2019  XCTAssertEqual(UITextStorageDirectionForward, position.affinity);
2020  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(225, 50)];
2021  XCTAssertEqual(1U, position.index);
2022  XCTAssertEqual(UITextStorageDirectionBackward, position.affinity);
2023  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(175, 50)];
2024  XCTAssertEqual(1U, position.index);
2025  XCTAssertEqual(UITextStorageDirectionForward, position.affinity);
2026  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(125, 50)];
2027  XCTAssertEqual(2U, position.index);
2028  XCTAssertEqual(UITextStorageDirectionBackward, position.affinity);
2029  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(75, 50)];
2030  XCTAssertEqual(2U, position.index);
2031  XCTAssertEqual(UITextStorageDirectionForward, position.affinity);
2032  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(25, 50)];
2033  XCTAssertEqual(3U, position.index);
2034  XCTAssertEqual(UITextStorageDirectionBackward, position.affinity);
2035  position = (FlutterTextPosition*)[inputView closestPositionToPoint:CGPointMake(-25, 50)];
2036  XCTAssertEqual(3U, position.index);
2037  XCTAssertEqual(UITextStorageDirectionBackward, position.affinity);
2038 }
2039 
2040 - (void)testSelectionRectsForRange {
2041  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2042  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
2043 
2044  CGRect testRect0 = CGRectMake(100, 100, 100, 100);
2045  CGRect testRect1 = CGRectMake(200, 200, 100, 100);
2046  [inputView setSelectionRects:@[
2047  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
2050  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 300, 100, 100) position:3U],
2051  ]];
2052 
2053  // Returns the matching rects within a range
2054  FlutterTextRange* range = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 2)];
2055  XCTAssertTrue(CGRectEqualToRect(testRect0, [inputView selectionRectsForRange:range][0].rect));
2056  XCTAssertTrue(CGRectEqualToRect(testRect1, [inputView selectionRectsForRange:range][1].rect));
2057  XCTAssertEqual(2U, [[inputView selectionRectsForRange:range] count]);
2058 
2059  // Returns a 0 width rect for a 0-length range
2060  range = [FlutterTextRange rangeWithNSRange:NSMakeRange(1, 0)];
2061  XCTAssertEqual(1U, [[inputView selectionRectsForRange:range] count]);
2062  XCTAssertTrue(CGRectEqualToRect(
2063  CGRectMake(testRect0.origin.x, testRect0.origin.y, 0, testRect0.size.height),
2064  [inputView selectionRectsForRange:range][0].rect));
2065 }
2066 
2067 - (void)testClosestPositionToPointWithinRange {
2068  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2069  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
2070 
2071  // Do not return a position before the start of the range
2072  [inputView setSelectionRects:@[
2073  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
2074  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
2075  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:2U],
2076  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:3U],
2077  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 200, 100, 100) position:4U],
2078  ]];
2079  CGPoint point = CGPointMake(125, 150);
2080  FlutterTextRange* range = [[FlutterTextRange rangeWithNSRange:NSMakeRange(3, 2)] copy];
2081  XCTAssertEqual(
2082  3U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point withinRange:range]).index);
2083  XCTAssertEqual(
2084  UITextStorageDirectionForward,
2085  ((FlutterTextPosition*)[inputView closestPositionToPoint:point withinRange:range]).affinity);
2086 
2087  // Do not return a position after the end of the range
2088  [inputView setSelectionRects:@[
2089  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U],
2090  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 100, 100, 100) position:1U],
2091  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:2U],
2092  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 100, 100, 100) position:3U],
2093  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 200, 100, 100) position:4U],
2094  ]];
2095  point = CGPointMake(125, 150);
2096  range = [[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 1)] copy];
2097  XCTAssertEqual(
2098  1U, ((FlutterTextPosition*)[inputView closestPositionToPoint:point withinRange:range]).index);
2099  XCTAssertEqual(
2100  UITextStorageDirectionForward,
2101  ((FlutterTextPosition*)[inputView closestPositionToPoint:point withinRange:range]).affinity);
2102 }
2103 
2104 - (void)testClosestPositionToPointWithPartialSelectionRects {
2105  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2106  [inputView setTextInputState:@{@"text" : @"COMPOSING"}];
2107 
2108  [inputView setSelectionRects:@[ [FlutterTextSelectionRect
2109  selectionRectWithRect:CGRectMake(0, 0, 100, 100)
2110  position:0U] ]];
2111  // Asking with a position at the end of selection rects should give you the trailing edge of
2112  // the last rect.
2113  XCTAssertTrue(CGRectEqualToRect(
2115  positionWithIndex:1
2116  affinity:UITextStorageDirectionForward]],
2117  CGRectMake(100, 0, 0, 100)));
2118  // Asking with a position beyond the end of selection rects should return CGRectZero without
2119  // crashing.
2120  XCTAssertTrue(CGRectEqualToRect(
2122  positionWithIndex:2
2123  affinity:UITextStorageDirectionForward]],
2124  CGRectZero));
2125 }
2126 
2127 #pragma mark - Floating Cursor - Tests
2128 
2129 - (void)testFloatingCursorDoesNotThrow {
2130  // The keyboard implementation may send unbalanced calls to the input view.
2131  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2132  [inputView beginFloatingCursorAtPoint:CGPointMake(123, 321)];
2133  [inputView beginFloatingCursorAtPoint:CGPointMake(123, 321)];
2134  [inputView endFloatingCursor];
2135  [inputView beginFloatingCursorAtPoint:CGPointMake(123, 321)];
2136  [inputView endFloatingCursor];
2137 }
2138 
2139 - (void)testFloatingCursor {
2140  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2141  [inputView setTextInputState:@{
2142  @"text" : @"test",
2143  @"selectionBase" : @1,
2144  @"selectionExtent" : @1,
2145  }];
2146 
2147  FlutterTextSelectionRect* first =
2148  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U];
2149  FlutterTextSelectionRect* second =
2150  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:1U];
2151  FlutterTextSelectionRect* third =
2152  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 200, 100, 100) position:2U];
2153  FlutterTextSelectionRect* fourth =
2154  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 300, 100, 100) position:3U];
2155  [inputView setSelectionRects:@[ first, second, third, fourth ]];
2156 
2157  // Verify zeroth caret rect is based on left edge of first character.
2158  XCTAssertTrue(CGRectEqualToRect(
2160  positionWithIndex:0
2161  affinity:UITextStorageDirectionForward]],
2162  CGRectMake(0, 0, 0, 100)));
2163  // Since the textAffinity is downstream, the caret rect will be based on the
2164  // left edge of the succeeding character.
2165  XCTAssertTrue(CGRectEqualToRect(
2167  positionWithIndex:1
2168  affinity:UITextStorageDirectionForward]],
2169  CGRectMake(100, 100, 0, 100)));
2170  XCTAssertTrue(CGRectEqualToRect(
2172  positionWithIndex:2
2173  affinity:UITextStorageDirectionForward]],
2174  CGRectMake(200, 200, 0, 100)));
2175  XCTAssertTrue(CGRectEqualToRect(
2177  positionWithIndex:3
2178  affinity:UITextStorageDirectionForward]],
2179  CGRectMake(300, 300, 0, 100)));
2180  // There is no subsequent character for the last position, so the caret rect
2181  // will be based on the right edge of the preceding character.
2182  XCTAssertTrue(CGRectEqualToRect(
2184  positionWithIndex:4
2185  affinity:UITextStorageDirectionForward]],
2186  CGRectMake(400, 300, 0, 100)));
2187  // Verify no caret rect for out-of-range character.
2188  XCTAssertTrue(CGRectEqualToRect(
2190  positionWithIndex:5
2191  affinity:UITextStorageDirectionForward]],
2192  CGRectZero));
2193 
2194  // Check caret rects again again when text affinity is upstream.
2195  [inputView setTextInputState:@{
2196  @"text" : @"test",
2197  @"selectionBase" : @2,
2198  @"selectionExtent" : @2,
2199  }];
2200  // Verify zeroth caret rect is based on left edge of first character.
2201  XCTAssertTrue(CGRectEqualToRect(
2203  positionWithIndex:0
2204  affinity:UITextStorageDirectionBackward]],
2205  CGRectMake(0, 0, 0, 100)));
2206  // Since the textAffinity is upstream, all below caret rects will be based on
2207  // the right edge of the preceding character.
2208  XCTAssertTrue(CGRectEqualToRect(
2210  positionWithIndex:1
2211  affinity:UITextStorageDirectionBackward]],
2212  CGRectMake(100, 0, 0, 100)));
2213  XCTAssertTrue(CGRectEqualToRect(
2215  positionWithIndex:2
2216  affinity:UITextStorageDirectionBackward]],
2217  CGRectMake(200, 100, 0, 100)));
2218  XCTAssertTrue(CGRectEqualToRect(
2220  positionWithIndex:3
2221  affinity:UITextStorageDirectionBackward]],
2222  CGRectMake(300, 200, 0, 100)));
2223  XCTAssertTrue(CGRectEqualToRect(
2225  positionWithIndex:4
2226  affinity:UITextStorageDirectionBackward]],
2227  CGRectMake(400, 300, 0, 100)));
2228  // Verify no caret rect for out-of-range character.
2229  XCTAssertTrue(CGRectEqualToRect(
2231  positionWithIndex:5
2232  affinity:UITextStorageDirectionBackward]],
2233  CGRectZero));
2234 
2235  // Verify floating cursor updates are relative to original position, and that there is no bounds
2236  // change.
2237  CGRect initialBounds = inputView.bounds;
2238  [inputView beginFloatingCursorAtPoint:CGPointMake(123, 321)];
2239  XCTAssertTrue(CGRectEqualToRect(initialBounds, inputView.bounds));
2240  OCMVerify([engine flutterTextInputView:inputView
2241  updateFloatingCursor:FlutterFloatingCursorDragStateStart
2242  withClient:0
2243  withPosition:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
2244  return ([state[@"X"] isEqualToNumber:@(0)]) &&
2245  ([state[@"Y"] isEqualToNumber:@(0)]);
2246  }]]);
2247 
2248  [inputView updateFloatingCursorAtPoint:CGPointMake(456, 654)];
2249  XCTAssertTrue(CGRectEqualToRect(initialBounds, inputView.bounds));
2250  OCMVerify([engine flutterTextInputView:inputView
2251  updateFloatingCursor:FlutterFloatingCursorDragStateUpdate
2252  withClient:0
2253  withPosition:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
2254  return ([state[@"X"] isEqualToNumber:@(333)]) &&
2255  ([state[@"Y"] isEqualToNumber:@(333)]);
2256  }]]);
2257 
2258  [inputView endFloatingCursor];
2259  XCTAssertTrue(CGRectEqualToRect(initialBounds, inputView.bounds));
2260  OCMVerify([engine flutterTextInputView:inputView
2261  updateFloatingCursor:FlutterFloatingCursorDragStateEnd
2262  withClient:0
2263  withPosition:[OCMArg checkWithBlock:^BOOL(NSDictionary* state) {
2264  return ([state[@"X"] isEqualToNumber:@(0)]) &&
2265  ([state[@"Y"] isEqualToNumber:@(0)]);
2266  }]]);
2267 }
2268 
2269 #pragma mark - UIKeyInput Overrides - Tests
2270 
2271 - (void)testInsertTextAddsPlaceholderSelectionRects {
2272  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2273  [inputView
2274  setTextInputState:@{@"text" : @"test", @"selectionBase" : @1, @"selectionExtent" : @1}];
2275 
2276  FlutterTextSelectionRect* first =
2277  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(0, 0, 100, 100) position:0U];
2278  FlutterTextSelectionRect* second =
2279  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(100, 100, 100, 100) position:1U];
2280  FlutterTextSelectionRect* third =
2281  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(200, 200, 100, 100) position:2U];
2282  FlutterTextSelectionRect* fourth =
2283  [FlutterTextSelectionRect selectionRectWithRect:CGRectMake(300, 300, 100, 100) position:3U];
2284  [inputView setSelectionRects:@[ first, second, third, fourth ]];
2285 
2286  // Inserts additional selection rects at the selection start
2287  [inputView insertText:@"in"];
2288  NSArray* selectionRects =
2289  [inputView selectionRectsForRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 6)]];
2290  XCTAssertEqual(6U, [selectionRects count]);
2291 
2292  XCTAssertEqual(first.position, ((FlutterTextSelectionRect*)selectionRects[0]).position);
2293  XCTAssertTrue(CGRectEqualToRect(first.rect, ((FlutterTextSelectionRect*)selectionRects[0]).rect));
2294 
2295  XCTAssertEqual(second.position, ((FlutterTextSelectionRect*)selectionRects[1]).position);
2296  XCTAssertTrue(
2297  CGRectEqualToRect(second.rect, ((FlutterTextSelectionRect*)selectionRects[1]).rect));
2298 
2299  XCTAssertEqual(second.position + 1, ((FlutterTextSelectionRect*)selectionRects[2]).position);
2300  XCTAssertTrue(
2301  CGRectEqualToRect(second.rect, ((FlutterTextSelectionRect*)selectionRects[2]).rect));
2302 
2303  XCTAssertEqual(second.position + 2, ((FlutterTextSelectionRect*)selectionRects[3]).position);
2304  XCTAssertTrue(
2305  CGRectEqualToRect(second.rect, ((FlutterTextSelectionRect*)selectionRects[3]).rect));
2306 
2307  XCTAssertEqual(third.position + 2, ((FlutterTextSelectionRect*)selectionRects[4]).position);
2308  XCTAssertTrue(CGRectEqualToRect(third.rect, ((FlutterTextSelectionRect*)selectionRects[4]).rect));
2309 
2310  XCTAssertEqual(fourth.position + 2, ((FlutterTextSelectionRect*)selectionRects[5]).position);
2311  XCTAssertTrue(
2312  CGRectEqualToRect(fourth.rect, ((FlutterTextSelectionRect*)selectionRects[5]).rect));
2313 }
2314 
2315 #pragma mark - Autofill - Utilities
2316 
2317 - (NSMutableDictionary*)mutablePasswordTemplateCopy {
2318  if (!_passwordTemplate) {
2319  _passwordTemplate = @{
2320  @"inputType" : @{@"name" : @"TextInuptType.text"},
2321  @"keyboardAppearance" : @"Brightness.light",
2322  @"obscureText" : @YES,
2323  @"inputAction" : @"TextInputAction.unspecified",
2324  @"smartDashesType" : @"0",
2325  @"smartQuotesType" : @"0",
2326  @"autocorrect" : @YES
2327  };
2328  }
2329 
2330  return [_passwordTemplate mutableCopy];
2331 }
2332 
2333 - (NSArray<FlutterTextInputView*>*)viewsVisibleToAutofill {
2334  return [self.installedInputViews
2335  filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"isVisibleToAutofill == YES"]];
2336 }
2337 
2338 - (void)commitAutofillContextAndVerify {
2339  FlutterMethodCall* methodCall =
2340  [FlutterMethodCall methodCallWithMethodName:@"TextInput.finishAutofillContext"
2341  arguments:@YES];
2342  [textInputPlugin handleMethodCall:methodCall
2343  result:^(id _Nullable result){
2344  }];
2345 
2346  XCTAssertEqual(self.viewsVisibleToAutofill.count,
2347  [textInputPlugin.activeView isVisibleToAutofill] ? 1ul : 0ul);
2348  XCTAssertNotEqual(textInputPlugin.textInputView, nil);
2349  // The active view should still be installed so it doesn't get
2350  // deallocated.
2351  XCTAssertEqual(self.installedInputViews.count, 1ul);
2352  XCTAssertEqual(textInputPlugin.autofillContext.count, 0ul);
2353 }
2354 
2355 #pragma mark - Autofill - Tests
2356 
2357 - (void)testDisablingAutofillOnInputClient {
2358  NSDictionary* config = self.mutableTemplateCopy;
2359  [config setValue:@"YES" forKey:@"obscureText"];
2360 
2361  [self setClientId:123 configuration:config];
2362 
2363  FlutterTextInputView* inputView = self.installedInputViews[0];
2364  XCTAssertEqualObjects(inputView.textContentType, @"");
2365 }
2366 
2367 - (void)testAutofillEnabledByDefault {
2368  NSDictionary* config = self.mutableTemplateCopy;
2369  [config setValue:@"NO" forKey:@"obscureText"];
2370  [config setValue:@{@"uniqueIdentifier" : @"field1", @"editingValue" : @{@"text" : @""}}
2371  forKey:@"autofill"];
2372 
2373  [self setClientId:123 configuration:config];
2374 
2375  FlutterTextInputView* inputView = self.installedInputViews[0];
2376  XCTAssertNil(inputView.textContentType);
2377 }
2378 
2379 - (void)testAutofillContext {
2380  NSMutableDictionary* field1 = self.mutableTemplateCopy;
2381 
2382  [field1 setValue:@{
2383  @"uniqueIdentifier" : @"field1",
2384  @"hints" : @[ @"hint1" ],
2385  @"editingValue" : @{@"text" : @""}
2386  }
2387  forKey:@"autofill"];
2388 
2389  NSMutableDictionary* field2 = self.mutablePasswordTemplateCopy;
2390  [field2 setValue:@{
2391  @"uniqueIdentifier" : @"field2",
2392  @"hints" : @[ @"hint2" ],
2393  @"editingValue" : @{@"text" : @""}
2394  }
2395  forKey:@"autofill"];
2396 
2397  NSMutableDictionary* config = [field1 mutableCopy];
2398  [config setValue:@[ field1, field2 ] forKey:@"fields"];
2399 
2400  [self setClientId:123 configuration:config];
2401  XCTAssertEqual(self.viewsVisibleToAutofill.count, 2ul);
2402 
2403  XCTAssertEqual(textInputPlugin.autofillContext.count, 2ul);
2404 
2405  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2406  XCTAssertEqual(self.installedInputViews.count, 2ul);
2407  XCTAssertEqual(textInputPlugin.textInputView, textInputPlugin.autofillContext[@"field1"]);
2408  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2409 
2410  // The configuration changes.
2411  NSMutableDictionary* field3 = self.mutablePasswordTemplateCopy;
2412  [field3 setValue:@{
2413  @"uniqueIdentifier" : @"field3",
2414  @"hints" : @[ @"hint3" ],
2415  @"editingValue" : @{@"text" : @""}
2416  }
2417  forKey:@"autofill"];
2418 
2419  NSMutableDictionary* oldContext = textInputPlugin.autofillContext;
2420  // Replace field2 with field3.
2421  [config setValue:@[ field1, field3 ] forKey:@"fields"];
2422 
2423  [self setClientId:123 configuration:config];
2424 
2425  XCTAssertEqual(self.viewsVisibleToAutofill.count, 2ul);
2426  XCTAssertEqual(textInputPlugin.autofillContext.count, 3ul);
2427 
2428  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2429  XCTAssertEqual(self.installedInputViews.count, 3ul);
2430  XCTAssertEqual(textInputPlugin.textInputView, textInputPlugin.autofillContext[@"field1"]);
2431  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2432 
2433  // Old autofill input fields are still installed and reused.
2434  for (NSString* key in oldContext.allKeys) {
2435  XCTAssertEqual(oldContext[key], textInputPlugin.autofillContext[key]);
2436  }
2437 
2438  // Switch to a password field that has no contentType and is not in an AutofillGroup.
2439  config = self.mutablePasswordTemplateCopy;
2440 
2441  oldContext = textInputPlugin.autofillContext;
2442  [self setClientId:124 configuration:config];
2443  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2444 
2445  XCTAssertEqual(self.viewsVisibleToAutofill.count, 1ul);
2446  XCTAssertEqual(textInputPlugin.autofillContext.count, 3ul);
2447 
2448  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2449  XCTAssertEqual(self.installedInputViews.count, 4ul);
2450 
2451  // Old autofill input fields are still installed and reused.
2452  for (NSString* key in oldContext.allKeys) {
2453  XCTAssertEqual(oldContext[key], textInputPlugin.autofillContext[key]);
2454  }
2455  // The active view should change.
2456  XCTAssertNotEqual(textInputPlugin.textInputView, textInputPlugin.autofillContext[@"field1"]);
2457  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2458 
2459  // Switch to a similar password field, the previous field should be reused.
2460  oldContext = textInputPlugin.autofillContext;
2461  [self setClientId:200 configuration:config];
2462 
2463  // Reuse the input view instance from the last time.
2464  XCTAssertEqual(self.viewsVisibleToAutofill.count, 1ul);
2465  XCTAssertEqual(textInputPlugin.autofillContext.count, 3ul);
2466 
2467  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2468  XCTAssertEqual(self.installedInputViews.count, 4ul);
2469 
2470  // Old autofill input fields are still installed and reused.
2471  for (NSString* key in oldContext.allKeys) {
2472  XCTAssertEqual(oldContext[key], textInputPlugin.autofillContext[key]);
2473  }
2474  XCTAssertNotEqual(textInputPlugin.textInputView, textInputPlugin.autofillContext[@"field1"]);
2475  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2476 }
2477 
2478 - (void)testCommitAutofillContext {
2479  NSMutableDictionary* field1 = self.mutableTemplateCopy;
2480  [field1 setValue:@{
2481  @"uniqueIdentifier" : @"field1",
2482  @"hints" : @[ @"hint1" ],
2483  @"editingValue" : @{@"text" : @""}
2484  }
2485  forKey:@"autofill"];
2486 
2487  NSMutableDictionary* field2 = self.mutablePasswordTemplateCopy;
2488  [field2 setValue:@{
2489  @"uniqueIdentifier" : @"field2",
2490  @"hints" : @[ @"hint2" ],
2491  @"editingValue" : @{@"text" : @""}
2492  }
2493  forKey:@"autofill"];
2494 
2495  NSMutableDictionary* field3 = self.mutableTemplateCopy;
2496  [field3 setValue:@{
2497  @"uniqueIdentifier" : @"field3",
2498  @"hints" : @[ @"hint3" ],
2499  @"editingValue" : @{@"text" : @""}
2500  }
2501  forKey:@"autofill"];
2502 
2503  NSMutableDictionary* config = [field1 mutableCopy];
2504  [config setValue:@[ field1, field2 ] forKey:@"fields"];
2505 
2506  [self setClientId:123 configuration:config];
2507  XCTAssertEqual(self.viewsVisibleToAutofill.count, 2ul);
2508  XCTAssertEqual(textInputPlugin.autofillContext.count, 2ul);
2509  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2510 
2511  [self commitAutofillContextAndVerify];
2512  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2513 
2514  // Install the password field again.
2515  [self setClientId:123 configuration:config];
2516  // Switch to a regular autofill group.
2517  [self setClientId:124 configuration:field3];
2518  XCTAssertEqual(self.viewsVisibleToAutofill.count, 1ul);
2519 
2520  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2521  XCTAssertEqual(self.installedInputViews.count, 3ul);
2522  XCTAssertEqual(textInputPlugin.autofillContext.count, 2ul);
2523  XCTAssertNotEqual(textInputPlugin.textInputView, nil);
2524  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2525 
2526  [self commitAutofillContextAndVerify];
2527  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2528 
2529  // Now switch to an input field that does not autofill.
2530  [self setClientId:125 configuration:self.mutableTemplateCopy];
2531 
2532  XCTAssertEqual(self.viewsVisibleToAutofill.count, 0ul);
2533  // The active view should still be installed so it doesn't get
2534  // deallocated.
2535 
2536  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2537  XCTAssertEqual(self.installedInputViews.count, 1ul);
2538  XCTAssertEqual(textInputPlugin.autofillContext.count, 0ul);
2539  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2540 
2541  [self commitAutofillContextAndVerify];
2542  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2543 }
2544 
2545 - (void)testAutofillInputViews {
2546  NSMutableDictionary* field1 = self.mutableTemplateCopy;
2547  [field1 setValue:@{
2548  @"uniqueIdentifier" : @"field1",
2549  @"hints" : @[ @"hint1" ],
2550  @"editingValue" : @{@"text" : @""}
2551  }
2552  forKey:@"autofill"];
2553 
2554  NSMutableDictionary* field2 = self.mutablePasswordTemplateCopy;
2555  [field2 setValue:@{
2556  @"uniqueIdentifier" : @"field2",
2557  @"hints" : @[ @"hint2" ],
2558  @"editingValue" : @{@"text" : @""}
2559  }
2560  forKey:@"autofill"];
2561 
2562  NSMutableDictionary* config = [field1 mutableCopy];
2563  [config setValue:@[ field1, field2 ] forKey:@"fields"];
2564 
2565  [self setClientId:123 configuration:config];
2566  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2567 
2568  // Find all the FlutterTextInputViews we created.
2569  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
2570 
2571  // Both fields are installed and visible because it's a password group.
2572  XCTAssertEqual(inputFields.count, 2ul);
2573  XCTAssertEqual(self.viewsVisibleToAutofill.count, 2ul);
2574 
2575  // Find the inactive autofillable input field.
2576  FlutterTextInputView* inactiveView = inputFields[1];
2577  [inactiveView replaceRange:[FlutterTextRange rangeWithNSRange:NSMakeRange(0, 0)]
2578  withText:@"Autofilled!"];
2579  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2580 
2581  // Verify behavior.
2582  OCMVerify([engine flutterTextInputView:inactiveView
2583  updateEditingClient:0
2584  withState:[OCMArg isNotNil]
2585  withTag:@"field2"]);
2586 }
2587 
2588 - (void)testPasswordAutofillHack {
2589  NSDictionary* config = self.mutableTemplateCopy;
2590  [config setValue:@"YES" forKey:@"obscureText"];
2591  [self setClientId:123 configuration:config];
2592 
2593  // Find all the FlutterTextInputViews we created.
2594  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
2595 
2596  FlutterTextInputView* inputView = inputFields[0];
2597 
2598  XCTAssert([inputView isKindOfClass:[UITextField class]]);
2599  // FlutterSecureTextInputView does not respond to font,
2600  // but it should return the default UITextField.font.
2601  XCTAssertNotEqual([inputView performSelector:@selector(font)], nil);
2602 }
2603 
2604 - (void)testClearAutofillContextClearsSelection {
2605  NSMutableDictionary* regularField = self.mutableTemplateCopy;
2606  NSDictionary* editingValue = @{
2607  @"text" : @"REGULAR_TEXT_FIELD",
2608  @"composingBase" : @0,
2609  @"composingExtent" : @3,
2610  @"selectionBase" : @1,
2611  @"selectionExtent" : @4
2612  };
2613  [regularField setValue:@{
2614  @"uniqueIdentifier" : @"field2",
2615  @"hints" : @[ @"hint2" ],
2616  @"editingValue" : editingValue,
2617  }
2618  forKey:@"autofill"];
2619  [regularField addEntriesFromDictionary:editingValue];
2620  [self setClientId:123 configuration:regularField];
2621  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2622  XCTAssertEqual(self.installedInputViews.count, 1ul);
2623 
2624  FlutterTextInputView* oldInputView = self.installedInputViews[0];
2625  XCTAssert([oldInputView.text isEqualToString:@"REGULAR_TEXT_FIELD"]);
2626  FlutterTextRange* selectionRange = (FlutterTextRange*)oldInputView.selectedTextRange;
2627  XCTAssert(NSEqualRanges(selectionRange.range, NSMakeRange(1, 3)));
2628 
2629  // Replace the original password field with new one. This should remove
2630  // the old password field, but not immediately.
2631  [self setClientId:124 configuration:self.mutablePasswordTemplateCopy];
2632  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2633 
2634  XCTAssertEqual(self.installedInputViews.count, 2ul);
2635 
2636  [textInputPlugin cleanUpViewHierarchy:NO clearText:YES delayRemoval:NO];
2637  XCTAssertEqual(self.installedInputViews.count, 1ul);
2638 
2639  // Verify the old input view is properly cleaned up.
2640  XCTAssert([oldInputView.text isEqualToString:@""]);
2641  selectionRange = (FlutterTextRange*)oldInputView.selectedTextRange;
2642  XCTAssert(NSEqualRanges(selectionRange.range, NSMakeRange(0, 0)));
2643 }
2644 
2645 - (void)testGarbageInputViewsAreNotRemovedImmediately {
2646  // Add a password field that should autofill.
2647  [self setClientId:123 configuration:self.mutablePasswordTemplateCopy];
2648  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2649 
2650  XCTAssertEqual(self.installedInputViews.count, 1ul);
2651  // Add an input field that doesn't autofill. This should remove the password
2652  // field, but not immediately.
2653  [self setClientId:124 configuration:self.mutableTemplateCopy];
2654  [self ensureOnlyActiveViewCanBecomeFirstResponder];
2655 
2656  XCTAssertEqual(self.installedInputViews.count, 2ul);
2657 
2658  [self commitAutofillContextAndVerify];
2659 }
2660 
2661 - (void)testScribbleSetSelectionRects {
2662  NSMutableDictionary* regularField = self.mutableTemplateCopy;
2663  NSDictionary* editingValue = @{
2664  @"text" : @"REGULAR_TEXT_FIELD",
2665  @"composingBase" : @0,
2666  @"composingExtent" : @3,
2667  @"selectionBase" : @1,
2668  @"selectionExtent" : @4
2669  };
2670  [regularField setValue:@{
2671  @"uniqueIdentifier" : @"field1",
2672  @"hints" : @[ @"hint2" ],
2673  @"editingValue" : editingValue,
2674  }
2675  forKey:@"autofill"];
2676  [regularField addEntriesFromDictionary:editingValue];
2677  [self setClientId:123 configuration:regularField];
2678  XCTAssertEqual(self.installedInputViews.count, 1ul);
2679  XCTAssertEqual([textInputPlugin.activeView.selectionRects count], 0u);
2680 
2681  NSArray<NSNumber*>* selectionRect = [NSArray arrayWithObjects:@0, @0, @100, @100, @0, @1, nil];
2682  NSArray* selectionRects = [NSArray arrayWithObjects:selectionRect, nil];
2683  FlutterMethodCall* methodCall =
2684  [FlutterMethodCall methodCallWithMethodName:@"Scribble.setSelectionRects"
2685  arguments:selectionRects];
2686  [textInputPlugin handleMethodCall:methodCall
2687  result:^(id _Nullable result){
2688  }];
2689 
2690  XCTAssertEqual([textInputPlugin.activeView.selectionRects count], 1u);
2691 }
2692 
2693 - (void)testDecommissionedViewAreNotReusedByAutofill {
2694  // Regression test for https://github.com/flutter/flutter/issues/84407.
2695  NSMutableDictionary* configuration = self.mutableTemplateCopy;
2696  [configuration setValue:@{
2697  @"uniqueIdentifier" : @"field1",
2698  @"hints" : @[ UITextContentTypePassword ],
2699  @"editingValue" : @{@"text" : @""}
2700  }
2701  forKey:@"autofill"];
2702  [configuration setValue:@[ [configuration copy] ] forKey:@"fields"];
2703 
2704  [self setClientId:123 configuration:configuration];
2705 
2706  [self setTextInputHide];
2707  UIView* previousActiveView = textInputPlugin.activeView;
2708 
2709  [self setClientId:124 configuration:configuration];
2710 
2711  // Make sure the autofillable view is reused.
2712  XCTAssertEqual(previousActiveView, textInputPlugin.activeView);
2713  XCTAssertNotNil(previousActiveView);
2714  // Does not crash.
2715 }
2716 
2717 - (void)testInitialActiveViewCantAccessTextInputDelegate {
2718  // Before the framework sends the first text input configuration,
2719  // the dummy "activeView" we use should never have access to
2720  // its textInputDelegate.
2721  XCTAssertNil(textInputPlugin.activeView.textInputDelegate);
2722 }
2723 
2724 - (void)testAutoFillDoesNotTriggerOnHideButTriggersOnCommit {
2725  // Regression test for https://github.com/flutter/flutter/issues/145681.
2726  NSMutableDictionary* configuration = self.mutableTemplateCopy;
2727  [configuration setValue:@{
2728  @"uniqueIdentifier" : @"field1",
2729  @"hints" : @[ UITextContentTypePassword ],
2730  @"editingValue" : @{@"text" : @""}
2731  }
2732  forKey:@"autofill"];
2733  [configuration setValue:@[ [configuration copy] ] forKey:@"fields"];
2734 
2735  [self setClientId:123 configuration:configuration];
2736  XCTAssertEqual(self.viewsVisibleToAutofill.count, 1ul);
2737 
2738  [self setTextInputHide];
2739  // Before the fix in https://github.com/flutter/flutter/pull/160653, it was 0ul.
2740  XCTAssertEqual(self.viewsVisibleToAutofill.count, 1ul);
2741 
2742  [self commitAutofillContextAndVerify];
2743 }
2744 
2745 #pragma mark - Accessibility - Tests
2746 
2747 - (void)testUITextInputAccessibilityNotHiddenWhenShowed {
2748  [self setClientId:123 configuration:self.mutableTemplateCopy];
2749 
2750  // Send show text input method call.
2751  [self setTextInputShow];
2752  // Find all the FlutterTextInputViews we created.
2753  NSArray<FlutterTextInputView*>* inputFields = self.installedInputViews;
2754 
2755  // The input view should not be hidden.
2756  XCTAssertEqual([inputFields count], 1u);
2757 
2758  // Send hide text input method call.
2759  [self setTextInputHide];
2760 
2761  inputFields = self.installedInputViews;
2762 
2763  // The input view should be hidden.
2764  XCTAssertEqual([inputFields count], 0u);
2765 }
2766 
2767 - (void)testFlutterTextInputViewDirectFocusToBackingTextInput {
2768  FlutterTextInputViewSpy* inputView =
2769  [[FlutterTextInputViewSpy alloc] initWithOwner:textInputPlugin];
2770  UIView* container = [[UIView alloc] init];
2771  UIAccessibilityElement* backing =
2772  [[UIAccessibilityElement alloc] initWithAccessibilityContainer:container];
2773  inputView.backingTextInputAccessibilityObject = backing;
2774  // Simulate accessibility focus.
2775  inputView.isAccessibilityFocused = YES;
2776  [inputView accessibilityElementDidBecomeFocused];
2777 
2778  XCTAssertEqual(inputView.receivedNotification, UIAccessibilityScreenChangedNotification);
2779  XCTAssertEqual(inputView.receivedNotificationTarget, backing);
2780 }
2781 
2782 - (void)testFlutterTokenizerCanParseLines {
2783  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2784  id<UITextInputTokenizer> tokenizer = [inputView tokenizer];
2785 
2786  // The tokenizer returns zero range When text is empty.
2787  FlutterTextRange* range = [self getLineRangeFromTokenizer:tokenizer atIndex:0];
2788  XCTAssertEqual(range.range.location, 0u);
2789  XCTAssertEqual(range.range.length, 0u);
2790 
2791  [inputView insertText:@"how are you\nI am fine, Thank you"];
2792 
2793  range = [self getLineRangeFromTokenizer:tokenizer atIndex:0];
2794  XCTAssertEqual(range.range.location, 0u);
2795  XCTAssertEqual(range.range.length, 11u);
2796 
2797  range = [self getLineRangeFromTokenizer:tokenizer atIndex:2];
2798  XCTAssertEqual(range.range.location, 0u);
2799  XCTAssertEqual(range.range.length, 11u);
2800 
2801  range = [self getLineRangeFromTokenizer:tokenizer atIndex:11];
2802  XCTAssertEqual(range.range.location, 0u);
2803  XCTAssertEqual(range.range.length, 11u);
2804 
2805  range = [self getLineRangeFromTokenizer:tokenizer atIndex:12];
2806  XCTAssertEqual(range.range.location, 12u);
2807  XCTAssertEqual(range.range.length, 20u);
2808 
2809  range = [self getLineRangeFromTokenizer:tokenizer atIndex:15];
2810  XCTAssertEqual(range.range.location, 12u);
2811  XCTAssertEqual(range.range.length, 20u);
2812 
2813  range = [self getLineRangeFromTokenizer:tokenizer atIndex:32];
2814  XCTAssertEqual(range.range.location, 12u);
2815  XCTAssertEqual(range.range.length, 20u);
2816 }
2817 
2818 - (void)testFlutterTokenizerLineEnclosingEndOfDocumentInBackwardDirectionShouldNotReturnNil {
2819  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2820  [inputView insertText:@"0123456789\n012345"];
2821  id<UITextInputTokenizer> tokenizer = [inputView tokenizer];
2822 
2823  FlutterTextRange* range =
2824  (FlutterTextRange*)[tokenizer rangeEnclosingPosition:[inputView endOfDocument]
2825  withGranularity:UITextGranularityLine
2826  inDirection:UITextStorageDirectionBackward];
2827  XCTAssertEqual(range.range.location, 11u);
2828  XCTAssertEqual(range.range.length, 6u);
2829 }
2830 
2831 - (void)testFlutterTokenizerLineEnclosingEndOfDocumentInForwardDirectionShouldReturnNilOnIOS17 {
2832  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2833  [inputView insertText:@"0123456789\n012345"];
2834  id<UITextInputTokenizer> tokenizer = [inputView tokenizer];
2835 
2836  FlutterTextRange* range =
2837  (FlutterTextRange*)[tokenizer rangeEnclosingPosition:[inputView endOfDocument]
2838  withGranularity:UITextGranularityLine
2839  inDirection:UITextStorageDirectionForward];
2840  if (@available(iOS 17.0, *)) {
2841  XCTAssertNil(range);
2842  } else {
2843  XCTAssertEqual(range.range.location, 11u);
2844  XCTAssertEqual(range.range.length, 6u);
2845  }
2846 }
2847 
2848 - (void)testFlutterTokenizerLineEnclosingOutOfRangePositionShouldReturnNilOnIOS17 {
2849  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2850  [inputView insertText:@"0123456789\n012345"];
2851  id<UITextInputTokenizer> tokenizer = [inputView tokenizer];
2852 
2854  FlutterTextRange* range =
2855  (FlutterTextRange*)[tokenizer rangeEnclosingPosition:position
2856  withGranularity:UITextGranularityLine
2857  inDirection:UITextStorageDirectionForward];
2858  if (@available(iOS 17.0, *)) {
2859  XCTAssertNil(range);
2860  } else {
2861  XCTAssertEqual(range.range.location, 0u);
2862  XCTAssertEqual(range.range.length, 0u);
2863  }
2864 }
2865 
2866 - (void)testFlutterTextInputPluginRetainsFlutterTextInputView {
2867  FlutterViewController* flutterViewController = [[FlutterViewController alloc] init];
2868  FlutterTextInputPlugin* myInputPlugin = [[FlutterTextInputPlugin alloc] initWithDelegate:engine];
2869  myInputPlugin.viewController = flutterViewController;
2870 
2871  __weak UIView* activeView;
2872  @autoreleasepool {
2873  FlutterMethodCall* setClientCall = [FlutterMethodCall
2874  methodCallWithMethodName:@"TextInput.setClient"
2875  arguments:@[
2876  [NSNumber numberWithInt:123], self.mutablePasswordTemplateCopy
2877  ]];
2878  [myInputPlugin handleMethodCall:setClientCall
2879  result:^(id _Nullable result){
2880  }];
2881  activeView = myInputPlugin.textInputView;
2882  FlutterMethodCall* hideCall = [FlutterMethodCall methodCallWithMethodName:@"TextInput.hide"
2883  arguments:@[]];
2884  [myInputPlugin handleMethodCall:hideCall
2885  result:^(id _Nullable result){
2886  }];
2887  XCTAssertNotNil(activeView);
2888  }
2889  // This assert proves the myInputPlugin.textInputView is not deallocated.
2890  XCTAssertNotNil(activeView);
2891 }
2892 
2893 - (void)testFlutterTextInputPluginHostViewNilCrash {
2894  FlutterTextInputPlugin* myInputPlugin = [[FlutterTextInputPlugin alloc] initWithDelegate:engine];
2895  myInputPlugin.viewController = nil;
2896  XCTAssertThrows([myInputPlugin hostView], @"Throws exception if host view is nil");
2897 }
2898 
2899 - (void)testFlutterTextInputPluginHostViewNotNil {
2900  FlutterViewController* flutterViewController = [[FlutterViewController alloc] init];
2901  FlutterEngine* flutterEngine = [[FlutterEngine alloc] init];
2902  [flutterEngine runWithEntrypoint:nil];
2903  flutterEngine.viewController = flutterViewController;
2904  XCTAssertNotNil(flutterEngine.textInputPlugin.viewController);
2905  XCTAssertNotNil([flutterEngine.textInputPlugin hostView]);
2906 }
2907 
2908 - (void)testSetPlatformViewClient {
2909  FlutterViewController* flutterViewController = [[FlutterViewController alloc] init];
2910  FlutterTextInputPlugin* myInputPlugin = [[FlutterTextInputPlugin alloc] initWithDelegate:engine];
2911  myInputPlugin.viewController = flutterViewController;
2912 
2913  FlutterMethodCall* setClientCall = [FlutterMethodCall
2914  methodCallWithMethodName:@"TextInput.setClient"
2915  arguments:@[ [NSNumber numberWithInt:123], self.mutablePasswordTemplateCopy ]];
2916  [myInputPlugin handleMethodCall:setClientCall
2917  result:^(id _Nullable result){
2918  }];
2919  UIView* activeView = myInputPlugin.textInputView;
2920  XCTAssertNotNil(activeView.superview, @"activeView must be added to the view hierarchy.");
2921  FlutterMethodCall* setPlatformViewClientCall = [FlutterMethodCall
2922  methodCallWithMethodName:@"TextInput.setPlatformViewClient"
2923  arguments:@{@"platformViewId" : [NSNumber numberWithLong:456]}];
2924  [myInputPlugin handleMethodCall:setPlatformViewClientCall
2925  result:^(id _Nullable result){
2926  }];
2927  XCTAssertNil(activeView.superview, @"activeView must be removed from view hierarchy.");
2928 }
2929 
2930 - (void)testEditMenu_shouldSetupEditMenuDelegateCorrectly {
2931  if (@available(iOS 16.0, *)) {
2932  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
2933  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
2934  XCTAssertEqual(inputView.editMenuInteraction.delegate, inputView,
2935  @"editMenuInteraction setup delegate correctly");
2936  }
2937 }
2938 
2939 - (void)testEditMenu_shouldNotPresentEditMenuIfNotFirstResponder {
2940  if (@available(iOS 16.0, *)) {
2941  FlutterTextInputPlugin* myInputPlugin =
2942  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
2943  BOOL shownEditMenu = [myInputPlugin showEditMenu:@{}];
2944  XCTAssertFalse(shownEditMenu, @"Should not show edit menu if not first responder.");
2945  }
2946 }
2947 
2948 - (void)testEditMenu_shouldPresentEditMenuWithCorrectConfiguration {
2949  if (@available(iOS 16.0, *)) {
2950  FlutterTextInputPlugin* myInputPlugin =
2951  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
2952  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
2953  myInputPlugin.viewController = myViewController;
2954  [myViewController loadView];
2955  FlutterMethodCall* setClientCall =
2956  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
2957  arguments:@[ @(123), self.mutableTemplateCopy ]];
2958  [myInputPlugin handleMethodCall:setClientCall
2959  result:^(id _Nullable result){
2960  }];
2961 
2962  FlutterTextInputView* myInputView = myInputPlugin.activeView;
2963  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
2964 
2965  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
2966 
2967  XCTestExpectation* expectation = [[XCTestExpectation alloc]
2968  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
2969 
2970  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
2971  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
2972  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
2973  .andDo(^(NSInvocation* invocation) {
2974  // arguments are released once invocation is released.
2975  [invocation retainArguments];
2976  UIEditMenuConfiguration* config;
2977  [invocation getArgument:&config atIndex:2];
2978  XCTAssertEqual(config.preferredArrowDirection, UIEditMenuArrowDirectionAutomatic,
2979  @"UIEditMenuConfiguration must use automatic arrow direction.");
2980  XCTAssert(CGPointEqualToPoint(config.sourcePoint, CGPointZero),
2981  @"UIEditMenuConfiguration must have the correct point.");
2982  [expectation fulfill];
2983  });
2984 
2985  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
2986  @{@"x" : @(0), @"y" : @(0), @"width" : @(0), @"height" : @(0)};
2987 
2988  BOOL shownEditMenu = [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect}];
2989  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
2990  [self waitForExpectations:@[ expectation ] timeout:1.0];
2991  }
2992 }
2993 
2994 - (void)testEditMenu_shouldPresentEditMenuWithCorectTargetRect {
2995  if (@available(iOS 16.0, *)) {
2996  FlutterTextInputPlugin* myInputPlugin =
2997  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
2998  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
2999  myInputPlugin.viewController = myViewController;
3000  [myViewController loadView];
3001 
3002  FlutterMethodCall* setClientCall =
3003  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
3004  arguments:@[ @(123), self.mutableTemplateCopy ]];
3005  [myInputPlugin handleMethodCall:setClientCall
3006  result:^(id _Nullable result){
3007  }];
3008 
3009  FlutterTextInputView* myInputView = myInputPlugin.activeView;
3010 
3011  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
3012  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
3013 
3014  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3015  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
3016 
3017  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
3018  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
3019  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
3020  .andDo(^(NSInvocation* invocation) {
3021  [expectation fulfill];
3022  });
3023 
3024  myInputView.frame = CGRectMake(10, 20, 30, 40);
3025  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
3026  @{@"x" : @(100), @"y" : @(200), @"width" : @(300), @"height" : @(400)};
3027 
3028  BOOL shownEditMenu = [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect}];
3029  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
3030  [self waitForExpectations:@[ expectation ] timeout:1.0];
3031 
3032  CGRect targetRect =
3033  [myInputView editMenuInteraction:mockInteraction
3034  targetRectForConfiguration:OCMClassMock([UIEditMenuConfiguration class])];
3035  // the encoded target rect is in global coordinate space.
3036  XCTAssert(CGRectEqualToRect(targetRect, CGRectMake(90, 180, 300, 400)),
3037  @"targetRectForConfiguration must return the correct target rect.");
3038  }
3039 }
3040 
3041 - (void)testEditMenu_shouldPresentEditMenuWithSuggestedItemsByDefaultIfNoFrameworkData {
3042  if (@available(iOS 16.0, *)) {
3043  FlutterTextInputPlugin* myInputPlugin =
3044  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
3045  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
3046  myInputPlugin.viewController = myViewController;
3047  [myViewController loadView];
3048 
3049  FlutterMethodCall* setClientCall =
3050  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
3051  arguments:@[ @(123), self.mutableTemplateCopy ]];
3052  [myInputPlugin handleMethodCall:setClientCall
3053  result:^(id _Nullable result){
3054  }];
3055 
3056  FlutterTextInputView* myInputView = myInputPlugin.activeView;
3057 
3058  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
3059  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
3060 
3061  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3062  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
3063 
3064  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
3065  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
3066  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
3067  .andDo(^(NSInvocation* invocation) {
3068  [expectation fulfill];
3069  });
3070 
3071  myInputView.frame = CGRectMake(10, 20, 30, 40);
3072  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
3073  @{@"x" : @(100), @"y" : @(200), @"width" : @(300), @"height" : @(400)};
3074  // No items provided from framework. Show the suggested items by default.
3075  BOOL shownEditMenu = [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect}];
3076  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
3077  [self waitForExpectations:@[ expectation ] timeout:1.0];
3078 
3079  UICommand* copyItem = [UICommand commandWithTitle:@"Copy"
3080  image:nil
3081  action:@selector(copy:)
3082  propertyList:nil];
3083  UICommand* pasteItem = [UICommand commandWithTitle:@"Paste"
3084  image:nil
3085  action:@selector(paste:)
3086  propertyList:nil];
3087  NSArray<UICommand*>* suggestedActions = @[ copyItem, pasteItem ];
3088 
3089  UIMenu* menu = [myInputView editMenuInteraction:mockInteraction
3090  menuForConfiguration:OCMClassMock([UIEditMenuConfiguration class])
3091  suggestedActions:suggestedActions];
3092  XCTAssertEqualObjects(menu.children, suggestedActions,
3093  @"Must show suggested items by default.");
3094  }
3095 }
3096 
3097 - (void)testEditMenu_shouldPresentEditMenuWithCorectItemsAndCorrectOrderingForBasicEditingActions {
3098  if (@available(iOS 16.0, *)) {
3099  FlutterTextInputPlugin* myInputPlugin =
3100  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
3101  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
3102  myInputPlugin.viewController = myViewController;
3103  [myViewController loadView];
3104 
3105  FlutterMethodCall* setClientCall =
3106  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
3107  arguments:@[ @(123), self.mutableTemplateCopy ]];
3108  [myInputPlugin handleMethodCall:setClientCall
3109  result:^(id _Nullable result){
3110  }];
3111 
3112  FlutterTextInputView* myInputView = myInputPlugin.activeView;
3113 
3114  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
3115  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
3116 
3117  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3118  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
3119 
3120  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
3121  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
3122  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
3123  .andDo(^(NSInvocation* invocation) {
3124  [expectation fulfill];
3125  });
3126 
3127  myInputView.frame = CGRectMake(10, 20, 30, 40);
3128  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
3129  @{@"x" : @(100), @"y" : @(200), @"width" : @(300), @"height" : @(400)};
3130 
3131  NSArray<NSDictionary<NSString*, id>*>* encodedItems =
3132  @[ @{@"type" : @"paste"}, @{@"type" : @"copy"} ];
3133 
3134  BOOL shownEditMenu =
3135  [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect, @"items" : encodedItems}];
3136  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
3137  [self waitForExpectations:@[ expectation ] timeout:1.0];
3138 
3139  UICommand* copyItem = [UICommand commandWithTitle:@"Copy"
3140  image:nil
3141  action:@selector(copy:)
3142  propertyList:nil];
3143  UICommand* pasteItem = [UICommand commandWithTitle:@"Paste"
3144  image:nil
3145  action:@selector(paste:)
3146  propertyList:nil];
3147  NSArray<UICommand*>* suggestedActions = @[ copyItem, pasteItem ];
3148 
3149  UIMenu* menu = [myInputView editMenuInteraction:mockInteraction
3150  menuForConfiguration:OCMClassMock([UIEditMenuConfiguration class])
3151  suggestedActions:suggestedActions];
3152  // The item ordering should follow the encoded data sent from the framework.
3153  NSArray<UICommand*>* expectedChildren = @[ pasteItem, copyItem ];
3154  XCTAssertEqualObjects(menu.children, expectedChildren);
3155  }
3156 }
3157 
3158 - (void)testEditMenu_shouldPresentEditMenuWithCorectItemsUnderNestedSubtreeForBasicEditingActions {
3159  if (@available(iOS 16.0, *)) {
3160  FlutterTextInputPlugin* myInputPlugin =
3161  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
3162  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
3163  myInputPlugin.viewController = myViewController;
3164  [myViewController loadView];
3165 
3166  FlutterMethodCall* setClientCall =
3167  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
3168  arguments:@[ @(123), self.mutableTemplateCopy ]];
3169  [myInputPlugin handleMethodCall:setClientCall
3170  result:^(id _Nullable result){
3171  }];
3172 
3173  FlutterTextInputView* myInputView = myInputPlugin.activeView;
3174 
3175  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
3176  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
3177 
3178  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3179  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
3180 
3181  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
3182  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
3183  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
3184  .andDo(^(NSInvocation* invocation) {
3185  [expectation fulfill];
3186  });
3187 
3188  myInputView.frame = CGRectMake(10, 20, 30, 40);
3189  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
3190  @{@"x" : @(100), @"y" : @(200), @"width" : @(300), @"height" : @(400)};
3191 
3192  NSArray<NSDictionary<NSString*, id>*>* encodedItems =
3193  @[ @{@"type" : @"cut"}, @{@"type" : @"paste"}, @{@"type" : @"copy"} ];
3194 
3195  BOOL shownEditMenu =
3196  [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect, @"items" : encodedItems}];
3197  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
3198  [self waitForExpectations:@[ expectation ] timeout:1.0];
3199 
3200  UICommand* copyItem = [UICommand commandWithTitle:@"Copy"
3201  image:nil
3202  action:@selector(copy:)
3203  propertyList:nil];
3204  UICommand* cutItem = [UICommand commandWithTitle:@"Cut"
3205  image:nil
3206  action:@selector(cut:)
3207  propertyList:nil];
3208  UICommand* pasteItem = [UICommand commandWithTitle:@"Paste"
3209  image:nil
3210  action:@selector(paste:)
3211  propertyList:nil];
3212  /*
3213  A more complex menu hierarchy for DFS:
3214 
3215  menu
3216  / | \
3217  copy menu menu
3218  | \
3219  paste menu
3220  |
3221  cut
3222  */
3223  NSArray<UIMenuElement*>* suggestedActions = @[
3224  copyItem, [UIMenu menuWithChildren:@[ pasteItem ]],
3225  [UIMenu menuWithChildren:@[ [UIMenu menuWithChildren:@[ cutItem ]] ]]
3226  ];
3227 
3228  UIMenu* menu = [myInputView editMenuInteraction:mockInteraction
3229  menuForConfiguration:OCMClassMock([UIEditMenuConfiguration class])
3230  suggestedActions:suggestedActions];
3231  // The item ordering should follow the encoded data sent from the framework.
3232  NSArray<UICommand*>* expectedActions = @[ cutItem, pasteItem, copyItem ];
3233  XCTAssertEqualObjects(menu.children, expectedActions);
3234  }
3235 }
3236 
3237 - (void)testEditMenu_shouldPresentEditMenuWithCorectItemsForMoreAdditionalItems {
3238  if (@available(iOS 16.0, *)) {
3239  FlutterTextInputPlugin* myInputPlugin =
3240  [[FlutterTextInputPlugin alloc] initWithDelegate:OCMClassMock([FlutterEngine class])];
3241  FlutterViewController* myViewController = [[FlutterViewController alloc] init];
3242  myInputPlugin.viewController = myViewController;
3243  [myViewController loadView];
3244 
3245  FlutterMethodCall* setClientCall =
3246  [FlutterMethodCall methodCallWithMethodName:@"TextInput.setClient"
3247  arguments:@[ @(123), self.mutableTemplateCopy ]];
3248  [myInputPlugin handleMethodCall:setClientCall
3249  result:^(id _Nullable result){
3250  }];
3251 
3252  FlutterTextInputView* myInputView = myInputPlugin.activeView;
3253 
3254  FlutterTextInputView* mockInputView = OCMPartialMock(myInputView);
3255  OCMStub([mockInputView isFirstResponder]).andReturn(YES);
3256 
3257  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3258  initWithDescription:@"presentEditMenuWithConfiguration must be called."];
3259 
3260  id mockInteraction = OCMClassMock([UIEditMenuInteraction class]);
3261  OCMStub([mockInputView editMenuInteraction]).andReturn(mockInteraction);
3262  OCMStub([mockInteraction presentEditMenuWithConfiguration:[OCMArg any]])
3263  .andDo(^(NSInvocation* invocation) {
3264  [expectation fulfill];
3265  });
3266 
3267  myInputView.frame = CGRectMake(10, 20, 30, 40);
3268  NSDictionary<NSString*, NSNumber*>* encodedTargetRect =
3269  @{@"x" : @(100), @"y" : @(200), @"width" : @(300), @"height" : @(400)};
3270 
3271  NSArray<NSDictionary<NSString*, id>*>* encodedItems = @[
3272  @{@"type" : @"searchWeb", @"title" : @"Search Web"},
3273  @{@"type" : @"lookUp", @"title" : @"Look Up"}, @{@"type" : @"share", @"title" : @"Share"}
3274  ];
3275 
3276  BOOL shownEditMenu =
3277  [myInputPlugin showEditMenu:@{@"targetRect" : encodedTargetRect, @"items" : encodedItems}];
3278  XCTAssertTrue(shownEditMenu, @"Should show edit menu with correct configuration.");
3279  [self waitForExpectations:@[ expectation ] timeout:1.0];
3280 
3281  NSArray<UICommand*>* suggestedActions = @[
3282  [UICommand commandWithTitle:@"copy" image:nil action:@selector(copy:) propertyList:nil],
3283  ];
3284 
3285  UIMenu* menu = [myInputView editMenuInteraction:mockInteraction
3286  menuForConfiguration:OCMClassMock([UIEditMenuConfiguration class])
3287  suggestedActions:suggestedActions];
3288  XCTAssert(menu.children.count == 3, @"There must be 3 menu items");
3289 
3290  XCTAssert(((UICommand*)menu.children[0]).action == @selector(handleSearchWebAction),
3291  @"Must create search web item in the tree.");
3292  XCTAssert(((UICommand*)menu.children[1]).action == @selector(handleLookUpAction),
3293  @"Must create look up item in the tree.");
3294  XCTAssert(((UICommand*)menu.children[2]).action == @selector(handleShareAction),
3295  @"Must create share item in the tree.");
3296  }
3297 }
3298 
3299 - (void)testInteractiveKeyboardAfterUserScrollWillResignFirstResponder {
3300  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3301  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3302 
3303  [inputView setTextInputClient:123];
3304  [inputView reloadInputViews];
3305  [inputView becomeFirstResponder];
3306  XCTAssert(inputView.isFirstResponder);
3307 
3308  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3309  [NSNotificationCenter.defaultCenter
3310  postNotificationName:UIKeyboardWillShowNotification
3311  object:nil
3312  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3313  FlutterMethodCall* onPointerMoveCall =
3314  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3315  arguments:@{@"pointerY" : @(500)}];
3316  [textInputPlugin handleMethodCall:onPointerMoveCall
3317  result:^(id _Nullable result){
3318  }];
3319  XCTAssertFalse(inputView.isFirstResponder);
3320  textInputPlugin.cachedFirstResponder = nil;
3321 }
3322 
3323 - (void)testInteractiveKeyboardAfterUserScrollToTopOfKeyboardWillTakeScreenshot {
3324  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3325  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3326  UIScene* scene = scenes.anyObject;
3327  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3328  UIWindowScene* windowScene = (UIWindowScene*)scene;
3329  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3330  UIWindow* window = windowScene.windows[0];
3331  [window addSubview:viewController.view];
3332 
3333  [viewController loadView];
3334 
3335  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3336  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3337 
3338  [inputView setTextInputClient:123];
3339  [inputView reloadInputViews];
3340  [inputView becomeFirstResponder];
3341 
3342  if (textInputPlugin.keyboardView.superview != nil) {
3343  for (UIView* subView in textInputPlugin.keyboardViewContainer.subviews) {
3344  [subView removeFromSuperview];
3345  }
3346  }
3347  XCTAssert(textInputPlugin.keyboardView.superview == nil);
3348  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3349  [NSNotificationCenter.defaultCenter
3350  postNotificationName:UIKeyboardWillShowNotification
3351  object:nil
3352  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3353  FlutterMethodCall* onPointerMoveCall =
3354  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3355  arguments:@{@"pointerY" : @(510)}];
3356  [textInputPlugin handleMethodCall:onPointerMoveCall
3357  result:^(id _Nullable result){
3358  }];
3359  XCTAssertFalse(textInputPlugin.keyboardView.superview == nil);
3360  for (UIView* subView in textInputPlugin.keyboardViewContainer.subviews) {
3361  [subView removeFromSuperview];
3362  }
3363  textInputPlugin.cachedFirstResponder = nil;
3364 }
3365 
3366 - (void)testInteractiveKeyboardScreenshotWillBeMovedDownAfterUserScroll {
3367  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3368  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3369  UIScene* scene = scenes.anyObject;
3370  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3371  UIWindowScene* windowScene = (UIWindowScene*)scene;
3372  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3373  UIWindow* window = windowScene.windows[0];
3374  [window addSubview:viewController.view];
3375 
3376  [viewController loadView];
3377 
3378  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3379  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3380 
3381  [inputView setTextInputClient:123];
3382  [inputView reloadInputViews];
3383  [inputView becomeFirstResponder];
3384 
3385  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3386  [NSNotificationCenter.defaultCenter
3387  postNotificationName:UIKeyboardWillShowNotification
3388  object:nil
3389  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3390  FlutterMethodCall* onPointerMoveCall =
3391  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3392  arguments:@{@"pointerY" : @(510)}];
3393  [textInputPlugin handleMethodCall:onPointerMoveCall
3394  result:^(id _Nullable result){
3395  }];
3396  XCTAssert(textInputPlugin.keyboardView.superview != nil);
3397 
3398  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y, keyboardFrame.origin.y);
3399 
3400  FlutterMethodCall* onPointerMoveCallMove =
3401  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3402  arguments:@{@"pointerY" : @(600)}];
3403  [textInputPlugin handleMethodCall:onPointerMoveCallMove
3404  result:^(id _Nullable result){
3405  }];
3406  XCTAssert(textInputPlugin.keyboardView.superview != nil);
3407 
3408  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y, 600.0);
3409 
3410  for (UIView* subView in textInputPlugin.keyboardViewContainer.subviews) {
3411  [subView removeFromSuperview];
3412  }
3413  textInputPlugin.cachedFirstResponder = nil;
3414 }
3415 
3416 - (void)testInteractiveKeyboardScreenshotWillBeMovedToOrginalPositionAfterUserScroll {
3417  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3418  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3419  UIScene* scene = scenes.anyObject;
3420  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3421  UIWindowScene* windowScene = (UIWindowScene*)scene;
3422  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3423  UIWindow* window = windowScene.windows[0];
3424  [window addSubview:viewController.view];
3425 
3426  [viewController loadView];
3427 
3428  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3429  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3430 
3431  [inputView setTextInputClient:123];
3432  [inputView reloadInputViews];
3433  [inputView becomeFirstResponder];
3434 
3435  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3436  [NSNotificationCenter.defaultCenter
3437  postNotificationName:UIKeyboardWillShowNotification
3438  object:nil
3439  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3440  FlutterMethodCall* onPointerMoveCall =
3441  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3442  arguments:@{@"pointerY" : @(500)}];
3443  [textInputPlugin handleMethodCall:onPointerMoveCall
3444  result:^(id _Nullable result){
3445  }];
3446  XCTAssert(textInputPlugin.keyboardView.superview != nil);
3447  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y, keyboardFrame.origin.y);
3448 
3449  FlutterMethodCall* onPointerMoveCallMove =
3450  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3451  arguments:@{@"pointerY" : @(600)}];
3452  [textInputPlugin handleMethodCall:onPointerMoveCallMove
3453  result:^(id _Nullable result){
3454  }];
3455  XCTAssert(textInputPlugin.keyboardView.superview != nil);
3456  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y, 600.0);
3457 
3458  FlutterMethodCall* onPointerMoveCallBackUp =
3459  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3460  arguments:@{@"pointerY" : @(10)}];
3461  [textInputPlugin handleMethodCall:onPointerMoveCallBackUp
3462  result:^(id _Nullable result){
3463  }];
3464  XCTAssert(textInputPlugin.keyboardView.superview != nil);
3465  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y, keyboardFrame.origin.y);
3466  for (UIView* subView in textInputPlugin.keyboardViewContainer.subviews) {
3467  [subView removeFromSuperview];
3468  }
3469  textInputPlugin.cachedFirstResponder = nil;
3470 }
3471 
3472 - (void)testInteractiveKeyboardFindFirstResponderRecursive {
3473  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3474  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3475  [inputView setTextInputClient:123];
3476  [inputView reloadInputViews];
3477  [inputView becomeFirstResponder];
3478 
3479  UIView* firstResponder = UIApplication.sharedApplication.keyWindow.flutterFirstResponder;
3480  XCTAssertEqualObjects(inputView, firstResponder);
3481  textInputPlugin.cachedFirstResponder = nil;
3482 }
3483 
3484 - (void)testInteractiveKeyboardFindFirstResponderRecursiveInMultipleSubviews {
3485  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3486  FlutterTextInputView* subInputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3487  FlutterTextInputView* otherSubInputView =
3488  [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3489  FlutterTextInputView* subFirstResponderInputView =
3490  [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3491  [subInputView addSubview:subFirstResponderInputView];
3492  [inputView addSubview:subInputView];
3493  [inputView addSubview:otherSubInputView];
3494  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3495  [inputView setTextInputClient:123];
3496  [inputView reloadInputViews];
3497  [subInputView setTextInputClient:123];
3498  [subInputView reloadInputViews];
3499  [otherSubInputView setTextInputClient:123];
3500  [otherSubInputView reloadInputViews];
3501  [subFirstResponderInputView setTextInputClient:123];
3502  [subFirstResponderInputView reloadInputViews];
3503  [subFirstResponderInputView becomeFirstResponder];
3504 
3505  UIView* firstResponder = UIApplication.sharedApplication.keyWindow.flutterFirstResponder;
3506  XCTAssertEqualObjects(subFirstResponderInputView, firstResponder);
3507  textInputPlugin.cachedFirstResponder = nil;
3508 }
3509 
3510 - (void)testInteractiveKeyboardFindFirstResponderIsNilRecursive {
3511  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3512  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3513  [inputView setTextInputClient:123];
3514  [inputView reloadInputViews];
3515 
3516  UIView* firstResponder = UIApplication.sharedApplication.keyWindow.flutterFirstResponder;
3517  XCTAssertNil(firstResponder);
3518  textInputPlugin.cachedFirstResponder = nil;
3519 }
3520 
3521 - (void)testInteractiveKeyboardDidResignFirstResponderDelegateisCalledAfterDismissedKeyboard {
3522  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3523  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3524  UIScene* scene = scenes.anyObject;
3525  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3526  UIWindowScene* windowScene = (UIWindowScene*)scene;
3527  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3528  UIWindow* window = windowScene.windows[0];
3529  [window addSubview:viewController.view];
3530 
3531  [viewController loadView];
3532 
3533  XCTestExpectation* expectation = [[XCTestExpectation alloc]
3534  initWithDescription:
3535  @"didResignFirstResponder is called after screenshot keyboard dismissed."];
3536  OCMStub([engine flutterTextInputView:[OCMArg any] didResignFirstResponderWithTextInputClient:0])
3537  .andDo(^(NSInvocation* invocation) {
3538  [expectation fulfill];
3539  });
3540  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3541  [NSNotificationCenter.defaultCenter
3542  postNotificationName:UIKeyboardWillShowNotification
3543  object:nil
3544  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3545  FlutterMethodCall* initialMoveCall =
3546  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3547  arguments:@{@"pointerY" : @(500)}];
3548  [textInputPlugin handleMethodCall:initialMoveCall
3549  result:^(id _Nullable result){
3550  }];
3551  FlutterMethodCall* subsequentMoveCall =
3552  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3553  arguments:@{@"pointerY" : @(1000)}];
3554  [textInputPlugin handleMethodCall:subsequentMoveCall
3555  result:^(id _Nullable result){
3556  }];
3557 
3558  FlutterMethodCall* pointerUpCall =
3559  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerUpForInteractiveKeyboard"
3560  arguments:@{@"pointerY" : @(1000)}];
3561  [textInputPlugin handleMethodCall:pointerUpCall
3562  result:^(id _Nullable result){
3563  }];
3564 
3565  [self waitForExpectations:@[ expectation ] timeout:2.0];
3566  textInputPlugin.cachedFirstResponder = nil;
3567 }
3568 
3569 - (void)testInteractiveKeyboardScreenshotDismissedAfterPointerLiftedAboveMiddleYOfKeyboard {
3570  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3571  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3572  UIScene* scene = scenes.anyObject;
3573  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3574  UIWindowScene* windowScene = (UIWindowScene*)scene;
3575  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3576  UIWindow* window = windowScene.windows[0];
3577  [window addSubview:viewController.view];
3578 
3579  [viewController loadView];
3580 
3581  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3582  [NSNotificationCenter.defaultCenter
3583  postNotificationName:UIKeyboardWillShowNotification
3584  object:nil
3585  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3586  FlutterMethodCall* initialMoveCall =
3587  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3588  arguments:@{@"pointerY" : @(500)}];
3589  [textInputPlugin handleMethodCall:initialMoveCall
3590  result:^(id _Nullable result){
3591  }];
3592  FlutterMethodCall* subsequentMoveCall =
3593  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3594  arguments:@{@"pointerY" : @(1000)}];
3595  [textInputPlugin handleMethodCall:subsequentMoveCall
3596  result:^(id _Nullable result){
3597  }];
3598 
3599  FlutterMethodCall* subsequentMoveBackUpCall =
3600  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3601  arguments:@{@"pointerY" : @(0)}];
3602  [textInputPlugin handleMethodCall:subsequentMoveBackUpCall
3603  result:^(id _Nullable result){
3604  }];
3605 
3606  FlutterMethodCall* pointerUpCall =
3607  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerUpForInteractiveKeyboard"
3608  arguments:@{@"pointerY" : @(0)}];
3609  [textInputPlugin handleMethodCall:pointerUpCall
3610  result:^(id _Nullable result){
3611  }];
3612  NSPredicate* predicate = [NSPredicate predicateWithBlock:^BOOL(id item, NSDictionary* bindings) {
3613  return textInputPlugin.keyboardViewContainer.subviews.count == 0;
3614  }];
3615  XCTNSPredicateExpectation* expectation =
3616  [[XCTNSPredicateExpectation alloc] initWithPredicate:predicate object:nil];
3617  [self waitForExpectations:@[ expectation ] timeout:10.0];
3618  textInputPlugin.cachedFirstResponder = nil;
3619 }
3620 
3621 - (void)testInteractiveKeyboardKeyboardReappearsAfterPointerLiftedAboveMiddleYOfKeyboard {
3622  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3623  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3624  UIScene* scene = scenes.anyObject;
3625  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3626  UIWindowScene* windowScene = (UIWindowScene*)scene;
3627  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3628  UIWindow* window = windowScene.windows[0];
3629  [window addSubview:viewController.view];
3630 
3631  [viewController loadView];
3632 
3633  FlutterTextInputView* inputView = [[FlutterTextInputView alloc] initWithOwner:textInputPlugin];
3634  [UIApplication.sharedApplication.keyWindow addSubview:inputView];
3635 
3636  [inputView setTextInputClient:123];
3637  [inputView reloadInputViews];
3638  [inputView becomeFirstResponder];
3639 
3640  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3641  [NSNotificationCenter.defaultCenter
3642  postNotificationName:UIKeyboardWillShowNotification
3643  object:nil
3644  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3645  FlutterMethodCall* initialMoveCall =
3646  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3647  arguments:@{@"pointerY" : @(500)}];
3648  [textInputPlugin handleMethodCall:initialMoveCall
3649  result:^(id _Nullable result){
3650  }];
3651  FlutterMethodCall* subsequentMoveCall =
3652  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3653  arguments:@{@"pointerY" : @(1000)}];
3654  [textInputPlugin handleMethodCall:subsequentMoveCall
3655  result:^(id _Nullable result){
3656  }];
3657 
3658  FlutterMethodCall* subsequentMoveBackUpCall =
3659  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3660  arguments:@{@"pointerY" : @(0)}];
3661  [textInputPlugin handleMethodCall:subsequentMoveBackUpCall
3662  result:^(id _Nullable result){
3663  }];
3664 
3665  FlutterMethodCall* pointerUpCall =
3666  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerUpForInteractiveKeyboard"
3667  arguments:@{@"pointerY" : @(0)}];
3668  [textInputPlugin handleMethodCall:pointerUpCall
3669  result:^(id _Nullable result){
3670  }];
3671  NSPredicate* predicate = [NSPredicate predicateWithBlock:^BOOL(id item, NSDictionary* bindings) {
3672  return textInputPlugin.cachedFirstResponder.isFirstResponder;
3673  }];
3674  XCTNSPredicateExpectation* expectation =
3675  [[XCTNSPredicateExpectation alloc] initWithPredicate:predicate object:nil];
3676  [self waitForExpectations:@[ expectation ] timeout:10.0];
3677  textInputPlugin.cachedFirstResponder = nil;
3678 }
3679 
3680 - (void)testInteractiveKeyboardKeyboardAnimatesToOriginalPositionalOnPointerUp {
3681  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3682  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3683  UIScene* scene = scenes.anyObject;
3684  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3685  UIWindowScene* windowScene = (UIWindowScene*)scene;
3686  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3687  UIWindow* window = windowScene.windows[0];
3688  [window addSubview:viewController.view];
3689 
3690  [viewController loadView];
3691 
3692  XCTestExpectation* expectation =
3693  [[XCTestExpectation alloc] initWithDescription:@"Keyboard animates to proper position."];
3694  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3695  [NSNotificationCenter.defaultCenter
3696  postNotificationName:UIKeyboardWillShowNotification
3697  object:nil
3698  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3699  FlutterMethodCall* initialMoveCall =
3700  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3701  arguments:@{@"pointerY" : @(500)}];
3702  [textInputPlugin handleMethodCall:initialMoveCall
3703  result:^(id _Nullable result){
3704  }];
3705  FlutterMethodCall* subsequentMoveCall =
3706  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3707  arguments:@{@"pointerY" : @(1000)}];
3708  [textInputPlugin handleMethodCall:subsequentMoveCall
3709  result:^(id _Nullable result){
3710  }];
3711  FlutterMethodCall* upwardVelocityMoveCall =
3712  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3713  arguments:@{@"pointerY" : @(500)}];
3714  [textInputPlugin handleMethodCall:upwardVelocityMoveCall
3715  result:^(id _Nullable result){
3716  }];
3717 
3718  FlutterMethodCall* pointerUpCall =
3719  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerUpForInteractiveKeyboard"
3720  arguments:@{@"pointerY" : @(0)}];
3721  [textInputPlugin
3722  handleMethodCall:pointerUpCall
3723  result:^(id _Nullable result) {
3724  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y,
3725  viewController.flutterScreenIfViewLoaded.bounds.size.height -
3726  keyboardFrame.origin.y);
3727  [expectation fulfill];
3728  }];
3729  textInputPlugin.cachedFirstResponder = nil;
3730 }
3731 
3732 - (void)testInteractiveKeyboardKeyboardAnimatesToDismissalPositionalOnPointerUp {
3733  NSSet<UIScene*>* scenes = UIApplication.sharedApplication.connectedScenes;
3734  XCTAssertEqual(scenes.count, 1UL, @"There must only be 1 scene for test");
3735  UIScene* scene = scenes.anyObject;
3736  XCTAssert([scene isKindOfClass:[UIWindowScene class]], @"Must be a window scene for test");
3737  UIWindowScene* windowScene = (UIWindowScene*)scene;
3738  XCTAssert(windowScene.windows.count > 0, @"There must be at least 1 window for test");
3739  UIWindow* window = windowScene.windows[0];
3740  [window addSubview:viewController.view];
3741 
3742  [viewController loadView];
3743 
3744  XCTestExpectation* expectation =
3745  [[XCTestExpectation alloc] initWithDescription:@"Keyboard animates to proper position."];
3746  CGRect keyboardFrame = CGRectMake(0, 500, 500, 500);
3747  [NSNotificationCenter.defaultCenter
3748  postNotificationName:UIKeyboardWillShowNotification
3749  object:nil
3750  userInfo:@{UIKeyboardFrameEndUserInfoKey : @(keyboardFrame)}];
3751  FlutterMethodCall* initialMoveCall =
3752  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3753  arguments:@{@"pointerY" : @(500)}];
3754  [textInputPlugin handleMethodCall:initialMoveCall
3755  result:^(id _Nullable result){
3756  }];
3757  FlutterMethodCall* subsequentMoveCall =
3758  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerMoveForInteractiveKeyboard"
3759  arguments:@{@"pointerY" : @(1000)}];
3760  [textInputPlugin handleMethodCall:subsequentMoveCall
3761  result:^(id _Nullable result){
3762  }];
3763 
3764  FlutterMethodCall* pointerUpCall =
3765  [FlutterMethodCall methodCallWithMethodName:@"TextInput.onPointerUpForInteractiveKeyboard"
3766  arguments:@{@"pointerY" : @(1000)}];
3767  [textInputPlugin
3768  handleMethodCall:pointerUpCall
3769  result:^(id _Nullable result) {
3770  XCTAssertEqual(textInputPlugin.keyboardViewContainer.frame.origin.y,
3771  viewController.flutterScreenIfViewLoaded.bounds.size.height);
3772  [expectation fulfill];
3773  }];
3774  textInputPlugin.cachedFirstResponder = nil;
3775 }
3776 - (void)testInteractiveKeyboardShowKeyboardAndRemoveScreenshotAnimationIsNotImmediatelyEnable {
3777  [UIView setAnimationsEnabled:YES];
3778  [textInputPlugin showKeyboardAndRemoveScreenshot];
3779  XCTAssertFalse(
3780  UIView.areAnimationsEnabled,
3781  @"The animation should still be disabled following showKeyboardAndRemoveScreenshot");
3782 }
3783 
3784 - (void)testInteractiveKeyboardShowKeyboardAndRemoveScreenshotAnimationIsReenabledAfterDelay {
3785  [UIView setAnimationsEnabled:YES];
3786  [textInputPlugin showKeyboardAndRemoveScreenshot];
3787 
3788  NSPredicate* predicate = [NSPredicate predicateWithBlock:^BOOL(id item, NSDictionary* bindings) {
3789  // This will be enabled after a delay
3790  return UIView.areAnimationsEnabled;
3791  }];
3792  XCTNSPredicateExpectation* expectation =
3793  [[XCTNSPredicateExpectation alloc] initWithPredicate:predicate object:nil];
3794  [self waitForExpectations:@[ expectation ] timeout:10.0];
3795 }
3796 
3797 @end
FlutterTextInputViewSpy
Definition: FlutterTextInputPluginTest.mm:38
caretRectForPosition
CGRect caretRectForPosition
Definition: FlutterTextInputPlugin.h:178
+[FlutterTextPosition positionWithIndex:]
instancetype positionWithIndex:(NSUInteger index)
Definition: FlutterTextInputPlugin.mm:525
selectionRects
NSArray< FlutterTextSelectionRect * > * selectionRects
Definition: FlutterTextInputPlugin.h:163
FlutterEngine
Definition: FlutterEngine.h:61
FlutterSecureTextInputView::textField
UITextField * textField
Definition: FlutterTextInputPlugin.mm:753
FlutterTextInputDelegate-p
Definition: FlutterTextInputDelegate.h:37
+[FlutterMethodCall methodCallWithMethodName:arguments:]
instancetype methodCallWithMethodName:arguments:(NSString *method,[arguments] id _Nullable arguments)
FlutterViewController
Definition: FlutterViewController.h:57
FlutterEngine.h
isScribbleAvailable
BOOL isScribbleAvailable
Definition: FlutterTextInputPlugin.h:167
-[FlutterEngine runWithEntrypoint:]
BOOL runWithEntrypoint:(nullable NSString *entrypoint)
FlutterEngine_Test.h
-[FlutterEngine flutterTextInputView:performAction:withClient:]
void flutterTextInputView:performAction:withClient:(FlutterTextInputView *textInputView,[performAction] FlutterTextInputAction action,[withClient] int client)
FlutterTextInputPlugin.h
FlutterTextSelectionRect::rect
CGRect rect
Definition: FlutterTextInputPlugin.h:95
-[FlutterTextInputPlugin showEditMenu:]
BOOL showEditMenu:(ios(16.0) API_AVAILABLE)
Definition: FlutterTextInputPlugin.mm:2684
FlutterTextRange
Definition: FlutterTextInputPlugin.h:81
FlutterMacros.h
-[FlutterEngine setBinaryMessenger:]
void setBinaryMessenger:(FlutterBinaryMessengerRelay *binaryMessenger)
FlutterTextInputViewSpy::isAccessibilityFocused
BOOL isAccessibilityFocused
Definition: FlutterTextInputPluginTest.mm:41
-[FlutterTextInputPlugin handleMethodCall:result:]
void handleMethodCall:result:(FlutterMethodCall *call,[result] FlutterResult result)
Definition: FlutterTextInputPlugin.mm:2521
-[FlutterTextInputPlugin textInputView]
UIView< UITextInput > * textInputView()
Definition: FlutterTextInputPlugin.mm:2517
kInvalidFirstRect
const CGRect kInvalidFirstRect
Definition: FlutterTextInputPlugin.mm:35
viewController
FlutterViewController * viewController
Definition: FlutterTextInputPluginTest.mm:95
+[FlutterTextRange rangeWithNSRange:]
instancetype rangeWithNSRange:(NSRange range)
Definition: FlutterTextInputPlugin.mm:548
FlutterSecureTextInputView
Definition: FlutterTextInputPlugin.mm:752
FlutterTextInputView
Definition: FlutterTextInputPlugin.mm:810
FlutterTextInputViewSpy::receivedNotification
UIAccessibilityNotifications receivedNotification
Definition: FlutterTextInputPluginTest.mm:39
FlutterBinaryMessengerRelay.h
selectedTextRange
API_AVAILABLE(ios(13.0)) @interface FlutterTextPlaceholder UITextRange * selectedTextRange
Definition: FlutterTextInputPlugin.h:127
FlutterMethodCall
Definition: FlutterCodecs.h:220
+[FlutterTextSelectionRect selectionRectWithRect:position:]
instancetype selectionRectWithRect:position:(CGRect rect,[position] NSUInteger position)
Definition: FlutterTextInputPlugin.mm:689
FlutterTextRange::range
NSRange range
Definition: FlutterTextInputPlugin.h:83
FlutterTextInputPlugin
Definition: FlutterTextInputPlugin.h:33
FlutterTextPosition::affinity
UITextStorageDirection affinity
Definition: FlutterTextInputPlugin.h:72
UIViewController+FlutterScreenAndSceneIfLoaded.h
FlutterTextInputPluginTest
Definition: FlutterTextInputPluginTest.mm:86
FlutterTextSelectionRect::position
NSUInteger position
Definition: FlutterTextInputPlugin.h:96
_passwordTemplate
NSDictionary * _passwordTemplate
Definition: FlutterTextInputPluginTest.mm:89
engine
id engine
Definition: FlutterTextInputPluginTest.mm:92
textInputPlugin
FlutterTextInputPlugin * textInputPlugin
Definition: FlutterTextInputPluginTest.mm:93
FlutterTextPosition
Definition: FlutterTextInputPlugin.h:69
FlutterBinaryMessengerRelay
Definition: FlutterBinaryMessengerRelay.h:14
FlutterJSONMethodCodec
Definition: FlutterCodecs.h:455
FlutterTextInputPlugin::viewController
UIIndirectScribbleInteractionDelegate UIViewController * viewController
Definition: FlutterTextInputPlugin.h:36
FlutterEngine::viewController
FlutterViewController * viewController
Definition: FlutterEngine.h:327
FlutterTextPosition::index
NSUInteger index
Definition: FlutterTextInputPlugin.h:71
-[FlutterEngine runWithEntrypoint:initialRoute:]
BOOL runWithEntrypoint:initialRoute:(nullable NSString *entrypoint,[initialRoute] nullable NSString *initialRoute)
FlutterTextSelectionRect
Definition: FlutterTextInputPlugin.h:93
+[FlutterTextSelectionRect selectionRectWithRect:position:writingDirection:]
instancetype selectionRectWithRect:position:writingDirection:(CGRect rect,[position] NSUInteger position,[writingDirection] NSWritingDirection writingDirection)
Definition: FlutterTextInputPlugin.mm:698
FLUTTER_ASSERT_ARC
Definition: FlutterChannelKeyResponder.mm:13
markedTextRange
UITextRange * markedTextRange
Definition: FlutterTextInputPlugin.h:139
FlutterTextInputViewSpy::receivedNotificationTarget
id receivedNotificationTarget
Definition: FlutterTextInputPluginTest.mm:40
FlutterViewController.h