8 #import <Foundation/Foundation.h>
9 #import <UIKit/UIKit.h>
11 #include "unicode/uchar.h"
13 #include "flutter/fml/logging.h"
14 #include "flutter/fml/platform/darwin/string_range_sanitization.h"
37 #pragma mark - TextInput channel method names.
46 @"TextInput.setEditableSizeAndTransform";
57 @"TextInput.onPointerMoveForInteractiveKeyboard";
59 @"TextInput.onPointerUpForInteractiveKeyboard";
61 #pragma mark - TextInputConfiguration Field Names
82 #pragma mark - Static Functions
85 static BOOL
IsEmoji(NSString* text, NSRange charRange) {
87 BOOL gotCodePoint = [text getBytes:&codePoint
88 maxLength:
sizeof(codePoint)
90 encoding:NSUTF32StringEncoding
94 return gotCodePoint && u_hasBinaryProperty(codePoint, UCHAR_EMOJI);
102 NSString* inputType = type[
@"name"];
103 return ![inputType isEqualToString:
@"TextInputType.none"];
106 NSString* inputType = type[
@"name"];
107 if ([inputType isEqualToString:
@"TextInputType.address"]) {
108 return UIKeyboardTypeDefault;
110 if ([inputType isEqualToString:
@"TextInputType.datetime"]) {
111 return UIKeyboardTypeNumbersAndPunctuation;
113 if ([inputType isEqualToString:
@"TextInputType.emailAddress"]) {
114 return UIKeyboardTypeEmailAddress;
116 if ([inputType isEqualToString:
@"TextInputType.multiline"]) {
117 return UIKeyboardTypeDefault;
119 if ([inputType isEqualToString:
@"TextInputType.name"]) {
120 return UIKeyboardTypeNamePhonePad;
122 if ([inputType isEqualToString:
@"TextInputType.number"]) {
123 if ([type[
@"signed"] boolValue]) {
124 return UIKeyboardTypeNumbersAndPunctuation;
126 if ([type[
@"decimal"] boolValue]) {
127 return UIKeyboardTypeDecimalPad;
129 return UIKeyboardTypeNumberPad;
131 if ([inputType isEqualToString:
@"TextInputType.phone"]) {
132 return UIKeyboardTypePhonePad;
134 if ([inputType isEqualToString:
@"TextInputType.text"]) {
135 return UIKeyboardTypeDefault;
137 if ([inputType isEqualToString:
@"TextInputType.url"]) {
138 return UIKeyboardTypeURL;
140 if ([inputType isEqualToString:
@"TextInputType.visiblePassword"]) {
141 return UIKeyboardTypeASCIICapable;
143 if ([inputType isEqualToString:
@"TextInputType.webSearch"]) {
144 return UIKeyboardTypeWebSearch;
146 if ([inputType isEqualToString:
@"TextInputType.twitter"]) {
147 return UIKeyboardTypeTwitter;
149 return UIKeyboardTypeDefault;
153 NSString* textCapitalization = type[
@"textCapitalization"];
154 if ([textCapitalization isEqualToString:
@"TextCapitalization.characters"]) {
155 return UITextAutocapitalizationTypeAllCharacters;
156 }
else if ([textCapitalization isEqualToString:
@"TextCapitalization.sentences"]) {
157 return UITextAutocapitalizationTypeSentences;
158 }
else if ([textCapitalization isEqualToString:
@"TextCapitalization.words"]) {
159 return UITextAutocapitalizationTypeWords;
161 return UITextAutocapitalizationTypeNone;
169 if ([inputType isEqualToString:
@"TextInputAction.unspecified"]) {
170 return UIReturnKeyDefault;
173 if ([inputType isEqualToString:
@"TextInputAction.done"]) {
174 return UIReturnKeyDone;
177 if ([inputType isEqualToString:
@"TextInputAction.go"]) {
178 return UIReturnKeyGo;
181 if ([inputType isEqualToString:
@"TextInputAction.send"]) {
182 return UIReturnKeySend;
185 if ([inputType isEqualToString:
@"TextInputAction.search"]) {
186 return UIReturnKeySearch;
189 if ([inputType isEqualToString:
@"TextInputAction.next"]) {
190 return UIReturnKeyNext;
193 if ([inputType isEqualToString:
@"TextInputAction.continueAction"]) {
194 return UIReturnKeyContinue;
197 if ([inputType isEqualToString:
@"TextInputAction.join"]) {
198 return UIReturnKeyJoin;
201 if ([inputType isEqualToString:
@"TextInputAction.route"]) {
202 return UIReturnKeyRoute;
205 if ([inputType isEqualToString:
@"TextInputAction.emergencyCall"]) {
206 return UIReturnKeyEmergencyCall;
209 if ([inputType isEqualToString:
@"TextInputAction.newline"]) {
210 return UIReturnKeyDefault;
214 return UIReturnKeyDefault;
218 if (!hints || hints.count == 0) {
223 NSString* hint = hints[0];
224 if ([hint isEqualToString:
@"addressCityAndState"]) {
225 return UITextContentTypeAddressCityAndState;
228 if ([hint isEqualToString:
@"addressState"]) {
229 return UITextContentTypeAddressState;
232 if ([hint isEqualToString:
@"addressCity"]) {
233 return UITextContentTypeAddressCity;
236 if ([hint isEqualToString:
@"sublocality"]) {
237 return UITextContentTypeSublocality;
240 if ([hint isEqualToString:
@"streetAddressLine1"]) {
241 return UITextContentTypeStreetAddressLine1;
244 if ([hint isEqualToString:
@"streetAddressLine2"]) {
245 return UITextContentTypeStreetAddressLine2;
248 if ([hint isEqualToString:
@"countryName"]) {
249 return UITextContentTypeCountryName;
252 if ([hint isEqualToString:
@"fullStreetAddress"]) {
253 return UITextContentTypeFullStreetAddress;
256 if ([hint isEqualToString:
@"postalCode"]) {
257 return UITextContentTypePostalCode;
260 if ([hint isEqualToString:
@"location"]) {
261 return UITextContentTypeLocation;
264 if ([hint isEqualToString:
@"creditCardNumber"]) {
265 return UITextContentTypeCreditCardNumber;
268 if ([hint isEqualToString:
@"email"]) {
269 return UITextContentTypeEmailAddress;
272 if ([hint isEqualToString:
@"jobTitle"]) {
273 return UITextContentTypeJobTitle;
276 if ([hint isEqualToString:
@"givenName"]) {
277 return UITextContentTypeGivenName;
280 if ([hint isEqualToString:
@"middleName"]) {
281 return UITextContentTypeMiddleName;
284 if ([hint isEqualToString:
@"familyName"]) {
285 return UITextContentTypeFamilyName;
288 if ([hint isEqualToString:
@"name"]) {
289 return UITextContentTypeName;
292 if ([hint isEqualToString:
@"namePrefix"]) {
293 return UITextContentTypeNamePrefix;
296 if ([hint isEqualToString:
@"nameSuffix"]) {
297 return UITextContentTypeNameSuffix;
300 if ([hint isEqualToString:
@"nickname"]) {
301 return UITextContentTypeNickname;
304 if ([hint isEqualToString:
@"organizationName"]) {
305 return UITextContentTypeOrganizationName;
308 if ([hint isEqualToString:
@"telephoneNumber"]) {
309 return UITextContentTypeTelephoneNumber;
312 if ([hint isEqualToString:
@"password"]) {
313 return UITextContentTypePassword;
316 if ([hint isEqualToString:
@"oneTimeCode"]) {
317 return UITextContentTypeOneTimeCode;
320 if ([hint isEqualToString:
@"newPassword"]) {
321 return UITextContentTypeNewPassword;
389 typedef NS_ENUM(NSInteger, FlutterAutofillType) {
393 kFlutterAutofillTypeNone,
394 kFlutterAutofillTypeRegular,
395 kFlutterAutofillTypePassword,
405 if (isSecureTextEntry) {
412 if ([contentType isEqualToString:UITextContentTypePassword] ||
413 [contentType isEqualToString:UITextContentTypeUsername]) {
417 if ([contentType isEqualToString:UITextContentTypeNewPassword]) {
427 return kFlutterAutofillTypePassword;
432 return kFlutterAutofillTypePassword;
437 return !autofill || [contentType isEqualToString:
@""] ? kFlutterAutofillTypeNone
438 : kFlutterAutofillTypeRegular;
442 return fabsf(x - y) <= delta;
468 CGRect selectionRect,
469 BOOL selectionRectIsRTL,
470 BOOL useTrailingBoundaryOfSelectionRect,
471 CGRect otherSelectionRect,
472 BOOL otherSelectionRectIsRTL,
473 CGFloat verticalPrecision) {
475 if (CGRectContainsPoint(
477 selectionRect.origin.x + ((useTrailingBoundaryOfSelectionRect ^ selectionRectIsRTL)
478 ? 0.5 * selectionRect.size.width
480 selectionRect.origin.y, 0.5 * selectionRect.size.width, selectionRect.size.height),
485 CGPoint pointForSelectionRect = CGPointMake(
486 selectionRect.origin.x +
487 (selectionRectIsRTL ^ useTrailingBoundaryOfSelectionRect ? selectionRect.size.width : 0),
488 selectionRect.origin.y + selectionRect.size.height * 0.5);
489 float yDist = fabs(pointForSelectionRect.y - point.y);
490 float xDist = fabs(pointForSelectionRect.x - point.x);
493 CGPoint pointForOtherSelectionRect = CGPointMake(
494 otherSelectionRect.origin.x + (otherSelectionRectIsRTL ? otherSelectionRect.size.width : 0),
495 otherSelectionRect.origin.y + otherSelectionRect.size.height * 0.5);
496 float yDistOther = fabs(pointForOtherSelectionRect.y - point.y);
497 float xDistOther = fabs(pointForOtherSelectionRect.x - point.x);
502 BOOL isCloserVertically = yDist < yDistOther - verticalPrecision;
504 BOOL isAboveBottomOfLine = point.y <= selectionRect.origin.y + selectionRect.size.height;
505 BOOL isCloserHorizontally = xDist < xDistOther;
506 BOOL isBelowBottomOfLine = point.y > selectionRect.origin.y + selectionRect.size.height;
509 if (selectionRectIsRTL) {
510 isFarther = selectionRect.origin.x < otherSelectionRect.origin.x;
512 isFarther = selectionRect.origin.x +
513 (useTrailingBoundaryOfSelectionRect ? selectionRect.size.width : 0) >
514 otherSelectionRect.origin.x;
516 return (isCloserVertically ||
517 (isEqualVertically &&
518 ((isAboveBottomOfLine && isCloserHorizontally) || (isBelowBottomOfLine && isFarther))));
521 #pragma mark - FlutterTextPosition
525 + (instancetype)positionWithIndex:(NSUInteger)index {
526 return [[
FlutterTextPosition alloc] initWithIndex:index affinity:UITextStorageDirectionForward];
529 + (instancetype)positionWithIndex:(NSUInteger)index affinity:(UITextStorageDirection)affinity {
533 - (instancetype)initWithIndex:(NSUInteger)index affinity:(UITextStorageDirection)affinity {
544 #pragma mark - FlutterTextRange
548 + (instancetype)rangeWithNSRange:(NSRange)range {
552 - (instancetype)initWithNSRange:(NSRange)range {
560 - (UITextPosition*)start {
562 affinity:UITextStorageDirectionForward];
565 - (UITextPosition*)end {
567 affinity:UITextStorageDirectionBackward];
571 return self.range.length == 0;
574 - (id)copyWithZone:(NSZone*)zone {
579 return NSEqualRanges(
self.
range, other.
range);
583 #pragma mark - FlutterTokenizer
593 - (instancetype)initWithTextInput:(UIResponder<UITextInput>*)textInput {
595 @"The FlutterTokenizer can only be used in a FlutterTextInputView");
596 self = [
super initWithTextInput:textInput];
603 - (UITextRange*)rangeEnclosingPosition:(UITextPosition*)position
604 withGranularity:(UITextGranularity)granularity
605 inDirection:(UITextDirection)direction {
607 switch (granularity) {
608 case UITextGranularityLine:
611 result = [
self lineEnclosingPosition:position inDirection:direction];
613 case UITextGranularityCharacter:
614 case UITextGranularityWord:
615 case UITextGranularitySentence:
616 case UITextGranularityParagraph:
617 case UITextGranularityDocument:
619 result = [
super rangeEnclosingPosition:position
620 withGranularity:granularity
621 inDirection:direction];
627 - (UITextRange*)lineEnclosingPosition:(UITextPosition*)position
628 inDirection:(UITextDirection)direction {
630 if (@available(iOS 17.0, *)) {
635 if (flutterPosition.
index > _textInputView.text.length ||
636 (flutterPosition.
index == _textInputView.text.length &&
637 direction == UITextStorageDirectionForward)) {
643 NSString* textAfter = [_textInputView
644 textInRange:[_textInputView textRangeFromPosition:position
645 toPosition:[_textInputView endOfDocument]]];
646 NSArray<NSString*>* linesAfter = [textAfter componentsSeparatedByString:@"\n"];
647 NSInteger offSetToLineBreak = [linesAfter firstObject].length;
648 UITextPosition* lineBreakAfter = [_textInputView positionFromPosition:position
649 offset:offSetToLineBreak];
651 NSString* textBefore = [_textInputView
652 textInRange:[_textInputView textRangeFromPosition:[_textInputView beginningOfDocument]
653 toPosition:position]];
654 NSArray<NSString*>* linesBefore = [textBefore componentsSeparatedByString:@"\n"];
655 NSInteger offSetFromLineBreak = [linesBefore lastObject].length;
656 UITextPosition* lineBreakBefore = [_textInputView positionFromPosition:position
657 offset:-offSetFromLineBreak];
659 return [_textInputView textRangeFromPosition:lineBreakBefore toPosition:lineBreakAfter];
664 #pragma mark - FlutterTextSelectionRect
669 @synthesize rect = _rect;
675 + (instancetype)selectionRectWithRectAndInfo:(CGRect)rect
676 position:(NSUInteger)position
677 writingDirection:(NSWritingDirection)writingDirection
678 containsStart:(BOOL)containsStart
679 containsEnd:(BOOL)containsEnd
680 isVertical:(BOOL)isVertical {
683 writingDirection:writingDirection
684 containsStart:containsStart
685 containsEnd:containsEnd
686 isVertical:isVertical];
689 + (instancetype)selectionRectWithRect:(CGRect)rect position:(NSUInteger)position {
692 writingDirection:NSWritingDirectionNatural
698 + (instancetype)selectionRectWithRect:(CGRect)rect
699 position:(NSUInteger)position
700 writingDirection:(NSWritingDirection)writingDirection {
703 writingDirection:writingDirection
709 - (instancetype)initWithRectAndInfo:(CGRect)rect
710 position:(NSUInteger)position
711 writingDirection:(NSWritingDirection)writingDirection
712 containsStart:(BOOL)containsStart
713 containsEnd:(BOOL)containsEnd
714 isVertical:(BOOL)isVertical {
728 return _writingDirection == NSWritingDirectionRightToLeft;
733 #pragma mark - FlutterTextPlaceholder
737 - (NSArray<UITextSelectionRect*>*)rects {
753 @property(nonatomic, retain, readonly) UITextField*
textField;
757 UITextField* _textField;
762 _textField = [[UITextField alloc] init];
767 - (BOOL)isKindOfClass:(Class)aClass {
768 return [
super isKindOfClass:aClass] || (aClass == [UITextField class]);
771 - (NSMethodSignature*)methodSignatureForSelector:(
SEL)aSelector {
772 NSMethodSignature* signature = [
super methodSignatureForSelector:aSelector];
774 signature = [
self.textField methodSignatureForSelector:aSelector];
779 - (void)forwardInvocation:(NSInvocation*)anInvocation {
780 [anInvocation invokeWithTarget:self.textField];
786 @property(nonatomic, readonly, weak) id<FlutterTextInputDelegate> textInputDelegate;
787 @property(nonatomic, readonly) UIView* hostView;
792 @property(nonatomic, copy) NSString* autofillId;
793 @property(nonatomic, readonly) CATransform3D editableTransform;
794 @property(nonatomic, assign) CGRect markedRect;
796 @property(nonatomic, assign) BOOL preventCursorDismissWhenResignFirstResponder;
797 @property(nonatomic) BOOL isVisibleToAutofill;
798 @property(nonatomic, assign) BOOL accessibilityEnabled;
799 @property(nonatomic, assign)
int textInputClient;
803 @property(nonatomic, copy) NSString* temporarilyDeletedComposedCharacter;
804 @property(nonatomic, assign) CGRect editMenuTargetRect;
805 @property(nonatomic, strong) NSArray<NSDictionary*>* editMenuItems;
807 - (void)setEditableTransform:(NSArray*)matrix;
811 int _textInputClient;
825 UITextInteraction* _textInteraction
API_AVAILABLE(ios(13.0));
828 @synthesize tokenizer = _tokenizer;
831 self = [
super initWithFrame:CGRectZero];
834 _textInputClient = 0;
836 _preventCursorDismissWhenResignFirstResponder = NO;
839 _text = [[NSMutableString alloc] init];
844 _pendingDeltas = [[NSMutableArray alloc] init];
847 _editableTransform = CATransform3D();
850 _autocapitalizationType = UITextAutocapitalizationTypeSentences;
851 _autocorrectionType = UITextAutocorrectionTypeDefault;
852 _spellCheckingType = UITextSpellCheckingTypeDefault;
853 _enablesReturnKeyAutomatically = NO;
854 _keyboardAppearance = UIKeyboardAppearanceDefault;
855 _keyboardType = UIKeyboardTypeDefault;
856 _returnKeyType = UIReturnKeyDone;
857 _secureTextEntry = NO;
858 _enableDeltaModel = NO;
860 _accessibilityEnabled = NO;
861 _smartQuotesType = UITextSmartQuotesTypeYes;
862 _smartDashesType = UITextSmartDashesTypeYes;
863 _selectionRects = [[NSArray alloc] init];
865 if (@available(iOS 14.0, *)) {
866 UIScribbleInteraction* interaction = [[UIScribbleInteraction alloc] initWithDelegate:self];
867 [
self addInteraction:interaction];
871 if (@available(iOS 16.0, *)) {
872 _editMenuInteraction = [[UIEditMenuInteraction alloc] initWithDelegate:self];
873 [
self addInteraction:_editMenuInteraction];
879 - (void)handleSearchWebAction {
880 [
self.textInputDelegate flutterTextInputView:self
881 searchWebWithSelectedText:[
self textInRange:_selectedTextRange]];
884 - (void)handleLookUpAction {
885 [
self.textInputDelegate flutterTextInputView:self
886 lookUpSelectedText:[
self textInRange:_selectedTextRange]];
889 - (void)handleShareAction {
890 [
self.textInputDelegate flutterTextInputView:self
891 shareSelectedText:[
self textInRange:_selectedTextRange]];
895 - (UICommand*)searchCommandWithSelector:(
SEL)selector
896 element:(UIMenuElement*)element API_AVAILABLE(ios(16.0)) {
897 if ([element isKindOfClass:UICommand.class]) {
898 UICommand* command = (UICommand*)element;
899 return command.action == selector ? command : nil;
900 }
else if ([element isKindOfClass:UIMenu.class]) {
901 NSArray<UIMenuElement*>* children = ((UIMenu*)element).children;
902 for (UIMenuElement* child in children) {
903 UICommand* result = [
self searchCommandWithSelector:selector element:child];
914 - (void)addBasicEditingCommandToItems:(NSMutableArray*)items
916 selector:(
SEL)selector
917 suggestedMenu:(UIMenu*)suggestedMenu {
918 UICommand* command = [
self searchCommandWithSelector:selector element:suggestedMenu];
920 [items addObject:command];
922 FML_LOG(ERROR) <<
"Cannot find context menu item of type \"" << type.UTF8String <<
"\".";
926 - (void)addAdditionalBasicCommandToItems:(NSMutableArray*)items
928 selector:(
SEL)selector
929 encodedItem:(NSDictionary<NSString*,
id>*)encodedItem {
930 NSString* title = encodedItem[@"title"];
932 UICommand* command = [UICommand commandWithTitle:title
936 [items addObject:command];
938 FML_LOG(ERROR) <<
"Missing title for context menu item of type \"" << type.UTF8String <<
"\".";
942 - (UIMenu*)editMenuInteraction:(UIEditMenuInteraction*)interaction
943 menuForConfiguration:(UIEditMenuConfiguration*)configuration
944 suggestedActions:(NSArray<UIMenuElement*>*)suggestedActions API_AVAILABLE(ios(16.0)) {
945 UIMenu* suggestedMenu = [UIMenu menuWithChildren:suggestedActions];
946 if (!_editMenuItems) {
947 return suggestedMenu;
950 NSMutableArray* items = [NSMutableArray array];
951 for (NSDictionary<NSString*, id>* encodedItem in _editMenuItems) {
952 NSString* type = encodedItem[@"type"];
953 if ([type isEqualToString:
@"copy"]) {
954 [
self addBasicEditingCommandToItems:items
956 selector:@selector(copy:)
957 suggestedMenu:suggestedMenu];
958 }
else if ([type isEqualToString:
@"paste"]) {
959 [
self addBasicEditingCommandToItems:items
961 selector:@selector(paste:)
962 suggestedMenu:suggestedMenu];
963 }
else if ([type isEqualToString:
@"cut"]) {
964 [
self addBasicEditingCommandToItems:items
966 selector:@selector(cut:)
967 suggestedMenu:suggestedMenu];
968 }
else if ([type isEqualToString:
@"delete"]) {
969 [
self addBasicEditingCommandToItems:items
971 selector:@selector(delete:)
972 suggestedMenu:suggestedMenu];
973 }
else if ([type isEqualToString:
@"selectAll"]) {
974 [
self addBasicEditingCommandToItems:items
976 selector:@selector(selectAll:)
977 suggestedMenu:suggestedMenu];
978 }
else if ([type isEqualToString:
@"searchWeb"]) {
979 [
self addAdditionalBasicCommandToItems:items
981 selector:@selector(handleSearchWebAction)
982 encodedItem:encodedItem];
983 }
else if ([type isEqualToString:
@"share"]) {
984 [
self addAdditionalBasicCommandToItems:items
986 selector:@selector(handleShareAction)
987 encodedItem:encodedItem];
988 }
else if ([type isEqualToString:
@"lookUp"]) {
989 [
self addAdditionalBasicCommandToItems:items
991 selector:@selector(handleLookUpAction)
992 encodedItem:encodedItem];
995 return [UIMenu menuWithChildren:items];
998 - (void)editMenuInteraction:(UIEditMenuInteraction*)interaction
999 willDismissMenuForConfiguration:(UIEditMenuConfiguration*)configuration
1000 animator:(
id<UIEditMenuInteractionAnimating>)animator
1001 API_AVAILABLE(ios(16.0)) {
1002 [
self.textInputDelegate flutterTextInputView:self
1003 willDismissEditMenuWithTextInputClient:_textInputClient];
1006 - (CGRect)editMenuInteraction:(UIEditMenuInteraction*)interaction
1007 targetRectForConfiguration:(UIEditMenuConfiguration*)configuration API_AVAILABLE(ios(16.0)) {
1008 return _editMenuTargetRect;
1011 - (void)showEditMenuWithTargetRect:(CGRect)targetRect
1012 items:(NSArray<NSDictionary*>*)items API_AVAILABLE(ios(16.0)) {
1013 _editMenuTargetRect = targetRect;
1014 _editMenuItems = items;
1015 UIEditMenuConfiguration* config =
1016 [UIEditMenuConfiguration configurationWithIdentifier:nil sourcePoint:CGPointZero];
1017 [
self.editMenuInteraction presentEditMenuWithConfiguration:config];
1021 [
self.editMenuInteraction dismissMenu];
1024 - (void)configureWithDictionary:(NSDictionary*)configuration {
1025 NSDictionary* inputType = configuration[kKeyboardType];
1027 NSDictionary* autofill = configuration[kAutofillProperties];
1029 self.secureTextEntry = [configuration[kSecureTextEntry] boolValue];
1030 self.enableDeltaModel = [configuration[kEnableDeltaModel] boolValue];
1037 NSString* smartDashesType = configuration[kSmartDashesType];
1039 bool smartDashesIsDisabled = smartDashesType && [smartDashesType isEqualToString:@"0"];
1040 self.smartDashesType = smartDashesIsDisabled ? UITextSmartDashesTypeNo : UITextSmartDashesTypeYes;
1041 NSString* smartQuotesType = configuration[kSmartQuotesType];
1043 bool smartQuotesIsDisabled = smartQuotesType && [smartQuotesType isEqualToString:@"0"];
1044 self.smartQuotesType = smartQuotesIsDisabled ? UITextSmartQuotesTypeNo : UITextSmartQuotesTypeYes;
1046 self.keyboardAppearance = UIKeyboardAppearanceDark;
1048 self.keyboardAppearance = UIKeyboardAppearanceLight;
1050 self.keyboardAppearance = UIKeyboardAppearanceDefault;
1052 NSString* autocorrect = configuration[kAutocorrectionType];
1053 bool autocorrectIsDisabled = autocorrect && ![autocorrect boolValue];
1054 self.autocorrectionType =
1055 autocorrectIsDisabled ? UITextAutocorrectionTypeNo : UITextAutocorrectionTypeDefault;
1056 self.spellCheckingType =
1057 autocorrectIsDisabled ? UITextSpellCheckingTypeNo : UITextSpellCheckingTypeDefault;
1059 if (autofill == nil) {
1060 self.textContentType =
@"";
1063 [
self setTextInputState:autofill[kAutofillEditingValue]];
1064 NSAssert(_autofillId,
@"The autofill configuration must contain an autofill id");
1068 self.isVisibleToAutofill = autofill || _secureTextEntry;
1071 - (UITextContentType)textContentType {
1072 return _textContentType;
1085 - (UIColor*)insertionPointColor {
1086 return [UIColor clearColor];
1089 - (UIColor*)selectionBarColor {
1090 return [UIColor clearColor];
1093 - (UIColor*)selectionHighlightColor {
1094 return [UIColor clearColor];
1097 - (UIInputViewController*)inputViewController {
1109 return _textInputPlugin.textInputDelegate;
1112 - (BOOL)respondsToSelector:(
SEL)selector {
1113 if (@available(iOS 17.0, *)) {
1115 if (selector ==
@selector(insertionPointColor)) {
1119 return [
super respondsToSelector:selector];
1122 - (void)setTextInputClient:(
int)client {
1123 _textInputClient = client;
1127 - (UITextInteraction*)textInteraction
API_AVAILABLE(ios(13.0)) {
1128 if (!_textInteraction) {
1129 _textInteraction = [UITextInteraction textInteractionForMode:UITextInteractionModeEditable];
1130 _textInteraction.textInput =
self;
1132 return _textInteraction;
1135 - (void)setTextInputState:(NSDictionary*)state {
1136 if (@available(iOS 13.0, *)) {
1143 [
self addInteraction:self.textInteraction];
1147 NSString* newText = state[@"text"];
1148 BOOL textChanged = ![
self.text isEqualToString:newText];
1150 [
self.inputDelegate textWillChange:self];
1151 [
self.text setString:newText];
1153 NSInteger composingBase = [state[@"composingBase"] intValue];
1154 NSInteger composingExtent = [state[@"composingExtent"] intValue];
1155 NSRange composingRange = [
self clampSelection:NSMakeRange(MIN(composingBase, composingExtent),
1156 ABS(composingBase - composingExtent))
1159 self.markedTextRange =
1162 NSRange selectedRange = [
self clampSelectionFromBase:[state[@"selectionBase"] intValue]
1163 extent:[state[@"selectionExtent"] intValue]
1166 NSRange oldSelectedRange = [(
FlutterTextRange*)
self.selectedTextRange range];
1167 if (!NSEqualRanges(selectedRange, oldSelectedRange)) {
1168 [
self.inputDelegate selectionWillChange:self];
1176 [
self.inputDelegate selectionDidChange:self];
1180 [
self.inputDelegate textDidChange:self];
1183 if (@available(iOS 13.0, *)) {
1184 if (_textInteraction) {
1185 [
self removeInteraction:_textInteraction];
1191 - (void)touchesBegan:(NSSet*)touches withEvent:(UIEvent*)event {
1192 _scribbleFocusStatus = FlutterScribbleFocusStatusUnfocused;
1193 [
self resetScribbleInteractionStatusIfEnding];
1194 [
self.viewResponder touchesBegan:touches withEvent:event];
1197 - (void)touchesMoved:(NSSet*)touches withEvent:(UIEvent*)event {
1198 [
self.viewResponder touchesMoved:touches withEvent:event];
1201 - (void)touchesEnded:(NSSet*)touches withEvent:(UIEvent*)event {
1202 [
self.viewResponder touchesEnded:touches withEvent:event];
1205 - (void)touchesCancelled:(NSSet*)touches withEvent:(UIEvent*)event {
1206 [
self.viewResponder touchesCancelled:touches withEvent:event];
1209 - (void)touchesEstimatedPropertiesUpdated:(NSSet*)touches {
1210 [
self.viewResponder touchesEstimatedPropertiesUpdated:touches];
1220 - (NSRange)clampSelectionFromBase:(
int)selectionBase
1221 extent:(
int)selectionExtent
1222 forText:(NSString*)text {
1223 int loc = MIN(selectionBase, selectionExtent);
1224 int len = ABS(selectionExtent - selectionBase);
1225 return loc < 0 ? NSMakeRange(0, 0)
1226 : [
self clampSelection:NSMakeRange(loc, len) forText:
self.text];
1229 - (NSRange)clampSelection:(NSRange)range forText:(NSString*)text {
1230 NSUInteger start = MIN(MAX(range.location, 0), text.length);
1231 NSUInteger length = MIN(range.length, text.length - start);
1232 return NSMakeRange(start, length);
1235 - (BOOL)isVisibleToAutofill {
1236 return self.frame.size.width > 0 &&
self.frame.size.height > 0;
1244 - (void)setIsVisibleToAutofill:(BOOL)isVisibleToAutofill {
1247 self.frame = isVisibleToAutofill ? CGRectMake(0, 0, 1, 1) : CGRectZero;
1250 #pragma mark UIScribbleInteractionDelegate
1255 if (@available(iOS 14.0, *)) {
1256 if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad) {
1263 - (void)scribbleInteractionWillBeginWriting:(UIScribbleInteraction*)interaction
1264 API_AVAILABLE(ios(14.0)) {
1266 [
self.textInputDelegate flutterTextInputViewScribbleInteractionBegan:self];
1269 - (void)scribbleInteractionDidFinishWriting:(UIScribbleInteraction*)interaction
1270 API_AVAILABLE(ios(14.0)) {
1272 [
self.textInputDelegate flutterTextInputViewScribbleInteractionFinished:self];
1275 - (BOOL)scribbleInteraction:(UIScribbleInteraction*)interaction
1276 shouldBeginAtLocation:(CGPoint)location API_AVAILABLE(ios(14.0)) {
1280 - (BOOL)scribbleInteractionShouldDelayFocus:(UIScribbleInteraction*)interaction
1281 API_AVAILABLE(ios(14.0)) {
1285 #pragma mark - UIResponder Overrides
1287 - (BOOL)canBecomeFirstResponder {
1292 return _textInputClient != 0;
1295 - (BOOL)resignFirstResponder {
1296 BOOL success = [
super resignFirstResponder];
1298 if (!_preventCursorDismissWhenResignFirstResponder) {
1299 [
self.textInputDelegate flutterTextInputView:self
1300 didResignFirstResponderWithTextInputClient:_textInputClient];
1306 - (BOOL)canPerformAction:(
SEL)action withSender:(
id)sender {
1307 if (action ==
@selector(paste:)) {
1309 return [UIPasteboard generalPasteboard].hasStrings;
1310 }
else if (action ==
@selector(copy:) || action ==
@selector(cut:) ||
1311 action ==
@selector(
delete:)) {
1312 return [
self textInRange:_selectedTextRange].length > 0;
1313 }
else if (action ==
@selector(selectAll:)) {
1314 return self.hasText;
1316 return [
super canPerformAction:action withSender:sender];
1319 #pragma mark - UIResponderStandardEditActions Overrides
1321 - (void)cut:(
id)sender {
1322 [UIPasteboard generalPasteboard].string = [
self textInRange:_selectedTextRange];
1323 [
self replaceRange:_selectedTextRange withText:@""];
1326 - (void)copy:(
id)sender {
1327 [UIPasteboard generalPasteboard].string = [
self textInRange:_selectedTextRange];
1330 - (void)paste:(
id)sender {
1331 NSString* pasteboardString = [UIPasteboard generalPasteboard].string;
1332 if (pasteboardString != nil) {
1333 [
self insertText:pasteboardString];
1337 - (void)delete:(
id)sender {
1338 [
self replaceRange:_selectedTextRange withText:@""];
1341 - (void)selectAll:(
id)sender {
1342 [
self setSelectedTextRange:[
self textRangeFromPosition:[
self beginningOfDocument]
1343 toPosition:[
self endOfDocument]]];
1346 #pragma mark - UITextInput Overrides
1348 - (id<UITextInputTokenizer>)tokenizer {
1349 if (_tokenizer == nil) {
1356 return [_selectedTextRange copy];
1360 - (void)setSelectedTextRangeLocal:(UITextRange*)selectedTextRange {
1365 rangeWithNSRange:fml::RangeForCharactersInRange(self.text, flutterTextRange.range)] copy];
1372 - (void)setSelectedTextRange:(UITextRange*)selectedTextRange {
1377 [
self setSelectedTextRangeLocal:selectedTextRange];
1379 if (_enableDeltaModel) {
1380 [
self updateEditingStateWithDelta:flutter::TextEditingDelta([
self.text UTF8String])];
1382 [
self updateEditingState];
1386 _scribbleFocusStatus == FlutterScribbleFocusStatusFocused) {
1390 if (flutterTextRange.
range.length > 0) {
1391 [
self.textInputDelegate flutterTextInputView:self showToolbar:_textInputClient];
1395 [
self resetScribbleInteractionStatusIfEnding];
1398 - (id)insertDictationResultPlaceholder {
1402 - (void)removeDictationResultPlaceholder:(
id)placeholder willInsertResult:(BOOL)willInsertResult {
1405 - (NSString*)textInRange:(UITextRange*)range {
1410 @"Expected a FlutterTextRange for range (got %@).", [range
class]);
1412 if (textRange.location == NSNotFound) {
1421 NSUInteger location = MIN(textRange.location,
self.text.length);
1422 NSUInteger length = MIN(
self.text.length - location, textRange.length);
1423 NSRange safeRange = NSMakeRange(location, length);
1424 return [
self.text substringWithRange:safeRange];
1429 - (void)replaceRangeLocal:(NSRange)range withText:(NSString*)text {
1430 [
self.text replaceCharactersInRange:[
self clampSelection:range forText:self.text]
1436 const NSRange newSelectionRange =
1437 [
self clampSelection:NSMakeRange(range.location + text.length, 0) forText:self.text];
1440 self.markedTextRange = nil;
1443 - (void)replaceRange:(UITextRange*)range withText:(NSString*)text {
1444 NSString* textBeforeChange = [
self.text copy];
1446 [
self replaceRangeLocal:replaceRange withText:text];
1447 if (_enableDeltaModel) {
1448 NSRange nextReplaceRange = [
self clampSelection:replaceRange forText:textBeforeChange];
1449 [
self updateEditingStateWithDelta:flutter::TextEditingDelta(
1450 [textBeforeChange UTF8String],
1452 nextReplaceRange.location,
1453 nextReplaceRange.location + nextReplaceRange.length),
1454 [text UTF8String])];
1456 [
self updateEditingState];
1460 - (BOOL)shouldChangeTextInRange:(UITextRange*)range replacementText:(NSString*)text {
1463 self.temporarilyDeletedComposedCharacter = nil;
1465 if (
self.
returnKeyType == UIReturnKeyDefault && [text isEqualToString:
@"\n"]) {
1466 [
self.textInputDelegate flutterTextInputView:self
1467 performAction:FlutterTextInputActionNewline
1468 withClient:_textInputClient];
1472 if ([text isEqualToString:
@"\n"]) {
1473 FlutterTextInputAction action;
1475 case UIReturnKeyDefault:
1476 action = FlutterTextInputActionUnspecified;
1478 case UIReturnKeyDone:
1479 action = FlutterTextInputActionDone;
1482 action = FlutterTextInputActionGo;
1484 case UIReturnKeySend:
1485 action = FlutterTextInputActionSend;
1487 case UIReturnKeySearch:
1488 case UIReturnKeyGoogle:
1489 case UIReturnKeyYahoo:
1490 action = FlutterTextInputActionSearch;
1492 case UIReturnKeyNext:
1493 action = FlutterTextInputActionNext;
1495 case UIReturnKeyContinue:
1496 action = FlutterTextInputActionContinue;
1498 case UIReturnKeyJoin:
1499 action = FlutterTextInputActionJoin;
1501 case UIReturnKeyRoute:
1502 action = FlutterTextInputActionRoute;
1504 case UIReturnKeyEmergencyCall:
1505 action = FlutterTextInputActionEmergencyCall;
1509 [
self.textInputDelegate flutterTextInputView:self
1510 performAction:action
1511 withClient:_textInputClient];
1520 - (void)setMarkedText:(NSString*)markedText selectedRange:(NSRange)markedSelectedRange {
1521 NSString* textBeforeChange = [
self.text copy];
1524 _scribbleFocusStatus != FlutterScribbleFocusStatusUnfocused) {
1528 if (markedText == nil) {
1533 const NSRange& actualReplacedRange = currentMarkedTextRange && !currentMarkedTextRange.isEmpty
1534 ? currentMarkedTextRange.
range
1538 [
self.text replaceCharactersInRange:actualReplacedRange withString:markedText];
1540 const NSRange newMarkedRange = NSMakeRange(actualReplacedRange.location, markedText.length);
1541 self.markedTextRange =
1544 [
self setSelectedTextRangeLocal:
1546 rangeWithNSRange:[
self clampSelection:NSMakeRange(markedSelectedRange.location +
1547 newMarkedRange.location,
1548 markedSelectedRange.length)
1549 forText:self.text]]];
1550 if (_enableDeltaModel) {
1551 NSRange nextReplaceRange = [
self clampSelection:actualReplacedRange forText:textBeforeChange];
1552 [
self updateEditingStateWithDelta:flutter::TextEditingDelta(
1553 [textBeforeChange UTF8String],
1555 nextReplaceRange.location,
1556 nextReplaceRange.location + nextReplaceRange.length),
1557 [markedText UTF8String])];
1559 [
self updateEditingState];
1563 - (void)unmarkText {
1567 self.markedTextRange = nil;
1568 if (_enableDeltaModel) {
1569 [
self updateEditingStateWithDelta:flutter::TextEditingDelta([
self.text UTF8String])];
1571 [
self updateEditingState];
1575 - (UITextRange*)textRangeFromPosition:(UITextPosition*)fromPosition
1576 toPosition:(UITextPosition*)toPosition {
1579 if (toIndex >= fromIndex) {
1592 - (NSUInteger)decrementOffsetPosition:(NSUInteger)position {
1593 return fml::RangeForCharacterAtIndex(
self.text, MAX(0, position - 1)).location;
1596 - (NSUInteger)incrementOffsetPosition:(NSUInteger)position {
1597 NSRange charRange = fml::RangeForCharacterAtIndex(
self.text, position);
1598 return MIN(position + charRange.length,
self.text.length);
1601 - (UITextPosition*)positionFromPosition:(UITextPosition*)position offset:(NSInteger)offset {
1604 NSInteger newLocation = (NSInteger)offsetPosition + offset;
1605 if (newLocation < 0 || newLocation > (NSInteger)
self.text.length) {
1614 for (NSInteger i = 0; i < offset && offsetPosition <
self.text.length; ++i) {
1615 offsetPosition = [
self incrementOffsetPosition:offsetPosition];
1618 for (NSInteger i = 0; i < ABS(offset) && offsetPosition > 0; ++i) {
1619 offsetPosition = [
self decrementOffsetPosition:offsetPosition];
1625 - (UITextPosition*)positionFromPosition:(UITextPosition*)position
1626 inDirection:(UITextLayoutDirection)direction
1627 offset:(NSInteger)offset {
1629 switch (direction) {
1630 case UITextLayoutDirectionLeft:
1631 case UITextLayoutDirectionUp:
1632 return [
self positionFromPosition:position offset:offset * -1];
1633 case UITextLayoutDirectionRight:
1634 case UITextLayoutDirectionDown:
1635 return [
self positionFromPosition:position offset:1];
1639 - (UITextPosition*)beginningOfDocument {
1643 - (UITextPosition*)endOfDocument {
1645 affinity:UITextStorageDirectionBackward];
1648 - (NSComparisonResult)comparePosition:(UITextPosition*)position toPosition:(UITextPosition*)other {
1651 if (positionIndex < otherIndex) {
1652 return NSOrderedAscending;
1654 if (positionIndex > otherIndex) {
1655 return NSOrderedDescending;
1659 if (positionAffinity == otherAffinity) {
1660 return NSOrderedSame;
1662 if (positionAffinity == UITextStorageDirectionBackward) {
1664 return NSOrderedAscending;
1667 return NSOrderedDescending;
1670 - (NSInteger)offsetFromPosition:(UITextPosition*)from toPosition:(UITextPosition*)toPosition {
1674 - (UITextPosition*)positionWithinRange:(UITextRange*)range
1675 farthestInDirection:(UITextLayoutDirection)direction {
1677 UITextStorageDirection affinity;
1678 switch (direction) {
1679 case UITextLayoutDirectionLeft:
1680 case UITextLayoutDirectionUp:
1682 affinity = UITextStorageDirectionForward;
1684 case UITextLayoutDirectionRight:
1685 case UITextLayoutDirectionDown:
1687 affinity = UITextStorageDirectionBackward;
1693 - (UITextRange*)characterRangeByExtendingPosition:(UITextPosition*)position
1694 inDirection:(UITextLayoutDirection)direction {
1696 NSUInteger startIndex;
1697 NSUInteger endIndex;
1698 switch (direction) {
1699 case UITextLayoutDirectionLeft:
1700 case UITextLayoutDirectionUp:
1701 startIndex = [
self decrementOffsetPosition:positionIndex];
1702 endIndex = positionIndex;
1704 case UITextLayoutDirectionRight:
1705 case UITextLayoutDirectionDown:
1706 startIndex = positionIndex;
1707 endIndex = [
self incrementOffsetPosition:positionIndex];
1713 #pragma mark - UITextInput text direction handling
1715 - (UITextWritingDirection)baseWritingDirectionForPosition:(UITextPosition*)position
1716 inDirection:(UITextStorageDirection)direction {
1718 return UITextWritingDirectionNatural;
1721 - (void)setBaseWritingDirection:(UITextWritingDirection)writingDirection
1722 forRange:(UITextRange*)range {
1726 #pragma mark - UITextInput cursor, selection rect handling
1728 - (void)setMarkedRect:(CGRect)markedRect {
1729 _markedRect = markedRect;
1736 - (void)setEditableTransform:(NSArray*)matrix {
1737 CATransform3D* transform = &_editableTransform;
1739 transform->m11 = [matrix[0] doubleValue];
1740 transform->m12 = [matrix[1] doubleValue];
1741 transform->m13 = [matrix[2] doubleValue];
1742 transform->m14 = [matrix[3] doubleValue];
1744 transform->m21 = [matrix[4] doubleValue];
1745 transform->m22 = [matrix[5] doubleValue];
1746 transform->m23 = [matrix[6] doubleValue];
1747 transform->m24 = [matrix[7] doubleValue];
1749 transform->m31 = [matrix[8] doubleValue];
1750 transform->m32 = [matrix[9] doubleValue];
1751 transform->m33 = [matrix[10] doubleValue];
1752 transform->m34 = [matrix[11] doubleValue];
1754 transform->m41 = [matrix[12] doubleValue];
1755 transform->m42 = [matrix[13] doubleValue];
1756 transform->m43 = [matrix[14] doubleValue];
1757 transform->m44 = [matrix[15] doubleValue];
1766 CGPoint points[] = {
1767 incomingRect.origin,
1768 CGPointMake(incomingRect.origin.x, incomingRect.origin.y + incomingRect.size.height),
1769 CGPointMake(incomingRect.origin.x + incomingRect.size.width, incomingRect.origin.y),
1770 CGPointMake(incomingRect.origin.x + incomingRect.size.width,
1771 incomingRect.origin.y + incomingRect.size.height)};
1773 CGPoint origin = CGPointMake(CGFLOAT_MAX, CGFLOAT_MAX);
1774 CGPoint farthest = CGPointMake(-CGFLOAT_MAX, -CGFLOAT_MAX);
1776 for (
int i = 0; i < 4; i++) {
1777 const CGPoint point = points[i];
1779 CGFloat x = _editableTransform.m11 * point.x + _editableTransform.m21 * point.y +
1780 _editableTransform.m41;
1781 CGFloat y = _editableTransform.m12 * point.x + _editableTransform.m22 * point.y +
1782 _editableTransform.m42;
1784 const CGFloat w = _editableTransform.m14 * point.x + _editableTransform.m24 * point.y +
1785 _editableTransform.m44;
1789 }
else if (w != 1.0) {
1794 origin.x = MIN(origin.x, x);
1795 origin.y = MIN(origin.y, y);
1796 farthest.x = MAX(farthest.x, x);
1797 farthest.y = MAX(farthest.y, y);
1799 return CGRectMake(origin.x, origin.y, farthest.x - origin.x, farthest.y - origin.y);
1808 - (CGRect)firstRectForRange:(UITextRange*)range {
1810 @"Expected a FlutterTextPosition for range.start (got %@).", [range.start
class]);
1812 @"Expected a FlutterTextPosition for range.end (got %@).", [range.end
class]);
1815 if (_markedTextRange != nil) {
1826 CGRect rect = _markedRect;
1827 if (CGRectIsEmpty(rect)) {
1828 rect = CGRectInset(rect, -0.1, 0);
1833 UIView* hostView = _textInputPlugin.hostView;
1834 NSAssert(hostView == nil || [
self isDescendantOfView:hostView],
@"%@ is not a descendant of %@",
1836 return hostView ? [hostView convertRect:_cachedFirstRect toView:self] :
_cachedFirstRect;
1840 _scribbleFocusStatus == FlutterScribbleFocusStatusUnfocused) {
1841 if (@available(iOS 17.0, *)) {
1851 [
self.textInputDelegate flutterTextInputView:self
1852 showAutocorrectionPromptRectForStart:start
1854 withClient:_textInputClient];
1862 if (@available(iOS 17, *)) {
1868 NSUInteger first = start;
1873 CGRect startSelectionRect = CGRectNull;
1874 CGRect endSelectionRect = CGRectNull;
1877 CGFloat minY = CGFLOAT_MAX;
1878 CGFloat maxY = CGFLOAT_MIN;
1881 rangeWithNSRange:fml::RangeForCharactersInRange(self.text, NSMakeRange(0, self.text.length))];
1882 for (NSUInteger i = 0; i < [_selectionRects count]; i++) {
1883 BOOL startsOnOrBeforeStartOfRange = _selectionRects[i].position <= first;
1884 BOOL isLastSelectionRect = i + 1 == [_selectionRects count];
1885 BOOL endOfTextIsAfterStartOfRange = isLastSelectionRect && textRange.
range.length > first;
1886 BOOL nextSelectionRectIsAfterStartOfRange =
1887 !isLastSelectionRect && _selectionRects[i + 1].position > first;
1888 if (startsOnOrBeforeStartOfRange &&
1889 (endOfTextIsAfterStartOfRange || nextSelectionRectIsAfterStartOfRange)) {
1891 if (@available(iOS 17, *)) {
1892 startSelectionRect = _selectionRects[i].rect;
1894 return _selectionRects[i].rect;
1897 if (!CGRectIsNull(startSelectionRect)) {
1898 minY = fmin(minY, CGRectGetMinY(_selectionRects[i].rect));
1899 maxY = fmax(maxY, CGRectGetMaxY(_selectionRects[i].rect));
1900 BOOL endsOnOrAfterEndOfRange = _selectionRects[i].position >= end - 1;
1901 BOOL nextSelectionRectIsOnNextLine =
1902 !isLastSelectionRect &&
1907 CGRectGetMidY(_selectionRects[i + 1].rect) > CGRectGetMaxY(_selectionRects[i].rect);
1908 if (endsOnOrAfterEndOfRange || isLastSelectionRect || nextSelectionRectIsOnNextLine) {
1909 endSelectionRect = _selectionRects[i].rect;
1914 if (CGRectIsNull(startSelectionRect) || CGRectIsNull(endSelectionRect)) {
1918 CGFloat minX = fmin(CGRectGetMinX(startSelectionRect), CGRectGetMinX(endSelectionRect));
1919 CGFloat maxX = fmax(CGRectGetMaxX(startSelectionRect), CGRectGetMaxX(endSelectionRect));
1920 return CGRectMake(minX, minY, maxX - minX, maxY - minY);
1928 NSArray<UITextSelectionRect*>* rects = [
self
1930 rangeWithNSRange:fml::RangeForCharactersInRange(
1934 (index >= (NSInteger)self.text.length)
1937 if (rects.count == 0) {
1943 CGRect characterAfterCaret = rects[0].rect;
1948 return CGRectMake(characterAfterCaret.origin.x + characterAfterCaret.size.width,
1949 characterAfterCaret.origin.y, 0, characterAfterCaret.size.height);
1951 return CGRectMake(characterAfterCaret.origin.x, characterAfterCaret.origin.y, 0,
1952 characterAfterCaret.size.height);
1954 }
else if (rects.count == 2 && affinity == UITextStorageDirectionForward) {
1957 CGRect characterAfterCaret = rects[1].rect;
1962 return CGRectMake(characterAfterCaret.origin.x + characterAfterCaret.size.width,
1963 characterAfterCaret.origin.y, 0, characterAfterCaret.size.height);
1965 return CGRectMake(characterAfterCaret.origin.x, characterAfterCaret.origin.y, 0,
1966 characterAfterCaret.size.height);
1975 CGRect characterBeforeCaret = rects[0].rect;
1978 return CGRectMake(characterBeforeCaret.origin.x, characterBeforeCaret.origin.y, 0,
1979 characterBeforeCaret.size.height);
1981 return CGRectMake(characterBeforeCaret.origin.x + characterBeforeCaret.size.width,
1982 characterBeforeCaret.origin.y, 0, characterBeforeCaret.size.height);
1986 - (UITextPosition*)closestPositionToPoint:(CGPoint)point {
1987 if ([_selectionRects count] == 0) {
1989 @"Expected a FlutterTextPosition for position (got %@).",
1992 UITextStorageDirection currentAffinity =
1998 rangeWithNSRange:fml::RangeForCharactersInRange(self.text, NSMakeRange(0, self.text.length))];
1999 return [
self closestPositionToPoint:point withinRange:range];
2002 - (NSArray*)selectionRectsForRange:(UITextRange*)range {
2010 @"Expected a FlutterTextPosition for range.start (got %@).", [range.start
class]);
2012 @"Expected a FlutterTextPosition for range.end (got %@).", [range.end
class]);
2015 NSMutableArray* rects = [[NSMutableArray alloc] init];
2016 for (NSUInteger i = 0; i < [_selectionRects count]; i++) {
2017 if (_selectionRects[i].position >= start &&
2018 (_selectionRects[i].position < end ||
2019 (start == end && _selectionRects[i].position <= end))) {
2020 float width = _selectionRects[i].rect.size.width;
2024 CGRect rect = CGRectMake(_selectionRects[i].rect.origin.x, _selectionRects[i].rect.origin.y,
2025 width, _selectionRects[i].rect.size.height);
2028 position:_selectionRects[i].position
2032 self.text, NSMakeRange(0, self.text.length))
2035 [rects addObject:selectionRect];
2041 - (UITextPosition*)closestPositionToPoint:(CGPoint)point withinRange:(UITextRange*)range {
2043 @"Expected a FlutterTextPosition for range.start (got %@).", [range.start
class]);
2045 @"Expected a FlutterTextPosition for range.end (got %@).", [range.end
class]);
2055 NSUInteger _closestRectIndex = 0;
2056 for (NSUInteger i = 0; i < [_selectionRects count]; i++) {
2057 NSUInteger position = _selectionRects[i].position;
2058 if (position >= start && position <= end) {
2061 point, _selectionRects[i].rect, _selectionRects[i].isRTL,
2062 NO, _selectionRects[_closestRectIndex].rect,
2063 _selectionRects[_closestRectIndex].isRTL, verticalPrecision)) {
2065 _closestRectIndex = i;
2072 affinity:UITextStorageDirectionForward];
2078 for (NSUInteger i = MAX(0, _closestRectIndex - 1);
2079 i < MIN(_closestRectIndex + 2, [_selectionRects count]); i++) {
2080 NSUInteger position = _selectionRects[i].position + 1;
2081 if (position >= start && position <= end) {
2083 point, _selectionRects[i].rect, _selectionRects[i].isRTL,
2084 YES, _selectionRects[_closestRectIndex].rect,
2085 _selectionRects[_closestRectIndex].isRTL, verticalPrecision)) {
2088 affinity:UITextStorageDirectionBackward];
2093 return closestPosition;
2096 - (UITextRange*)characterRangeAtPoint:(CGPoint)point {
2099 return [
FlutterTextRange rangeWithNSRange:fml::RangeForCharacterAtIndex(self.text, currentIndex)];
2130 - (void)beginFloatingCursorAtPoint:(CGPoint)point {
2147 [
self.textInputDelegate flutterTextInputView:self
2148 updateFloatingCursor:FlutterFloatingCursorDragStateStart
2149 withClient:_textInputClient
2150 withPosition:@{@"X" : @0, @"Y" : @0}];
2153 - (void)updateFloatingCursorAtPoint:(CGPoint)point {
2154 [
self.textInputDelegate flutterTextInputView:self
2155 updateFloatingCursor:FlutterFloatingCursorDragStateUpdate
2156 withClient:_textInputClient
2158 @"X" : @(point.x - _floatingCursorOffset.x),
2159 @"Y" : @(point.y - _floatingCursorOffset.y)
2163 - (void)endFloatingCursor {
2165 [
self.textInputDelegate flutterTextInputView:self
2166 updateFloatingCursor:FlutterFloatingCursorDragStateEnd
2167 withClient:_textInputClient
2168 withPosition:@{@"X" : @0, @"Y" : @0}];
2171 #pragma mark - UIKeyInput Overrides
2173 - (void)updateEditingState {
2178 NSInteger composingBase = -1;
2179 NSInteger composingExtent = -1;
2184 NSDictionary* state = @{
2185 @"selectionBase" : @(selectionBase),
2186 @"selectionExtent" : @(selectionExtent),
2188 @"selectionIsDirectional" : @(
false),
2189 @"composingBase" : @(composingBase),
2190 @"composingExtent" : @(composingExtent),
2191 @"text" : [NSString stringWithString:self.text],
2194 if (_textInputClient == 0 && _autofillId != nil) {
2195 [
self.textInputDelegate flutterTextInputView:self
2196 updateEditingClient:_textInputClient
2198 withTag:_autofillId];
2200 [
self.textInputDelegate flutterTextInputView:self
2201 updateEditingClient:_textInputClient
2206 - (void)updateEditingStateWithDelta:(
flutter::TextEditingDelta)delta {
2211 NSInteger composingBase = -1;
2212 NSInteger composingExtent = -1;
2218 NSDictionary* deltaToFramework = @{
2219 @"oldText" : @(delta.old_text().c_str()),
2220 @"deltaText" : @(delta.delta_text().c_str()),
2221 @"deltaStart" : @(delta.delta_start()),
2222 @"deltaEnd" : @(delta.delta_end()),
2223 @"selectionBase" : @(selectionBase),
2224 @"selectionExtent" : @(selectionExtent),
2226 @"selectionIsDirectional" : @(
false),
2227 @"composingBase" : @(composingBase),
2228 @"composingExtent" : @(composingExtent),
2231 [_pendingDeltas addObject:deltaToFramework];
2233 if (_pendingDeltas.count == 1) {
2235 dispatch_async(dispatch_get_main_queue(), ^{
2237 if (strongSelf && strongSelf.pendingDeltas.count > 0) {
2238 NSDictionary* deltas = @{
2239 @"deltas" : strongSelf.pendingDeltas,
2242 [strongSelf.textInputDelegate flutterTextInputView:strongSelf
2243 updateEditingClient:strongSelf->_textInputClient
2245 [strongSelf.pendingDeltas removeAllObjects];
2252 return self.text.length > 0;
2255 - (void)insertText:(NSString*)text {
2256 if (
self.temporarilyDeletedComposedCharacter.length > 0 && text.length == 1 && !text.UTF8String &&
2257 [text characterAtIndex:0] == [
self.temporarilyDeletedComposedCharacter characterAtIndex:0]) {
2261 text =
self.temporarilyDeletedComposedCharacter;
2262 self.temporarilyDeletedComposedCharacter = nil;
2265 NSMutableArray<FlutterTextSelectionRect*>* copiedRects =
2266 [[NSMutableArray alloc] initWithCapacity:[_selectionRects count]];
2268 @"Expected a FlutterTextPosition for position (got %@).",
2271 for (NSUInteger i = 0; i < [_selectionRects count]; i++) {
2272 NSUInteger rectPosition = _selectionRects[i].position;
2273 if (rectPosition == insertPosition) {
2274 for (NSUInteger j = 0; j <= text.length; j++) {
2281 if (rectPosition > insertPosition) {
2282 rectPosition = rectPosition + text.length;
2291 _scribbleFocusStatus = FlutterScribbleFocusStatusUnfocused;
2292 [
self resetScribbleInteractionStatusIfEnding];
2293 self.selectionRects = copiedRects;
2295 [
self replaceRange:_selectedTextRange withText:text];
2298 - (UITextPlaceholder*)insertTextPlaceholderWithSize:(CGSize)size API_AVAILABLE(ios(13.0)) {
2299 [
self.textInputDelegate flutterTextInputView:self
2300 insertTextPlaceholderWithSize:size
2301 withClient:_textInputClient];
2306 - (void)removeTextPlaceholder:(UITextPlaceholder*)textPlaceholder API_AVAILABLE(ios(13.0)) {
2308 [
self.textInputDelegate flutterTextInputView:self removeTextPlaceholder:_textInputClient];
2311 - (void)deleteBackward {
2313 _scribbleFocusStatus = FlutterScribbleFocusStatusUnfocused;
2314 [
self resetScribbleInteractionStatusIfEnding];
2331 if (oldRange.location > 0) {
2332 NSRange newRange = NSMakeRange(oldRange.location - 1, 1);
2336 NSRange charRange = fml::RangeForCharacterAtIndex(
self.text, oldRange.location - 1);
2337 if (
IsEmoji(
self.text, charRange)) {
2338 newRange = NSMakeRange(charRange.location, oldRange.location - charRange.location);
2350 NSString* deletedText = [
self.text substringWithRange:_selectedTextRange.range];
2351 NSRange deleteFirstCharacterRange = fml::RangeForCharacterAtIndex(deletedText, 0);
2352 self.temporarilyDeletedComposedCharacter =
2353 [deletedText substringWithRange:deleteFirstCharacterRange];
2355 [
self replaceRange:_selectedTextRange withText:@""];
2359 - (void)postAccessibilityNotification:(UIAccessibilityNotifications)notification target:(
id)target {
2360 UIAccessibilityPostNotification(notification, target);
2363 - (void)accessibilityElementDidBecomeFocused {
2364 if ([
self accessibilityElementIsFocused]) {
2368 FML_DCHECK(_backingTextInputAccessibilityObject);
2369 [
self postAccessibilityNotification:UIAccessibilityScreenChangedNotification
2370 target:_backingTextInputAccessibilityObject];
2374 - (BOOL)accessibilityElementsHidden {
2375 return !_accessibilityEnabled;
2384 #pragma mark - Key Events Handling
2385 - (void)pressesBegan:(NSSet<UIPress*>*)presses
2386 withEvent:(UIPressesEvent*)event API_AVAILABLE(ios(9.0)) {
2387 [_textInputPlugin.viewController pressesBegan:presses withEvent:event];
2390 - (void)pressesChanged:(NSSet<UIPress*>*)presses
2391 withEvent:(UIPressesEvent*)event API_AVAILABLE(ios(9.0)) {
2392 [_textInputPlugin.viewController pressesChanged:presses withEvent:event];
2395 - (void)pressesEnded:(NSSet<UIPress*>*)presses
2396 withEvent:(UIPressesEvent*)event API_AVAILABLE(ios(9.0)) {
2397 [_textInputPlugin.viewController pressesEnded:presses withEvent:event];
2400 - (void)pressesCancelled:(NSSet<UIPress*>*)presses
2401 withEvent:(UIPressesEvent*)event API_AVAILABLE(ios(9.0)) {
2402 [_textInputPlugin.viewController pressesCancelled:presses withEvent:event];
2431 - (BOOL)accessibilityElementsHidden {
2438 - (void)enableActiveViewAccessibility;
2455 - (void)enableActiveViewAccessibility {
2456 [
self.target enableActiveViewAccessibility];
2463 @property(nonatomic, readonly)
2464 NSMutableDictionary<NSString*, FlutterTextInputView*>* autofillContext;
2469 @property(nonatomic, strong) UIView* keyboardViewContainer;
2470 @property(nonatomic, strong) UIView* keyboardView;
2471 @property(nonatomic, strong) UIView* cachedFirstResponder;
2472 @property(nonatomic, assign) CGRect keyboardRect;
2473 @property(nonatomic, assign) CGFloat previousPointerYPosition;
2474 @property(nonatomic, assign) CGFloat pointerYVelocity;
2478 NSTimer* _enableFlutterTextInputViewAccessibilityTimer;
2482 self = [
super init];
2485 _textInputDelegate = textInputDelegate;
2486 _autofillContext = [[NSMutableDictionary alloc] init];
2488 _scribbleElements = [[NSMutableDictionary alloc] init];
2489 _keyboardViewContainer = [[UIView alloc] init];
2491 [[NSNotificationCenter defaultCenter] addObserver:self
2492 selector:@selector(handleKeyboardWillShow:)
2493 name:UIKeyboardWillShowNotification
2500 - (void)handleKeyboardWillShow:(NSNotification*)notification {
2501 NSDictionary* keyboardInfo = [notification userInfo];
2502 NSValue* keyboardFrameEnd = [keyboardInfo valueForKey:UIKeyboardFrameEndUserInfoKey];
2503 _keyboardRect = [keyboardFrameEnd CGRectValue];
2507 [
self hideTextInput];
2510 - (void)removeEnableFlutterTextInputViewAccessibilityTimer {
2511 if (_enableFlutterTextInputViewAccessibilityTimer) {
2512 [_enableFlutterTextInputViewAccessibilityTimer invalidate];
2513 _enableFlutterTextInputViewAccessibilityTimer = nil;
2517 - (UIView<UITextInput>*)textInputView {
2522 NSString* method = call.
method;
2525 [
self showTextInput];
2527 }
else if ([method isEqualToString:
kHideMethod]) {
2528 [
self hideTextInput];
2531 [
self setTextInputClient:[args[0] intValue] withConfiguration:args[1]];
2535 [
self setPlatformViewTextInputClient];
2538 [
self setTextInputEditingState:args];
2541 [
self clearTextInputClient];
2544 [
self setEditableSizeAndTransform:args];
2547 [
self updateMarkedRect:args];
2550 [
self triggerAutofillSave:[args boolValue]];
2556 [
self setSelectionRects:args];
2559 [
self setSelectionRects:args];
2562 [
self startLiveTextInput];
2565 [
self updateConfig:args];
2568 CGFloat pointerY = (CGFloat)[args[
@"pointerY"] doubleValue];
2569 [
self handlePointerMove:pointerY];
2572 CGFloat pointerY = (CGFloat)[args[
@"pointerY"] doubleValue];
2573 [
self handlePointerUp:pointerY];
2580 - (void)handlePointerUp:(CGFloat)pointerY {
2581 if (_keyboardView.superview != nil) {
2584 UIScreen* screen = _viewController.flutterScreenIfViewLoaded;
2585 CGFloat screenHeight = screen.bounds.size.height;
2586 CGFloat keyboardHeight = _keyboardRect.size.height;
2588 BOOL shouldDismissKeyboardBasedOnVelocity = _pointerYVelocity < 0;
2589 [UIView animateWithDuration:kKeyboardAnimationTimeToCompleteion
2591 double keyboardDestination =
2592 shouldDismissKeyboardBasedOnVelocity ? screenHeight : screenHeight - keyboardHeight;
2593 _keyboardViewContainer.frame = CGRectMake(
2594 0, keyboardDestination, _viewController.flutterScreenIfViewLoaded.bounds.size.width,
2595 _keyboardViewContainer.frame.size.height);
2597 completion:^(BOOL finished) {
2598 if (shouldDismissKeyboardBasedOnVelocity) {
2599 [
self.textInputDelegate flutterTextInputView:self.activeView
2600 didResignFirstResponderWithTextInputClient:self.activeView.textInputClient];
2601 [
self dismissKeyboardScreenshot];
2603 [
self showKeyboardAndRemoveScreenshot];
2609 - (void)dismissKeyboardScreenshot {
2610 for (UIView* subView in _keyboardViewContainer.subviews) {
2611 [subView removeFromSuperview];
2615 - (void)showKeyboardAndRemoveScreenshot {
2616 [UIView setAnimationsEnabled:NO];
2617 [_cachedFirstResponder becomeFirstResponder];
2621 dispatch_get_main_queue(), ^{
2622 [UIView setAnimationsEnabled:YES];
2623 [
self dismissKeyboardScreenshot];
2627 - (void)handlePointerMove:(CGFloat)pointerY {
2629 UIScreen* screen = _viewController.flutterScreenIfViewLoaded;
2630 CGFloat screenHeight = screen.bounds.size.height;
2631 CGFloat keyboardHeight = _keyboardRect.size.height;
2632 if (screenHeight - keyboardHeight <= pointerY) {
2634 if (_keyboardView.superview == nil) {
2636 [
self takeKeyboardScreenshotAndDisplay];
2637 [
self hideKeyboardWithoutAnimationAndAvoidCursorDismissUpdate];
2639 [
self setKeyboardContainerHeight:pointerY];
2640 _pointerYVelocity = _previousPointerYPosition - pointerY;
2643 if (_keyboardView.superview != nil) {
2645 _keyboardViewContainer.frame = _keyboardRect;
2646 _pointerYVelocity = _previousPointerYPosition - pointerY;
2649 _previousPointerYPosition = pointerY;
2652 - (void)setKeyboardContainerHeight:(CGFloat)pointerY {
2653 CGRect frameRect = _keyboardRect;
2654 frameRect.origin.y = pointerY;
2655 _keyboardViewContainer.frame = frameRect;
2658 - (void)hideKeyboardWithoutAnimationAndAvoidCursorDismissUpdate {
2659 [UIView setAnimationsEnabled:NO];
2660 _cachedFirstResponder = UIApplication.sharedApplication.keyWindow.flutterFirstResponder;
2661 _activeView.preventCursorDismissWhenResignFirstResponder = YES;
2662 [_cachedFirstResponder resignFirstResponder];
2663 _activeView.preventCursorDismissWhenResignFirstResponder = NO;
2664 [UIView setAnimationsEnabled:YES];
2667 - (void)takeKeyboardScreenshotAndDisplay {
2669 UIScreen* screen = _viewController.flutterScreenIfViewLoaded;
2670 UIView* keyboardSnap = [screen snapshotViewAfterScreenUpdates:YES];
2671 keyboardSnap = [keyboardSnap resizableSnapshotViewFromRect:_keyboardRect
2672 afterScreenUpdates:YES
2673 withCapInsets:UIEdgeInsetsZero];
2674 _keyboardView = keyboardSnap;
2675 [_keyboardViewContainer addSubview:_keyboardView];
2676 if (_keyboardViewContainer.superview == nil) {
2677 [UIApplication.sharedApplication.delegate.window.rootViewController.view
2678 addSubview:_keyboardViewContainer];
2680 _keyboardViewContainer.layer.zPosition = NSIntegerMax;
2681 _keyboardViewContainer.frame = _keyboardRect;
2684 - (BOOL)showEditMenu:(NSDictionary*)args API_AVAILABLE(ios(16.0)) {
2685 if (!
self.activeView.isFirstResponder) {
2688 NSDictionary<NSString*, NSNumber*>* encodedTargetRect = args[@"targetRect"];
2689 CGRect globalTargetRect = CGRectMake(
2690 [encodedTargetRect[
@"x"] doubleValue], [encodedTargetRect[
@"y"] doubleValue],
2691 [encodedTargetRect[
@"width"] doubleValue], [encodedTargetRect[
@"height"] doubleValue]);
2692 CGRect localTargetRect = [
self.hostView convertRect:globalTargetRect toView:self.activeView];
2693 [
self.activeView showEditMenuWithTargetRect:localTargetRect items:args[@"items"]];
2697 - (void)hideEditMenu {
2698 [
self.activeView hideEditMenu];
2701 - (void)setEditableSizeAndTransform:(NSDictionary*)dictionary {
2702 NSArray* transform = dictionary[@"transform"];
2703 [_activeView setEditableTransform:transform];
2704 const int leftIndex = 12;
2705 const int topIndex = 13;
2709 CGRectMake([transform[leftIndex] intValue], [transform[topIndex] intValue],
2710 [dictionary[
@"width"] intValue], [dictionary[
@"height"] intValue]);
2712 CGRectMake(0, 0, [dictionary[
@"width"] intValue], [dictionary[
@"height"] intValue]);
2713 _activeView.tintColor = [UIColor clearColor];
2718 if (@available(iOS 17, *)) {
2726 CGRectMake([transform[leftIndex] intValue], [transform[topIndex] intValue], 0, 0);
2731 - (void)updateMarkedRect:(NSDictionary*)dictionary {
2732 NSAssert(dictionary[
@"x"] != nil && dictionary[
@"y"] != nil && dictionary[
@"width"] != nil &&
2733 dictionary[
@"height"] != nil,
2734 @"Expected a dictionary representing a CGRect, got %@", dictionary);
2735 CGRect rect = CGRectMake([dictionary[
@"x"] doubleValue], [dictionary[
@"y"] doubleValue],
2736 [dictionary[
@"width"] doubleValue], [dictionary[
@"height"] doubleValue]);
2737 _activeView.markedRect = rect.size.width < 0 && rect.size.height < 0 ?
kInvalidFirstRect : rect;
2740 - (void)setSelectionRects:(NSArray*)encodedRects {
2741 NSMutableArray<FlutterTextSelectionRect*>* rectsAsRect =
2742 [[NSMutableArray alloc] initWithCapacity:[encodedRects count]];
2743 for (NSUInteger i = 0; i < [encodedRects count]; i++) {
2744 NSArray<NSNumber*>* encodedRect = encodedRects[i];
2746 selectionRectWithRect:CGRectMake([encodedRect[0] floatValue],
2747 [encodedRect[1] floatValue],
2748 [encodedRect[2] floatValue],
2749 [encodedRect[3] floatValue])
2750 position:[encodedRect[4] unsignedIntegerValue]
2751 writingDirection:[encodedRect[5] unsignedIntegerValue] == 1
2752 ? NSWritingDirectionLeftToRight
2753 : NSWritingDirectionRightToLeft]];
2759 _activeView.selectionRects = rectsAsRect;
2762 - (void)startLiveTextInput {
2763 if (@available(iOS 15.0, *)) {
2764 if (_activeView == nil || !_activeView.isFirstResponder) {
2767 [_activeView captureTextFromCamera:nil];
2771 - (void)showTextInput {
2772 _activeView.viewResponder = _viewResponder;
2773 [
self addToInputParentViewIfNeeded:_activeView];
2782 if (!_enableFlutterTextInputViewAccessibilityTimer) {
2783 _enableFlutterTextInputViewAccessibilityTimer =
2784 [NSTimer scheduledTimerWithTimeInterval:kUITextInputAccessibilityEnablingDelaySeconds
2786 selector:@selector(enableActiveViewAccessibility)
2790 [_activeView becomeFirstResponder];
2793 - (void)enableActiveViewAccessibility {
2794 if (_activeView.isFirstResponder) {
2795 _activeView.accessibilityEnabled = YES;
2797 [
self removeEnableFlutterTextInputViewAccessibilityTimer];
2800 - (void)hideTextInput {
2801 [
self removeEnableFlutterTextInputViewAccessibilityTimer];
2802 _activeView.accessibilityEnabled = NO;
2803 [_activeView resignFirstResponder];
2808 [
self cleanUpViewHierarchy:YES clearText:YES delayRemoval:NO];
2811 - (void)triggerAutofillSave:(BOOL)saveEntries {
2812 [_activeView resignFirstResponder];
2817 [
self cleanUpViewHierarchy:YES clearText:YES delayRemoval:NO];
2818 [_autofillContext removeAllObjects];
2819 [
self changeInputViewsAutofillVisibility:YES];
2821 [_autofillContext removeAllObjects];
2824 [
self cleanUpViewHierarchy:YES clearText:!saveEntries delayRemoval:NO];
2825 [
self addToInputParentViewIfNeeded:_activeView];
2828 - (void)setPlatformViewTextInputClient {
2832 [
self removeEnableFlutterTextInputViewAccessibilityTimer];
2833 _activeView.accessibilityEnabled = NO;
2834 [_activeView removeFromSuperview];
2835 [_inputHider removeFromSuperview];
2838 - (void)setTextInputClient:(
int)client withConfiguration:(NSDictionary*)configuration {
2839 [
self resetAllClientIds];
2842 [
self changeInputViewsAutofillVisibility:NO];
2846 case kFlutterAutofillTypeNone:
2847 self.activeView = [
self createInputViewWith:configuration];
2849 case kFlutterAutofillTypeRegular:
2852 self.activeView = [
self updateAndShowAutofillViews:nil
2853 focusedField:configuration
2854 isPasswordRelated:NO];
2856 case kFlutterAutofillTypePassword:
2857 self.activeView = [
self updateAndShowAutofillViews:configuration[kAssociatedAutofillFields]
2858 focusedField:configuration
2859 isPasswordRelated:YES];
2862 [_activeView setTextInputClient:client];
2863 [_activeView reloadInputViews];
2875 [
self cleanUpViewHierarchy:NO clearText:YES delayRemoval:YES];
2886 [_autofillContext removeObjectForKey:autofillId];
2889 [newView configureWithDictionary:configuration];
2890 [
self addToInputParentViewIfNeeded:newView];
2894 if (autofillId &&
AutofillTypeOf(field) == kFlutterAutofillTypeNone) {
2895 [_autofillContext removeObjectForKey:autofillId];
2902 focusedField:(NSDictionary*)focusedField
2903 isPasswordRelated:(BOOL)isPassword {
2906 NSAssert(focusedId,
@"autofillId must not be null for the focused field: %@", focusedField);
2911 focused = [
self getOrCreateAutofillableView:focusedField isPasswordAutofill:isPassword];
2912 [_autofillContext removeObjectForKey:focusedId];
2915 for (NSDictionary* field in fields) {
2917 NSAssert(autofillId,
@"autofillId must not be null for field: %@", field);
2919 BOOL hasHints =
AutofillTypeOf(field) != kFlutterAutofillTypeNone;
2920 BOOL isFocused = [focusedId isEqualToString:autofillId];
2923 focused = [
self getOrCreateAutofillableView:field isPasswordAutofill:isPassword];
2928 _autofillContext[autofillId] = isFocused ? focused
2929 : [
self getOrCreateAutofillableView:field
2930 isPasswordAutofill:isPassword];
2933 [_autofillContext removeObjectForKey:autofillId];
2937 NSAssert(focused,
@"The current focused input view must not be nil.");
2947 isPasswordAutofill:(BOOL)needsPasswordAutofill {
2953 inputView = [inputView initWithOwner:self];
2954 [
self addToInputParentViewIfNeeded:inputView];
2957 [inputView configureWithDictionary:field];
2962 - (UIView*)hostView {
2963 UIView* host = _viewController.view;
2964 NSAssert(host !=
nullptr,
2965 @"The application must have a host view since the keyboard client "
2966 @"must be part of the responder chain to function. The host view controller is %@",
2972 - (NSArray<UIView*>*)textInputViews {
2973 return _inputHider.subviews;
2986 - (void)cleanUpViewHierarchy:(BOOL)includeActiveView
2987 clearText:(BOOL)clearText
2988 delayRemoval:(BOOL)delayRemoval {
2989 for (UIView* view in
self.textInputViews) {
2991 (includeActiveView || view != _activeView)) {
2993 if (_autofillContext[inputView.autofillId] != view) {
2995 [inputView replaceRangeLocal:NSMakeRange(0, inputView.text.length) withText:@""];
2998 [inputView performSelector:@selector(removeFromSuperview) withObject:nil afterDelay:0.1];
3000 [inputView removeFromSuperview];
3009 - (void)changeInputViewsAutofillVisibility:(BOOL)newVisibility {
3010 for (UIView* view in
self.textInputViews) {
3013 inputView.isVisibleToAutofill = newVisibility;
3025 - (void)resetAllClientIds {
3026 for (UIView* view in
self.textInputViews) {
3029 [inputView setTextInputClient:0];
3035 if (![inputView isDescendantOfView:_inputHider]) {
3036 [_inputHider addSubview:inputView];
3039 if (_viewController.view == nil) {
3045 UIView* parentView =
self.hostView;
3046 if (_inputHider.superview != parentView) {
3047 [parentView addSubview:_inputHider];
3051 - (void)setTextInputEditingState:(NSDictionary*)state {
3052 [_activeView setTextInputState:state];
3055 - (void)clearTextInputClient {
3056 [_activeView setTextInputClient:0];
3057 _activeView.frame = CGRectZero;
3060 - (void)updateConfig:(NSDictionary*)dictionary {
3061 BOOL isSecureTextEntry = [dictionary[kSecureTextEntry] boolValue];
3062 for (UIView* view in
self.textInputViews) {
3069 if (inputView.isSecureTextEntry != isSecureTextEntry) {
3070 inputView.secureTextEntry = isSecureTextEntry;
3071 [inputView reloadInputViews];
3077 #pragma mark UIIndirectScribbleInteractionDelegate
3079 - (BOOL)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3080 isElementFocused:(UIScribbleElementIdentifier)elementIdentifier
3081 API_AVAILABLE(ios(14.0)) {
3082 return _activeView.scribbleFocusStatus == FlutterScribbleFocusStatusFocused;
3085 - (void)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3086 focusElementIfNeeded:(UIScribbleElementIdentifier)elementIdentifier
3087 referencePoint:(CGPoint)focusReferencePoint
3088 completion:(
void (^)(UIResponder<UITextInput>* focusedInput))completion
3089 API_AVAILABLE(ios(14.0)) {
3090 _activeView.scribbleFocusStatus = FlutterScribbleFocusStatusFocusing;
3091 [_indirectScribbleDelegate flutterTextInputPlugin:self
3092 focusElement:elementIdentifier
3093 atPoint:focusReferencePoint
3094 result:^(id _Nullable result) {
3095 _activeView.scribbleFocusStatus =
3096 FlutterScribbleFocusStatusFocused;
3097 completion(_activeView);
3101 - (BOOL)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3102 shouldDelayFocusForElement:(UIScribbleElementIdentifier)elementIdentifier
3103 API_AVAILABLE(ios(14.0)) {
3107 - (void)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3108 willBeginWritingInElement:(UIScribbleElementIdentifier)elementIdentifier
3109 API_AVAILABLE(ios(14.0)) {
3112 - (void)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3113 didFinishWritingInElement:(UIScribbleElementIdentifier)elementIdentifier
3114 API_AVAILABLE(ios(14.0)) {
3117 - (CGRect)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3118 frameForElement:(UIScribbleElementIdentifier)elementIdentifier
3119 API_AVAILABLE(ios(14.0)) {
3120 NSValue* elementValue = [_scribbleElements objectForKey:elementIdentifier];
3121 if (elementValue == nil) {
3124 return [elementValue CGRectValue];
3127 - (void)indirectScribbleInteraction:(UIIndirectScribbleInteraction*)interaction
3128 requestElementsInRect:(CGRect)rect
3130 (
void (^)(NSArray<UIScribbleElementIdentifier>* elements))completion
3131 API_AVAILABLE(ios(14.0)) {
3132 [_indirectScribbleDelegate
3133 flutterTextInputPlugin:self
3134 requestElementsInRect:rect
3135 result:^(id _Nullable result) {
3136 NSMutableArray<UIScribbleElementIdentifier>* elements =
3137 [[NSMutableArray alloc] init];
3138 if ([result isKindOfClass:[NSArray class]]) {
3139 for (NSArray* elementArray in result) {
3140 [elements addObject:elementArray[0]];
3143 valueWithCGRect:CGRectMake(
3144 [elementArray[1] floatValue],
3145 [elementArray[2] floatValue],
3146 [elementArray[3] floatValue],
3147 [elementArray[4] floatValue])]
3148 forKey:elementArray[0]];
3151 completion(elements);
3155 #pragma mark - Methods related to Scribble support
3159 if (@available(iOS 14.0, *)) {
3161 if (parentView != nil) {
3162 UIIndirectScribbleInteraction* scribbleInteraction = [[UIIndirectScribbleInteraction alloc]
3163 initWithDelegate:(id<UIIndirectScribbleInteractionDelegate>)self];
3164 [parentView addInteraction:scribbleInteraction];
3171 - (void)resetViewResponder {
3172 _viewResponder = nil;
3176 #pragma mark FlutterKeySecondaryResponder
3191 - (id)flutterFirstResponder {
3192 if (
self.isFirstResponder) {
3195 for (UIView* subView in
self.subviews) {
3196 UIView* firstResponder = subView.flutterFirstResponder;
3197 if (firstResponder) {
3198 return firstResponder;