iOS: input support textfield

This commit is contained in:
pengfei.zhou 2021-09-08 20:20:06 +08:00 committed by osborn
parent 5b3d45f607
commit 88c594ae10
2 changed files with 263 additions and 68 deletions

View File

@ -23,8 +23,7 @@
#import "DoricViewNode.h" #import "DoricViewNode.h"
NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_BEGIN
@interface DoricInputView : UITextView @interface DoricInputView : UIView
@property(nonatomic, strong) UILabel *placeholderLabel;
@end @end
@interface DoricInputNode : DoricViewNode<DoricInputView *> @interface DoricInputNode : DoricViewNode<DoricInputView *>

View File

@ -31,8 +31,12 @@
typedef void (^onSubmitEditingBlock)(NSString *text, DoricInputNode *node); typedef void (^onSubmitEditingBlock)(NSString *text, DoricInputNode *node);
@implementation DoricInputView
@interface DoricMultilineInput : UITextView
@property(nonatomic, strong) UILabel *placeholderLabel;
@end
@implementation DoricMultilineInput
- (instancetype)init { - (instancetype)init {
if (self = [super init]) { if (self = [super init]) {
self.font = [UIFont systemFontOfSize:12]; self.font = [UIFont systemFontOfSize:12];
@ -57,7 +61,7 @@ - (void)layoutSubviews {
self.placeholderLabel.x = lineFragmentPadding + textContainerInset.left; self.placeholderLabel.x = lineFragmentPadding + textContainerInset.left;
self.placeholderLabel.y = textContainerInset.top; self.placeholderLabel.y = textContainerInset.top;
float desiredWidth = self.width - lineFragmentPadding * 2 - textContainerInset.left - textContainerInset.right; CGFloat desiredWidth = self.width - lineFragmentPadding * 2 - textContainerInset.left - textContainerInset.right;
CGSize fitSize = [self.placeholderLabel sizeThatFits:CGSizeMake(desiredWidth, 0)]; CGSize fitSize = [self.placeholderLabel sizeThatFits:CGSizeMake(desiredWidth, 0)];
if (fitSize.width < desiredWidth) { if (fitSize.width < desiredWidth) {
@ -76,9 +80,162 @@ - (CGSize)sizeThatFits:(CGSize)size {
} }
} }
- (void)setTextAlignment:(NSTextAlignment)textAlignment {
[super setTextAlignment:textAlignment];
self.placeholderLabel.textAlignment = textAlignment;
}
@end @end
@interface DoricInputNode () <UITextViewDelegate> @interface DoricInputView ()
@property(nonatomic, strong) DoricMultilineInput *multiLineInput;
@property(nonatomic, strong) UITextField *singleLineInput;
@end
@implementation DoricInputView
- (instancetype)init {
if (self = [super init]) {
_multiLineInput = [DoricMultilineInput new];
_multiLineInput.backgroundColor = UIColor.clearColor;
[self addSubview:_multiLineInput];
_singleLineInput = [UITextField new];
_singleLineInput.backgroundColor = UIColor.clearColor;
[self addSubview:_singleLineInput];
self.multiline = YES;
}
return self;
}
- (void)setMultiline:(BOOL)multiline {
if (multiline == self.multiline) {
return;
}
self.singleLineInput.hidden = multiline;
self.multiLineInput.hidden = !multiline;
}
- (BOOL)multiline {
return self.singleLineInput.hidden;
}
- (void)layoutSubviews {
[super layoutSubviews];
if (self.multiline) {
self.multiLineInput.width = self.width;
self.multiLineInput.height = self.height;
} else {
self.singleLineInput.width = self.width;
self.singleLineInput.height = self.height;
}
}
- (CGSize)sizeThatFits:(CGSize)size {
if (self.multiline) {
return [self.multiLineInput sizeThatFits:size];
} else {
return [self.singleLineInput sizeThatFits:size];
}
}
- (void)setText:(NSString *)text {
self.multiLineInput.text = text;
self.singleLineInput.text = text;
}
- (NSString *)text {
if (self.multiline) {
return self.multiLineInput.text;
} else {
return self.singleLineInput.text;
}
}
- (void)setFont:(UIFont *)font {
self.multiLineInput.font = font;
self.singleLineInput.font = font;
}
- (UIFont *)font {
if (self.multiline) {
return self.multiLineInput.font;
} else {
return self.singleLineInput.font;
}
}
- (void)setTextColor:(UIColor *)color {
self.multiLineInput.textColor = color;
self.singleLineInput.textColor = color;
}
- (UIColor *)textColor {
if (self.multiline) {
return self.multiLineInput.textColor;
} else {
return self.singleLineInput.textColor;
}
}
- (void)setTextAlignment:(NSTextAlignment)textAlignment {
self.multiLineInput.textAlignment = textAlignment;
self.singleLineInput.textAlignment = textAlignment;
}
- (NSTextAlignment)textAlignment {
if (self.multiline) {
return self.multiLineInput.textAlignment;
} else {
return self.singleLineInput.textAlignment;
}
}
- (void)setHintText:(NSString *)text {
self.multiLineInput.placeholderLabel.text = text;
self.singleLineInput.placeholder = text;
}
- (NSString *)hintText {
if (self.multiline) {
return self.multiLineInput.placeholderLabel.text;
} else {
return self.singleLineInput.placeholder;
}
}
- (void)setHintTextColor:(UIColor *)color {
self.multiLineInput.placeholderLabel.textColor = color;
[self.singleLineInput setValue:color forKeyPath:@"_placeholderLabel.textColor"];
}
- (void)setHintFont:(UIFont *)font {
self.multiLineInput.placeholderLabel.font = font;
[self.singleLineInput setValue:font forKeyPath:@"_placeholderLabel.font"];
}
- (void)setKeyboardType:(UIKeyboardType)keyboardType {
self.multiLineInput.keyboardType = keyboardType;
self.singleLineInput.keyboardType = keyboardType;
}
- (void)setReturnKeyType:(UIReturnKeyType)returnKeyType {
self.multiLineInput.returnKeyType = returnKeyType;
self.singleLineInput.returnKeyType = returnKeyType;
}
- (void)setSecureTextEntry:(BOOL)secureTextEntry {
self.multiLineInput.secureTextEntry = secureTextEntry;
self.singleLineInput.secureTextEntry = secureTextEntry;
}
- (void)setEditable:(BOOL)editable {
self.multiLineInput.editable = editable;
self.singleLineInput.enabled = editable;
}
@end
@interface DoricInputNode () <UITextViewDelegate, UITextFieldDelegate>
@property(nonatomic, copy) onTextChangeBlock onTextChange; @property(nonatomic, copy) onTextChangeBlock onTextChange;
@property(nonatomic, copy) onFocusChangeBlock onFocusChange; @property(nonatomic, copy) onFocusChangeBlock onFocusChange;
@property(nonatomic, copy) onSubmitEditingBlock onSubmitEditing; @property(nonatomic, copy) onSubmitEditingBlock onSubmitEditing;
@ -88,13 +245,14 @@ @interface DoricInputNode () <UITextViewDelegate>
@implementation DoricInputNode @implementation DoricInputNode
- (DoricInputView *)build { - (DoricInputView *)build {
DoricInputView *v = [[DoricInputView alloc] init]; DoricInputView *v = [DoricInputView new];
v.delegate = self; v.singleLineInput.delegate = self;
v.textContainer.lineFragmentPadding = 0; v.multiLineInput.delegate = self;
v.doricLayout.paddingTop = v.textContainerInset.top; v.multiLineInput.textContainer.lineFragmentPadding = 0;
v.doricLayout.paddingBottom = v.textContainerInset.bottom; v.doricLayout.paddingTop = v.multiLineInput.textContainerInset.top;
v.doricLayout.paddingLeft = v.textContainerInset.left; v.doricLayout.paddingBottom = v.multiLineInput.textContainerInset.bottom;
v.doricLayout.paddingRight = v.textContainerInset.right; v.doricLayout.paddingLeft = v.multiLineInput.textContainerInset.left;
v.doricLayout.paddingRight = v.multiLineInput.textContainerInset.right;
return v; return v;
} }
@ -119,7 +277,6 @@ - (void)blendView:(DoricInputView *)view forPropName:(NSString *)name propValue:
alignment = NSTextAlignmentRight; alignment = NSTextAlignmentRight;
} }
view.textAlignment = alignment; view.textAlignment = alignment;
view.placeholderLabel.textAlignment = alignment;
} else if ([name isEqualToString:@"font"]) { } else if ([name isEqualToString:@"font"]) {
NSString *iconfont = prop; NSString *iconfont = prop;
UIFont *font = [UIFont fontWithName:[iconfont stringByReplacingOccurrencesOfString:@".ttf" withString:@""] UIFont *font = [UIFont fontWithName:[iconfont stringByReplacingOccurrencesOfString:@".ttf" withString:@""]
@ -127,25 +284,18 @@ - (void)blendView:(DoricInputView *)view forPropName:(NSString *)name propValue:
view.font = font; view.font = font;
} else if ([name isEqualToString:@"multiline"]) { } else if ([name isEqualToString:@"multiline"]) {
BOOL value = [(NSNumber *) prop boolValue]; BOOL value = [(NSNumber *) prop boolValue];
if (!value) { view.multiline = value;
view.textContainer.maximumNumberOfLines = 1;
if (view.text.length > 0) {
view.text = [view.text stringByReplacingOccurrencesOfString:@"\n" withString:@" "];
}
} else {
view.textContainer.maximumNumberOfLines = 0;
}
} else if ([name isEqualToString:@"beforeTextChange"]) { } else if ([name isEqualToString:@"beforeTextChange"]) {
self.beforeTextChangeFuncId = prop; self.beforeTextChangeFuncId = prop;
} else if ([name isEqualToString:@"hintText"]) { } else if ([name isEqualToString:@"hintText"]) {
view.placeholderLabel.text = (NSString *) prop; view.hintText = (NSString *) prop;
} else if ([name isEqualToString:@"hintTextColor"]) { } else if ([name isEqualToString:@"hintTextColor"]) {
view.placeholderLabel.textColor = DoricColor(prop); view.hintTextColor = DoricColor(prop);
} else if ([name isEqualToString:@"hintFont"]) { } else if ([name isEqualToString:@"hintFont"]) {
NSString *iconfont = prop; NSString *iconfont = prop;
UIFont *font = [UIFont fontWithName:[iconfont stringByReplacingOccurrencesOfString:@".ttf" withString:@""] UIFont *font = [UIFont fontWithName:[iconfont stringByReplacingOccurrencesOfString:@".ttf" withString:@""]
size:view.font.pointSize]; size:view.font.pointSize];
view.placeholderLabel.font = font; view.hintFont = font;
} else if ([name isEqualToString:@"onTextChange"]) { } else if ([name isEqualToString:@"onTextChange"]) {
if ([prop isKindOfClass:[NSString class]]) { if ([prop isKindOfClass:[NSString class]]) {
self.onTextChange = ^(NSString *text, DoricInputNode *node) { self.onTextChange = ^(NSString *text, DoricInputNode *node) {
@ -193,9 +343,6 @@ - (void)blendView:(DoricInputView *)view forPropName:(NSString *)name propValue:
} else if ([name isEqualToString:@"editable"]) { } else if ([name isEqualToString:@"editable"]) {
view.editable = [(NSNumber *) prop boolValue]; view.editable = [(NSNumber *) prop boolValue];
} else if ([name isEqualToString:@"returnKeyType"]) { } else if ([name isEqualToString:@"returnKeyType"]) {
if (view.textContainer.maximumNumberOfLines == 1) {
return;
}
switch ([(NSNumber *) prop integerValue]) { switch ([(NSNumber *) prop integerValue]) {
case 1: case 1:
view.returnKeyType = UIReturnKeyDone; view.returnKeyType = UIReturnKeyDone;
@ -226,9 +373,9 @@ - (void)blendView:(DoricInputView *)view forPropName:(NSString *)name propValue:
self.onSubmitEditing = nil; self.onSubmitEditing = nil;
} }
} else if ([name isEqualToString:@"enableHorizontalScrollBar"]) { } else if ([name isEqualToString:@"enableHorizontalScrollBar"]) {
view.showsHorizontalScrollIndicator = [prop boolValue];; view.multiLineInput.showsHorizontalScrollIndicator = [prop boolValue];;
} else if ([name isEqualToString:@"enableVerticalScrollBar"]) { } else if ([name isEqualToString:@"enableVerticalScrollBar"]) {
view.showsVerticalScrollIndicator = [prop boolValue];; view.multiLineInput.showsVerticalScrollIndicator = [prop boolValue];;
} else { } else {
[super blendView:view forPropName:name propValue:prop]; [super blendView:view forPropName:name propValue:prop];
} }
@ -240,21 +387,22 @@ - (void)blend:(NSDictionary *)props {
- (void)afterBlended:(NSDictionary *)props { - (void)afterBlended:(NSDictionary *)props {
[super afterBlended:props]; [super afterBlended:props];
if (self.view.doricLayout.paddingTop != self.view.textContainerInset.top if (self.view.multiline) {
|| self.view.doricLayout.paddingLeft != self.view.textContainerInset.left if (self.view.doricLayout.paddingTop != self.view.multiLineInput.textContainerInset.top
|| self.view.doricLayout.paddingBottom != self.view.textContainerInset.bottom || self.view.doricLayout.paddingLeft != self.view.multiLineInput.textContainerInset.left
|| self.view.doricLayout.paddingRight != self.view.textContainerInset.right) { || self.view.doricLayout.paddingBottom != self.view.multiLineInput.textContainerInset.bottom
self.view.textContainerInset = UIEdgeInsetsMake(self.view.doricLayout.paddingTop, self.view.doricLayout.paddingLeft, self.view.doricLayout.paddingBottom, self.view.doricLayout.paddingRight); || self.view.doricLayout.paddingRight != self.view.multiLineInput.textContainerInset.right) {
self.view.multiLineInput.textContainerInset = UIEdgeInsetsMake(self.view.doricLayout.paddingTop, self.view.doricLayout.paddingLeft, self.view.doricLayout.paddingBottom, self.view.doricLayout.paddingRight);
} }
UIFont *font = self.view.placeholderLabel.font; UIFont *font = self.view.multiLineInput.placeholderLabel.font;
if (font) { if (font) {
self.view.placeholderLabel.font = [self.view.placeholderLabel.font fontWithSize:self.view.font.pointSize]; self.view.multiLineInput.placeholderLabel.font = [self.view.multiLineInput.placeholderLabel.font fontWithSize:self.view.font.pointSize];
} else { } else {
self.view.placeholderLabel.font = self.view.font; self.view.multiLineInput.placeholderLabel.font = self.view.multiLineInput.font;
}
self.view.multiLineInput.placeholderLabel.numberOfLines = self.view.multiLineInput.textContainer.maximumNumberOfLines;
} }
self.view.placeholderLabel.numberOfLines = self.view.textContainer.maximumNumberOfLines;
} }
- (void)requestLayout { - (void)requestLayout {
@ -269,22 +417,39 @@ - (NSString *)getText {
} }
- (void)setSelection:(NSDictionary *)params withPromise:(DoricPromise *)promise { - (void)setSelection:(NSDictionary *)params withPromise:(DoricPromise *)promise {
NSString *start = params[@"start"]; NSNumber *start = params[@"start"];
NSString *end = params[@"end"]; NSNumber *end = params[@"end"];
if (self.view.multiline) {
if (([start isKindOfClass:[NSString class]] || [start isKindOfClass:[NSNumber class]]) && self.view.multiLineInput.selectedRange = NSMakeRange(start.unsignedIntegerValue, end.unsignedIntegerValue - start.unsignedIntegerValue);
([start isKindOfClass:[NSString class]] || [start isKindOfClass:[NSNumber class]])) { } else {
self.view.selectedRange = NSMakeRange(start.intValue, end.intValue - start.intValue); UITextPosition *startPos = [self.view.singleLineInput positionFromPosition:self.view.singleLineInput.beginningOfDocument
offset:start.unsignedIntegerValue];
UITextPosition *endPos = [self.view.singleLineInput positionFromPosition:self.view.singleLineInput.beginningOfDocument
offset:end.unsignedIntegerValue];
self.view.singleLineInput.selectedTextRange = [self.view.singleLineInput textRangeFromPosition:startPos
toPosition:endPos];
} }
[promise resolve:nil]; [promise resolve:nil];
} }
- (NSDictionary *)getSelection { - (NSDictionary *)getSelection {
if (self.view.multiline) {
return @{ return @{
@"start": @([self.view offsetFromPosition:self.view.beginningOfDocument toPosition:self.view.selectedTextRange.start]), @"start": @([self.view.multiLineInput offsetFromPosition:self.view.multiLineInput.beginningOfDocument
@"end": @([self.view offsetFromPosition:self.view.beginningOfDocument toPosition:self.view.selectedTextRange.end]), toPosition:self.view.multiLineInput.selectedTextRange.start]),
@"end": @([self.view.multiLineInput offsetFromPosition:self.view.multiLineInput.beginningOfDocument
toPosition:self.view.multiLineInput.selectedTextRange.end]),
}; };
} else {
UITextRange *range = self.view.singleLineInput.selectedTextRange;
return @{
@"start": @( [self.view.singleLineInput offsetFromPosition:self.view.singleLineInput.beginningOfDocument
toPosition:range.start]),
@"end": @( [self.view.singleLineInput offsetFromPosition:self.view.singleLineInput.beginningOfDocument
toPosition:range.end]),
};
}
} }
- (void)requestFocus { - (void)requestFocus {
@ -312,15 +477,6 @@ - (BOOL)textViewShouldEndEditing:(UITextView *)textView {
} }
- (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text { - (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text {
if ([text isEqualToString:@"\n"]) {
if (textView.textContainer.maximumNumberOfLines == 1) {
if (self.onSubmitEditing) {
self.onSubmitEditing(textView.text, self);
}
return NO;
}
}
if (self.beforeTextChangeFuncId) { if (self.beforeTextChangeFuncId) {
DoricAsyncResult *asyncResult = [self DoricAsyncResult *asyncResult = [self
pureCallJSResponse:self.beforeTextChangeFuncId, pureCallJSResponse:self.beforeTextChangeFuncId,
@ -341,9 +497,9 @@ - (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range r
- (void)textViewDidChange:(UITextView *)textView { - (void)textViewDidChange:(UITextView *)textView {
if (textView.markedTextRange || textView.text.length > 0) { if (textView.markedTextRange || textView.text.length > 0) {
self.view.placeholderLabel.hidden = YES; self.view.multiLineInput.placeholderLabel.hidden = YES;
} else { } else {
self.view.placeholderLabel.hidden = NO; self.view.multiLineInput.placeholderLabel.hidden = NO;
} }
if (textView.markedTextRange) return; if (textView.markedTextRange) return;
@ -378,4 +534,44 @@ - (NSString *)limitToHansMaxLength:(NSUInteger)maxLen text:(NSString *)text {
}]; }];
return [text substringWithRange:NSMakeRange(0, subStringRangeLen)]; return [text substringWithRange:NSMakeRange(0, subStringRangeLen)];
} }
- (void)textFieldDidEndEditing:(UITextField *)textField {
if (self.maxLength) {
textField.text = [self limitToHansMaxLength:self.maxLength.unsignedIntValue text:textField.text];
}
if (self.onTextChange) {
self.onTextChange(textField.text, self);
}
[textField setNeedsLayout];
}
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string {
if (self.beforeTextChangeFuncId) {
DoricAsyncResult *asyncResult = [self
pureCallJSResponse:self.beforeTextChangeFuncId,
@{
@"editing": textField.text,
@"start": @(range.location),
@"length": @(range.length),
@"replacement": string,
},
nil];
NSNumber *ret = [asyncResult waitUntilResult:^(JSValue *model) {
return [model toNumber];
}];
return [ret boolValue];
}
return YES;
}
- (BOOL)textFieldShouldReturn:(UITextField *)textField {
if (self.onSubmitEditing) {
self.onSubmitEditing(textField.text, self);
return YES;
} else {
return NO;
}
}
@end @end