mirror of
https://github.com/status-im/react-native.git
synced 2025-01-25 00:39:03 +00:00
2716f53220
Summary: This is a complete rewrite of RCTText, the part of React Native which manages Text and TextInput components. Key points: * It's understandable now. It follows a simple architectural pattern, and it's easy to debug and iterate. Text flow layout is a first-class citizen in React Native layout system now, not just a wired special case. It also brings entirely new possibilities such as nested interleaving <Text> and <View> components. * All <Text>-specific APIs were removed from UIManager and co (it's about ~16 public methods which were used exclusively only by <Text>). * It relies on new Yoga measurement/cloning API and on-dirty handler. So, it removes built-in dirty propagation subsystem from RN completely. * It caches string fragments properly and granularly on a per-node basis which makes updating text-containing components more performant. * It does not instantiate UIView for virtual components which reduces memory utilization. * It drastically improves <TextInput> capabilities (e.g. rich text inside single line <TextInput> is now supported). Screenshots: https://cl.ly/2j3r1V0L0324 https://cl.ly/3N2V3C3d3q3R Reviewed By: mmmulani Differential Revision: D6617326 fbshipit-source-id: 35d4d81b35c9870e9557d0211c0e934e6072a41e
271 lines
10 KiB
Objective-C
271 lines
10 KiB
Objective-C
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "RCTTextAttributes.h"
|
|
|
|
#import <React/RCTAssert.h>
|
|
#import <React/RCTFont.h>
|
|
#import <React/RCTLog.h>
|
|
|
|
NSString *const RCTTextAttributesIsHighlightedAttributeName = @"RCTTextAttributesIsHighlightedAttributeName";
|
|
NSString *const RCTTextAttributesTagAttributeName = @"RCTTextAttributesTagAttributeName";
|
|
|
|
@implementation RCTTextAttributes
|
|
|
|
- (instancetype)init
|
|
{
|
|
if (self = [super init]) {
|
|
_fontSize = NAN;
|
|
_letterSpacing = NAN;
|
|
_textDecorationStyle = NSUnderlineStyleSingle;
|
|
_fontSizeMultiplier = NAN;
|
|
_alignment = NSTextAlignmentNatural;
|
|
_baseWritingDirection = NSWritingDirectionNatural;
|
|
_textShadowRadius = NAN;
|
|
_opacity = NAN;
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)applyTextAttributes:(RCTTextAttributes *)textAttributes
|
|
{
|
|
// Note: All lines marked with `*` does not use explicit/correct rules to compare old and new values becuase
|
|
// their types do not have special designated value representing undefined/unspecified/inherit meaning.
|
|
// We will address this in the future.
|
|
|
|
// Color
|
|
_foregroundColor = textAttributes->_foregroundColor ?: _foregroundColor;
|
|
_backgroundColor = textAttributes->_backgroundColor ?: _backgroundColor;
|
|
_opacity = !isnan(textAttributes->_opacity) ? (isnan(_opacity) ? 1.0 : _opacity) * textAttributes->_opacity : _opacity;
|
|
|
|
// Font
|
|
_fontFamily = textAttributes->_fontFamily ?: _fontFamily;
|
|
_fontSize = !isnan(textAttributes->_fontSize) ? textAttributes->_fontSize : _fontSize;
|
|
_fontSizeMultiplier = !isnan(textAttributes->_fontSizeMultiplier) ? textAttributes->_fontSizeMultiplier : _fontSizeMultiplier;
|
|
_fontWeight = textAttributes->_fontWeight ?: _fontWeight;
|
|
_fontStyle = textAttributes->_fontStyle ?: _fontStyle;
|
|
_fontVariant = textAttributes->_fontVariant ?: _fontVariant;
|
|
_allowFontScaling = textAttributes->_allowFontScaling || _allowFontScaling; // *
|
|
_letterSpacing = !isnan(textAttributes->_letterSpacing) ? textAttributes->_letterSpacing : _letterSpacing;
|
|
|
|
// Paragraph Styles
|
|
_lineHeight = !isnan(textAttributes->_lineHeight) ? textAttributes->_lineHeight : _lineHeight;
|
|
_alignment = textAttributes->_alignment != NSTextAlignmentNatural ? textAttributes->_alignment : _alignment; // *
|
|
_baseWritingDirection = textAttributes->_baseWritingDirection != NSWritingDirectionNatural ? textAttributes->_baseWritingDirection : _baseWritingDirection; // *
|
|
|
|
// Decoration
|
|
_textDecorationColor = textAttributes->_textDecorationColor ?: _textDecorationColor;
|
|
_textDecorationStyle = textAttributes->_textDecorationStyle != NSUnderlineStyleSingle ? textAttributes->_textDecorationStyle : _textDecorationStyle; // *
|
|
_textDecorationLine = textAttributes->_textDecorationLine != RCTTextDecorationLineTypeNone ? textAttributes->_textDecorationLine : _textDecorationLine; // *
|
|
|
|
// Shadow
|
|
_textShadowOffset = !CGSizeEqualToSize(textAttributes->_textShadowOffset, CGSizeZero) ? textAttributes->_textShadowOffset : _textShadowOffset; // *
|
|
_textShadowRadius = !isnan(textAttributes->_textShadowRadius) ? textAttributes->_textShadowRadius : _textShadowRadius;
|
|
_textShadowColor = textAttributes->_textShadowColor ?: _textShadowColor;
|
|
|
|
// Special
|
|
_isHighlighted = textAttributes->_isHighlighted || _isHighlighted; // *
|
|
_tag = textAttributes->_tag ?: _tag;
|
|
_layoutDirection = textAttributes->_layoutDirection != UIUserInterfaceLayoutDirectionLeftToRight ? textAttributes->_layoutDirection : _layoutDirection;
|
|
}
|
|
|
|
- (NSDictionary<NSAttributedStringKey, id> *)effectiveTextAttributes
|
|
{
|
|
NSMutableDictionary<NSAttributedStringKey, id> *attributes =
|
|
[NSMutableDictionary dictionaryWithCapacity:10];
|
|
|
|
// Font
|
|
UIFont *font = self.effectiveFont;
|
|
if (font) {
|
|
attributes[NSFontAttributeName] = font;
|
|
}
|
|
|
|
// Colors
|
|
UIColor *effectiveForegroundColor = self.effectiveForegroundColor;
|
|
|
|
if (_foregroundColor || !isnan(_opacity)) {
|
|
attributes[NSForegroundColorAttributeName] = effectiveForegroundColor;
|
|
}
|
|
|
|
if (_backgroundColor || !isnan(_opacity)) {
|
|
attributes[NSBackgroundColorAttributeName] = self.effectiveBackgroundColor;
|
|
}
|
|
|
|
// Kerning
|
|
if (!isnan(_letterSpacing)) {
|
|
attributes[NSKernAttributeName] = @(_letterSpacing);
|
|
}
|
|
|
|
// Paragraph Style
|
|
NSMutableParagraphStyle *paragraphStyle = [NSMutableParagraphStyle new];
|
|
BOOL isParagraphStyleUsed = NO;
|
|
if (_alignment != NSTextAlignmentNatural) {
|
|
NSTextAlignment alignment = _alignment;
|
|
if (_layoutDirection == UIUserInterfaceLayoutDirectionRightToLeft) {
|
|
if (alignment == NSTextAlignmentRight) {
|
|
alignment = NSTextAlignmentLeft;
|
|
} else if (alignment == NSTextAlignmentLeft) {
|
|
alignment = NSTextAlignmentRight;
|
|
}
|
|
}
|
|
|
|
paragraphStyle.alignment = alignment;
|
|
isParagraphStyleUsed = YES;
|
|
}
|
|
|
|
if (_baseWritingDirection != NSWritingDirectionNatural) {
|
|
paragraphStyle.baseWritingDirection = _baseWritingDirection;
|
|
isParagraphStyleUsed = YES;
|
|
}
|
|
|
|
if (!isnan(_lineHeight)) {
|
|
CGFloat lineHeight = _lineHeight * self.effectiveFontSizeMultiplier;
|
|
paragraphStyle.minimumLineHeight = lineHeight;
|
|
paragraphStyle.maximumLineHeight = lineHeight;
|
|
isParagraphStyleUsed = YES;
|
|
}
|
|
|
|
if (isParagraphStyleUsed) {
|
|
attributes[NSParagraphStyleAttributeName] = paragraphStyle;
|
|
}
|
|
|
|
// Decoration
|
|
BOOL isTextDecorationEnabled = NO;
|
|
if (_textDecorationLine == RCTTextDecorationLineTypeUnderline ||
|
|
_textDecorationLine == RCTTextDecorationLineTypeUnderlineStrikethrough) {
|
|
isTextDecorationEnabled = YES;
|
|
attributes[NSUnderlineStyleAttributeName] = @(_textDecorationStyle);
|
|
}
|
|
|
|
if (_textDecorationLine == RCTTextDecorationLineTypeStrikethrough ||
|
|
_textDecorationLine == RCTTextDecorationLineTypeUnderlineStrikethrough){
|
|
isTextDecorationEnabled = YES;
|
|
attributes[NSStrikethroughStyleAttributeName] = @(_textDecorationStyle);
|
|
}
|
|
|
|
if (_textDecorationColor || isTextDecorationEnabled) {
|
|
attributes[NSStrikethroughColorAttributeName] = _textDecorationColor ?: effectiveForegroundColor;
|
|
attributes[NSUnderlineColorAttributeName] = _textDecorationColor ?: effectiveForegroundColor;
|
|
}
|
|
|
|
// Shadow
|
|
if (!CGSizeEqualToSize(_textShadowOffset, CGSizeZero)) {
|
|
NSShadow *shadow = [NSShadow new];
|
|
shadow.shadowOffset = _textShadowOffset;
|
|
shadow.shadowBlurRadius = _textShadowRadius;
|
|
shadow.shadowColor = _textShadowColor;
|
|
attributes[NSShadowAttributeName] = shadow;
|
|
}
|
|
|
|
// Special
|
|
if (_isHighlighted) {
|
|
attributes[RCTTextAttributesIsHighlightedAttributeName] = @YES;
|
|
}
|
|
|
|
if (_tag) {
|
|
attributes[RCTTextAttributesTagAttributeName] = _tag;
|
|
}
|
|
|
|
return [attributes copy];
|
|
}
|
|
|
|
- (UIFont *)effectiveFont
|
|
{
|
|
// FIXME: RCTFont has thread-safety issues and must be rewritten.
|
|
return [RCTFont updateFont:nil
|
|
withFamily:_fontFamily
|
|
size:@(isnan(_fontSize) ? 0 : _fontSize)
|
|
weight:_fontWeight
|
|
style:_fontStyle
|
|
variant:_fontVariant
|
|
scaleMultiplier:self.effectiveFontSizeMultiplier];
|
|
}
|
|
|
|
- (CGFloat)effectiveFontSizeMultiplier
|
|
{
|
|
return _allowFontScaling && !isnan(_fontSizeMultiplier) ? _fontSizeMultiplier : 1.0;
|
|
}
|
|
|
|
- (UIColor *)effectiveForegroundColor
|
|
{
|
|
UIColor *effectiveForegroundColor = _foregroundColor ?: [UIColor blackColor];
|
|
|
|
if (!isnan(_opacity)) {
|
|
effectiveForegroundColor = [effectiveForegroundColor colorWithAlphaComponent:CGColorGetAlpha(effectiveForegroundColor.CGColor) * _opacity];
|
|
}
|
|
|
|
return effectiveForegroundColor;
|
|
}
|
|
|
|
- (UIColor *)effectiveBackgroundColor
|
|
{
|
|
UIColor *effectiveBackgroundColor = _backgroundColor;// ?: [[UIColor whiteColor] colorWithAlphaComponent:0];
|
|
|
|
if (effectiveBackgroundColor && !isnan(_opacity)) {
|
|
effectiveBackgroundColor = [effectiveBackgroundColor colorWithAlphaComponent:CGColorGetAlpha(effectiveBackgroundColor.CGColor) * _opacity];
|
|
}
|
|
|
|
return effectiveBackgroundColor ?: [UIColor clearColor];
|
|
}
|
|
|
|
- (RCTTextAttributes *)copyWithZone:(NSZone *)zone
|
|
{
|
|
RCTTextAttributes *textAttributes = [RCTTextAttributes new];
|
|
[textAttributes applyTextAttributes:self];
|
|
return textAttributes;
|
|
}
|
|
|
|
#pragma mark - NSObject
|
|
|
|
- (BOOL)isEqual:(RCTTextAttributes *)textAttributes
|
|
{
|
|
if (self == textAttributes) {
|
|
return YES;
|
|
}
|
|
|
|
#define RCTTextAttributesCompareFloats(a) ((a == textAttributes->a) || (isnan(a) && isnan(textAttributes->a)))
|
|
#define RCTTextAttributesCompareSize(a) CGSizeEqualToSize(a, textAttributes->a)
|
|
#define RCTTextAttributesCompareObjects(a) ((a == textAttributes->a) || [a isEqual:textAttributes->a])
|
|
#define RCTTextAttributesCompareStrings(a) ((a == textAttributes->a) || [a isEqualToString:textAttributes->a])
|
|
#define RCTTextAttributesCompareOthers(a) (a == textAttributes->a)
|
|
|
|
return
|
|
RCTTextAttributesCompareObjects(_foregroundColor) &&
|
|
RCTTextAttributesCompareObjects(_backgroundColor) &&
|
|
RCTTextAttributesCompareFloats(_opacity) &&
|
|
// Font
|
|
RCTTextAttributesCompareObjects(_fontFamily) &&
|
|
RCTTextAttributesCompareFloats(_fontSize) &&
|
|
RCTTextAttributesCompareFloats(_fontSizeMultiplier) &&
|
|
RCTTextAttributesCompareStrings(_fontWeight) &&
|
|
RCTTextAttributesCompareObjects(_fontStyle) &&
|
|
RCTTextAttributesCompareObjects(_fontVariant) &&
|
|
RCTTextAttributesCompareOthers(_allowFontScaling) &&
|
|
RCTTextAttributesCompareFloats(_letterSpacing) &&
|
|
// Paragraph Styles
|
|
RCTTextAttributesCompareFloats(_lineHeight) &&
|
|
RCTTextAttributesCompareFloats(_alignment) &&
|
|
RCTTextAttributesCompareOthers(_baseWritingDirection) &&
|
|
// Decoration
|
|
RCTTextAttributesCompareObjects(_textDecorationColor) &&
|
|
RCTTextAttributesCompareOthers(_textDecorationStyle) &&
|
|
RCTTextAttributesCompareOthers(_textDecorationLine) &&
|
|
// Shadow
|
|
RCTTextAttributesCompareSize(_textShadowOffset) &&
|
|
RCTTextAttributesCompareFloats(_textShadowRadius) &&
|
|
RCTTextAttributesCompareObjects(_textShadowColor) &&
|
|
// Special
|
|
RCTTextAttributesCompareOthers(_isHighlighted) &&
|
|
RCTTextAttributesCompareObjects(_tag) &&
|
|
RCTTextAttributesCompareOthers(_layoutDirection);
|
|
}
|
|
|
|
@end
|