mirror of
https://github.com/status-im/react-native.git
synced 2025-01-14 19:44:13 +00:00
Better TextInput: Fixing multiline <TextInput> insets and prepare for auto-expanding feature
Summary: Several things: * The mess with insets was fixed. Previously we tried to compensate the insets difference with `UITextField` by adjusting `textContainerInset` property, moreover we delegated negative part of this compensation to the view inset. That was terrible because it breaks `contentSize` computation, complicates whole insets consept, complicates everything; it just was not right. Now we are fixing the top and left inset differences in different places. We disable left and right 5pt margin by setting `_textView.textContainer.lineFragmentPadding = 0` and we introduce top 5px inset as a DEFAULT value for top inset for common multiline <TextInput> (this value can be easilly overwritten in Javascript). * Internal layout and contentSize computations were unified and simplified. * Now we report `intrinsicContentSize` value to Yoga, one step before auto-expandable TextInput. Depends on D4640207. Reviewed By: mmmulani Differential Revision: D4645921 fbshipit-source-id: da5988ebac50be967caecd71e780c014f6eb257a
This commit is contained in:
parent
3acafd1f3d
commit
1b013cd30c
@ -678,6 +678,7 @@ const TextInput = React.createClass({
|
|||||||
if (props.inputView) {
|
if (props.inputView) {
|
||||||
children = [children, props.inputView];
|
children = [children, props.inputView];
|
||||||
}
|
}
|
||||||
|
props.style.unshift(styles.multilineInput);
|
||||||
textContainer =
|
textContainer =
|
||||||
<RCTTextView
|
<RCTTextView
|
||||||
ref={this._setNativeRef}
|
ref={this._setNativeRef}
|
||||||
@ -867,6 +868,12 @@ var styles = StyleSheet.create({
|
|||||||
input: {
|
input: {
|
||||||
alignSelf: 'stretch',
|
alignSelf: 'stretch',
|
||||||
},
|
},
|
||||||
|
multilineInput: {
|
||||||
|
// This default top inset makes RCTTextView seem as close as possible
|
||||||
|
// to single-line RCTTextField defaults, using the system defaults
|
||||||
|
// of font size 17 and a height of 31 points.
|
||||||
|
paddingTop: 5,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
module.exports = TextInput;
|
module.exports = TextInput;
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
#import <React/RCTView.h>
|
#import <React/RCTView.h>
|
||||||
#import <React/UIView+React.h>
|
#import <React/UIView+React.h>
|
||||||
|
|
||||||
@class RCTEventDispatcher;
|
@class RCTBridge;
|
||||||
|
|
||||||
@interface RCTTextView : RCTView <UITextViewDelegate>
|
@interface RCTTextView : RCTView <UITextViewDelegate>
|
||||||
|
|
||||||
@ -28,6 +28,7 @@
|
|||||||
@property (nonatomic, strong) UIFont *font;
|
@property (nonatomic, strong) UIFont *font;
|
||||||
@property (nonatomic, assign) NSInteger mostRecentEventCount;
|
@property (nonatomic, assign) NSInteger mostRecentEventCount;
|
||||||
@property (nonatomic, strong) NSNumber *maxLength;
|
@property (nonatomic, strong) NSNumber *maxLength;
|
||||||
|
@property (nonatomic, assign, readonly) CGSize contentSize;
|
||||||
|
|
||||||
@property (nonatomic, copy) RCTDirectEventBlock onChange;
|
@property (nonatomic, copy) RCTDirectEventBlock onChange;
|
||||||
@property (nonatomic, copy) RCTDirectEventBlock onContentSizeChange;
|
@property (nonatomic, copy) RCTDirectEventBlock onContentSizeChange;
|
||||||
@ -35,7 +36,7 @@
|
|||||||
@property (nonatomic, copy) RCTDirectEventBlock onTextInput;
|
@property (nonatomic, copy) RCTDirectEventBlock onTextInput;
|
||||||
@property (nonatomic, copy) RCTDirectEventBlock onScroll;
|
@property (nonatomic, copy) RCTDirectEventBlock onScroll;
|
||||||
|
|
||||||
- (instancetype)initWithEventDispatcher:(RCTEventDispatcher *)eventDispatcher NS_DESIGNATED_INITIALIZER;
|
- (instancetype)initWithBridge:(RCTBridge *)bridge NS_DESIGNATED_INITIALIZER;
|
||||||
|
|
||||||
- (void)performTextUpdate;
|
- (void)performTextUpdate;
|
||||||
|
|
||||||
|
@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
#import <React/RCTConvert.h>
|
#import <React/RCTConvert.h>
|
||||||
#import <React/RCTEventDispatcher.h>
|
#import <React/RCTEventDispatcher.h>
|
||||||
|
#import <React/RCTUIManager.h>
|
||||||
#import <React/RCTUtils.h>
|
#import <React/RCTUtils.h>
|
||||||
#import <React/UIView+React.h>
|
#import <React/UIView+React.h>
|
||||||
|
|
||||||
@ -69,6 +70,7 @@
|
|||||||
|
|
||||||
@implementation RCTTextView
|
@implementation RCTTextView
|
||||||
{
|
{
|
||||||
|
RCTBridge *_bridge;
|
||||||
RCTEventDispatcher *_eventDispatcher;
|
RCTEventDispatcher *_eventDispatcher;
|
||||||
|
|
||||||
NSString *_placeholder;
|
NSString *_placeholder;
|
||||||
@ -87,22 +89,25 @@
|
|||||||
NSInteger _nativeEventCount;
|
NSInteger _nativeEventCount;
|
||||||
|
|
||||||
CGSize _previousContentSize;
|
CGSize _previousContentSize;
|
||||||
BOOL _viewDidCompleteInitialLayout;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (instancetype)initWithEventDispatcher:(RCTEventDispatcher *)eventDispatcher
|
- (instancetype)initWithBridge:(RCTBridge *)bridge
|
||||||
{
|
{
|
||||||
RCTAssertParam(eventDispatcher);
|
RCTAssertParam(bridge);
|
||||||
|
|
||||||
if ((self = [super initWithFrame:CGRectZero])) {
|
if (self = [super initWithFrame:CGRectZero]) {
|
||||||
_contentInset = UIEdgeInsetsZero;
|
_contentInset = UIEdgeInsetsZero;
|
||||||
_eventDispatcher = eventDispatcher;
|
_bridge = bridge;
|
||||||
|
_eventDispatcher = bridge.eventDispatcher;
|
||||||
_placeholderTextColor = [self defaultPlaceholderTextColor];
|
_placeholderTextColor = [self defaultPlaceholderTextColor];
|
||||||
_blurOnSubmit = NO;
|
_blurOnSubmit = NO;
|
||||||
|
|
||||||
_textView = [[RCTUITextView alloc] initWithFrame:CGRectZero];
|
_textView = [[RCTUITextView alloc] initWithFrame:self.bounds];
|
||||||
|
_textView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
|
||||||
_textView.backgroundColor = [UIColor clearColor];
|
_textView.backgroundColor = [UIColor clearColor];
|
||||||
_textView.textColor = [UIColor blackColor];
|
_textView.textColor = [UIColor blackColor];
|
||||||
|
// This line actually removes 5pt (default value) left and right padding in UITextView.
|
||||||
|
_textView.textContainer.lineFragmentPadding = 0;
|
||||||
#if !TARGET_OS_TV
|
#if !TARGET_OS_TV
|
||||||
_textView.scrollsToTop = NO;
|
_textView.scrollsToTop = NO;
|
||||||
#endif
|
#endif
|
||||||
@ -132,7 +137,7 @@ RCT_NOT_IMPLEMENTED(- (instancetype)initWithCoder:(NSCoder *)aDecoder)
|
|||||||
// If this <TextInput> is in rich text editing mode, and the child <Text> node providing rich text
|
// If this <TextInput> is in rich text editing mode, and the child <Text> node providing rich text
|
||||||
// styling has a backgroundColor, then the attributedText produced by the child <Text> node will have an
|
// styling has a backgroundColor, then the attributedText produced by the child <Text> node will have an
|
||||||
// NSBackgroundColor attribute. We need to forward this attribute to the text view manually because the text view
|
// NSBackgroundColor attribute. We need to forward this attribute to the text view manually because the text view
|
||||||
// always has a clear background color in -initWithEventDispatcher:.
|
// always has a clear background color in `initWithBridge:`.
|
||||||
//
|
//
|
||||||
// TODO: This should be removed when the related hack in -performPendingTextUpdate is removed.
|
// TODO: This should be removed when the related hack in -performPendingTextUpdate is removed.
|
||||||
if (subview.backgroundColor) {
|
if (subview.backgroundColor) {
|
||||||
@ -237,60 +242,20 @@ static NSAttributedString *removeReactTagFromString(NSAttributedString *string)
|
|||||||
[_textView layoutIfNeeded];
|
[_textView layoutIfNeeded];
|
||||||
|
|
||||||
[self updatePlaceholderVisibility];
|
[self updatePlaceholderVisibility];
|
||||||
[self updateContentSize];
|
[self invalidateContentSize];
|
||||||
|
|
||||||
_blockTextShouldChange = NO;
|
_blockTextShouldChange = NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)updateFrames
|
|
||||||
{
|
|
||||||
// Adjust the insets so that they are as close as possible to single-line
|
|
||||||
// RCTTextField defaults, using the system defaults of font size 17 and a
|
|
||||||
// height of 31 points.
|
|
||||||
//
|
|
||||||
// We apply the left inset to the frame since a negative left text-container
|
|
||||||
// inset mysteriously causes the text to be hidden until the text view is
|
|
||||||
// first focused.
|
|
||||||
UIEdgeInsets adjustedFrameInset = UIEdgeInsetsZero;
|
|
||||||
adjustedFrameInset.left = _contentInset.left - 5;
|
|
||||||
|
|
||||||
UIEdgeInsets adjustedTextContainerInset = _contentInset;
|
|
||||||
adjustedTextContainerInset.top += 5;
|
|
||||||
adjustedTextContainerInset.left = 0;
|
|
||||||
|
|
||||||
CGRect frame = UIEdgeInsetsInsetRect(self.bounds, adjustedFrameInset);
|
|
||||||
_textView.frame = frame;
|
|
||||||
_placeholderView.frame = frame;
|
|
||||||
[self updateContentSize];
|
|
||||||
|
|
||||||
_textView.textContainerInset = adjustedTextContainerInset;
|
|
||||||
_placeholderView.textContainerInset = adjustedTextContainerInset;
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)updateContentSize
|
|
||||||
{
|
|
||||||
CGSize size = _textView.frame.size;
|
|
||||||
size.height = [_textView sizeThatFits:CGSizeMake(size.width, INFINITY)].height;
|
|
||||||
|
|
||||||
if (_viewDidCompleteInitialLayout && _onContentSizeChange && !CGSizeEqualToSize(_previousContentSize, size)) {
|
|
||||||
_previousContentSize = size;
|
|
||||||
_onContentSizeChange(@{
|
|
||||||
@"contentSize": @{
|
|
||||||
@"height": @(size.height),
|
|
||||||
@"width": @(size.width),
|
|
||||||
},
|
|
||||||
@"target": self.reactTag,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- (void)updatePlaceholder
|
- (void)updatePlaceholder
|
||||||
{
|
{
|
||||||
[_placeholderView removeFromSuperview];
|
[_placeholderView removeFromSuperview];
|
||||||
_placeholderView = nil;
|
_placeholderView = nil;
|
||||||
|
|
||||||
if (_placeholder) {
|
if (_placeholder) {
|
||||||
_placeholderView = [[UITextView alloc] initWithFrame:self.bounds];
|
_placeholderView = [[UITextView alloc] initWithFrame:_textView.frame];
|
||||||
|
_placeholderView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
|
||||||
|
_placeholderView.textContainer.lineFragmentPadding = 0;
|
||||||
_placeholderView.userInteractionEnabled = NO;
|
_placeholderView.userInteractionEnabled = NO;
|
||||||
_placeholderView.backgroundColor = [UIColor clearColor];
|
_placeholderView.backgroundColor = [UIColor clearColor];
|
||||||
_placeholderView.scrollEnabled = NO;
|
_placeholderView.scrollEnabled = NO;
|
||||||
@ -340,7 +305,9 @@ static NSAttributedString *removeReactTagFromString(NSAttributedString *string)
|
|||||||
- (void)setContentInset:(UIEdgeInsets)contentInset
|
- (void)setContentInset:(UIEdgeInsets)contentInset
|
||||||
{
|
{
|
||||||
_contentInset = contentInset;
|
_contentInset = contentInset;
|
||||||
[self updateFrames];
|
_textView.textContainerInset = contentInset;
|
||||||
|
_placeholderView.textContainerInset = contentInset;
|
||||||
|
[self setNeedsLayout];
|
||||||
}
|
}
|
||||||
|
|
||||||
#pragma mark - UITextViewDelegate
|
#pragma mark - UITextViewDelegate
|
||||||
@ -503,8 +470,7 @@ static NSAttributedString *removeReactTagFromString(NSAttributedString *string)
|
|||||||
}
|
}
|
||||||
|
|
||||||
[self updatePlaceholderVisibility];
|
[self updatePlaceholderVisibility];
|
||||||
[self updateContentSize]; //keep the text wrapping when the length of
|
[self invalidateContentSize];
|
||||||
//the textline has been extended longer than the length of textinputView
|
|
||||||
} else if (eventLag > RCTTextUpdateLagWarningThreshold) {
|
} else if (eventLag > RCTTextUpdateLagWarningThreshold) {
|
||||||
RCTLogWarn(@"Native TextInput(%@) is %zd events ahead of JS - try to make your JS faster.", self.text, eventLag);
|
RCTLogWarn(@"Native TextInput(%@) is %zd events ahead of JS - try to make your JS faster.", self.text, eventLag);
|
||||||
}
|
}
|
||||||
@ -595,7 +561,7 @@ static BOOL findMismatch(NSString *first, NSString *second, NSRange *firstRange,
|
|||||||
- (void)textViewDidChange:(UITextView *)textView
|
- (void)textViewDidChange:(UITextView *)textView
|
||||||
{
|
{
|
||||||
[self updatePlaceholderVisibility];
|
[self updatePlaceholderVisibility];
|
||||||
[self updateContentSize];
|
[self invalidateContentSize];
|
||||||
|
|
||||||
// Detect when textView updates happend that didn't invoke `shouldChangeTextInRange`
|
// Detect when textView updates happend that didn't invoke `shouldChangeTextInRange`
|
||||||
// (e.g. typing simplified chinese in pinyin will insert and remove spaces without
|
// (e.g. typing simplified chinese in pinyin will insert and remove spaces without
|
||||||
@ -664,6 +630,8 @@ static BOOL findMismatch(NSString *first, NSString *second, NSRange *firstRange,
|
|||||||
eventCount:_nativeEventCount];
|
eventCount:_nativeEventCount];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pragma mark - UIResponder
|
||||||
|
|
||||||
- (BOOL)isFirstResponder
|
- (BOOL)isFirstResponder
|
||||||
{
|
{
|
||||||
return [_textView isFirstResponder];
|
return [_textView isFirstResponder];
|
||||||
@ -695,17 +663,63 @@ static BOOL findMismatch(NSString *first, NSString *second, NSRange *firstRange,
|
|||||||
return [_textView resignFirstResponder];
|
return [_textView resignFirstResponder];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pragma mark - Content Size
|
||||||
|
|
||||||
|
- (CGSize)contentSize
|
||||||
|
{
|
||||||
|
// Returning value does NOT include insets.
|
||||||
|
CGSize contentSize = self.intrinsicContentSize;
|
||||||
|
contentSize.width -= _contentInset.left + _contentInset.right;
|
||||||
|
contentSize.height -= _contentInset.top + _contentInset.bottom;
|
||||||
|
return contentSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)invalidateContentSize
|
||||||
|
{
|
||||||
|
CGSize contentSize = self.contentSize;
|
||||||
|
|
||||||
|
if (CGSizeEqualToSize(_previousContentSize, contentSize)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_previousContentSize = contentSize;
|
||||||
|
|
||||||
|
[_bridge.uiManager setIntrinsicContentSize:contentSize forView:self];
|
||||||
|
|
||||||
|
if (_onContentSizeChange) {
|
||||||
|
_onContentSizeChange(@{
|
||||||
|
@"contentSize": @{
|
||||||
|
@"height": @(contentSize.height),
|
||||||
|
@"width": @(contentSize.width),
|
||||||
|
},
|
||||||
|
@"target": self.reactTag,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#pragma mark - Layout
|
||||||
|
|
||||||
|
- (CGSize)intrinsicContentSize
|
||||||
|
{
|
||||||
|
// Calling `sizeThatFits:` is probably more expensive method to compute
|
||||||
|
// content size compare to direct access `_textView.contentSize` property,
|
||||||
|
// but seems `sizeThatFits:` returns more reliable and consistent result.
|
||||||
|
// Returning value DOES include insets.
|
||||||
|
return [self sizeThatFits:CGSizeMake(self.bounds.size.width, INFINITY)];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (CGSize)sizeThatFits:(CGSize)size
|
||||||
|
{
|
||||||
|
return [_textView sizeThatFits:size];
|
||||||
|
}
|
||||||
|
|
||||||
- (void)layoutSubviews
|
- (void)layoutSubviews
|
||||||
{
|
{
|
||||||
[super layoutSubviews];
|
[super layoutSubviews];
|
||||||
|
[self invalidateContentSize];
|
||||||
// Start sending content size updates only after the view has been laid out
|
|
||||||
// otherwise we send multiple events with bad dimensions on initial render.
|
|
||||||
_viewDidCompleteInitialLayout = YES;
|
|
||||||
|
|
||||||
[self updateFrames];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pragma mark - Default values
|
||||||
|
|
||||||
- (UIFont *)defaultPlaceholderFont
|
- (UIFont *)defaultPlaceholderFont
|
||||||
{
|
{
|
||||||
return [UIFont systemFontOfSize:17];
|
return [UIFont systemFontOfSize:17];
|
||||||
|
@ -29,7 +29,7 @@ RCT_EXPORT_MODULE()
|
|||||||
|
|
||||||
- (UIView *)view
|
- (UIView *)view
|
||||||
{
|
{
|
||||||
return [[RCTTextView alloc] initWithEventDispatcher:self.bridge.eventDispatcher];
|
return [[RCTTextView alloc] initWithBridge:self.bridge];
|
||||||
}
|
}
|
||||||
|
|
||||||
RCT_REMAP_VIEW_PROPERTY(autoCapitalize, textView.autocapitalizationType, UITextAutocapitalizationType)
|
RCT_REMAP_VIEW_PROPERTY(autoCapitalize, textView.autocapitalizationType, UITextAutocapitalizationType)
|
||||||
|
@ -87,7 +87,8 @@ RCT_EXTERN NSString *const RCTUIManagerRootViewKey;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the natural size of a view, which is used when no explicit size is set.
|
* Set the natural size of a view, which is used when no explicit size is set.
|
||||||
* Use UIViewNoIntrinsicMetric to ignore a dimension.
|
* Use `UIViewNoIntrinsicMetric` to ignore a dimension.
|
||||||
|
* The `size` must NOT include padding and border.
|
||||||
*/
|
*/
|
||||||
- (void)setIntrinsicContentSize:(CGSize)size forView:(UIView *)view;
|
- (void)setIntrinsicContentSize:(CGSize)size forView:(UIView *)view;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user