2015-03-23 13:28:42 -07:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
#import "RCTTextManager.h"
|
|
|
|
|
2016-04-12 07:01:31 -07:00
|
|
|
#import "Layout.h"
|
2015-07-31 07:37:12 -07:00
|
|
|
#import "RCTAccessibilityManager.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
#import "RCTAssert.h"
|
|
|
|
#import "RCTConvert.h"
|
|
|
|
#import "RCTLog.h"
|
|
|
|
#import "RCTShadowRawText.h"
|
|
|
|
#import "RCTShadowText.h"
|
|
|
|
#import "RCTText.h"
|
2015-11-04 04:04:37 -08:00
|
|
|
#import "RCTTextView.h"
|
2015-03-26 02:58:06 -07:00
|
|
|
#import "UIView+React.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2016-05-17 10:31:07 -07:00
|
|
|
static void collectDirtyNonTextDescendants(RCTShadowText *shadowView, NSMutableArray *nonTextDescendants) {
|
|
|
|
for (RCTShadowView *child in shadowView.reactSubviews) {
|
|
|
|
if ([child isKindOfClass:[RCTShadowText class]]) {
|
|
|
|
collectDirtyNonTextDescendants((RCTShadowText *)child, nonTextDescendants);
|
|
|
|
} else if ([child isKindOfClass:[RCTShadowRawText class]]) {
|
|
|
|
// no-op
|
|
|
|
} else if ([child isTextDirty]) {
|
|
|
|
[nonTextDescendants addObject:child];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-04 04:04:37 -08:00
|
|
|
@interface RCTShadowText (Private)
|
|
|
|
|
2016-04-12 07:01:31 -07:00
|
|
|
- (NSTextStorage *)buildTextStorageForWidth:(CGFloat)width widthMode:(css_measure_mode_t)widthMode;
|
2015-11-04 04:04:37 -08:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
@implementation RCTTextManager
|
|
|
|
|
2015-04-08 05:42:43 -07:00
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
- (UIView *)view
|
|
|
|
{
|
2015-08-17 07:35:34 -07:00
|
|
|
return [RCTText new];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
- (RCTShadowView *)shadowView
|
|
|
|
{
|
2015-08-17 07:35:34 -07:00
|
|
|
return [RCTShadowText new];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-03-17 07:04:39 -07:00
|
|
|
#pragma mark - Shadow properties
|
|
|
|
|
2015-03-25 21:29:28 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(color, UIColor)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(fontFamily, NSString)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(fontSize, CGFloat)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(fontWeight, NSString)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(fontStyle, NSString)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(isHighlighted, BOOL)
|
2015-05-12 00:25:08 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(letterSpacing, CGFloat)
|
2015-03-25 21:29:28 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(lineHeight, CGFloat)
|
2015-07-07 06:03:56 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(numberOfLines, NSUInteger)
|
2015-03-25 21:29:28 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textAlign, NSTextAlignment)
|
2015-07-07 06:03:56 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textDecorationStyle, NSUnderlineStyle)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textDecorationColor, UIColor)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textDecorationLine, RCTTextDecorationLineType)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(writingDirection, NSWritingDirection)
|
2015-07-31 07:37:12 -07:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(allowFontScaling, BOOL)
|
2015-11-04 08:52:46 -08:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(opacity, CGFloat)
|
2016-01-01 09:32:59 -08:00
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textShadowOffset, CGSize)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textShadowRadius, CGFloat)
|
|
|
|
RCT_EXPORT_SHADOW_PROPERTY(textShadowColor, UIColor)
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2015-11-14 10:25:00 -08:00
|
|
|
- (RCTViewManagerUIBlock)uiBlockToAmendWithShadowViewRegistry:(NSDictionary<NSNumber *, RCTShadowView *> *)shadowViewRegistry
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-11-04 04:04:37 -08:00
|
|
|
NSMutableSet *textViewTagsToUpdate = [NSMutableSet new];
|
2015-11-14 10:25:00 -08:00
|
|
|
for (RCTShadowView *rootView in shadowViewRegistry.allValues) {
|
2015-02-19 20:10:52 -08:00
|
|
|
if (![rootView isReactRootView]) {
|
|
|
|
// This isn't a root view
|
|
|
|
continue;
|
|
|
|
}
|
2015-03-23 13:28:42 -07:00
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
if (![rootView isTextDirty]) {
|
|
|
|
// No text processing to be done
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-11-03 14:45:46 -08:00
|
|
|
NSMutableArray<RCTShadowView *> *queue = [NSMutableArray arrayWithObject:rootView];
|
2015-08-24 09:14:33 -01:00
|
|
|
for (NSInteger i = 0; i < queue.count; i++) {
|
2015-02-19 20:10:52 -08:00
|
|
|
RCTShadowView *shadowView = queue[i];
|
|
|
|
RCTAssert([shadowView isTextDirty], @"Don't process any nodes that don't have dirty text");
|
|
|
|
|
|
|
|
if ([shadowView isKindOfClass:[RCTShadowText class]]) {
|
2015-08-24 09:14:33 -01:00
|
|
|
((RCTShadowText *)shadowView).fontSizeMultiplier = self.bridge.accessibilityManager.multiplier;
|
2015-05-29 10:27:14 -07:00
|
|
|
[(RCTShadowText *)shadowView recomputeText];
|
2016-05-17 10:31:07 -07:00
|
|
|
collectDirtyNonTextDescendants((RCTShadowText *)shadowView, queue);
|
2015-02-19 20:10:52 -08:00
|
|
|
} else if ([shadowView isKindOfClass:[RCTShadowRawText class]]) {
|
2015-05-26 18:39:37 -07:00
|
|
|
RCTLogError(@"Raw text cannot be used outside of a <Text> tag. Not rendering string: '%@'",
|
|
|
|
[(RCTShadowRawText *)shadowView text]);
|
2015-02-19 20:10:52 -08:00
|
|
|
} else {
|
2015-11-04 04:04:37 -08:00
|
|
|
NSNumber *reactTag = shadowView.reactTag;
|
|
|
|
// This isn't pretty, but hopefully it's temporary
|
|
|
|
// the problem is, there's no easy way (besides the viewName)
|
|
|
|
// to tell from the shadowView if the view is an RKTextView
|
|
|
|
if ([shadowView.viewName hasSuffix:@"TextView"]) {
|
|
|
|
// Add to textViewTagsToUpdate only if has a RCTShadowText subview
|
|
|
|
for (RCTShadowView *subview in shadowView.reactSubviews) {
|
|
|
|
if ([subview isKindOfClass:[RCTShadowText class]]) {
|
|
|
|
[textViewTagsToUpdate addObject:reactTag];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
for (RCTShadowView *child in [shadowView reactSubviews]) {
|
|
|
|
if ([child isTextDirty]) {
|
|
|
|
[queue addObject:child];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[shadowView setTextComputed];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-04 04:04:37 -08:00
|
|
|
/**
|
|
|
|
* NOTE: this logic is included to support rich text editing inside multiline
|
2016-04-21 03:56:41 -07:00
|
|
|
* `<TextInput>` controls. It is required in order to ensure that the
|
|
|
|
* textStorage (aka attributed string) is copied over from the RCTShadowText
|
|
|
|
* to the RCTText view in time to be used to update the editable text content.
|
2015-11-04 04:04:37 -08:00
|
|
|
*/
|
|
|
|
if (textViewTagsToUpdate.count) {
|
|
|
|
|
|
|
|
NSMutableArray<RCTViewManagerUIBlock> *uiBlocks = [NSMutableArray new];
|
|
|
|
for (NSNumber *reactTag in textViewTagsToUpdate) {
|
|
|
|
RCTShadowView *shadowTextView = shadowViewRegistry[reactTag];
|
|
|
|
RCTShadowText *shadowText;
|
|
|
|
for (RCTShadowText *subview in shadowTextView.reactSubviews) {
|
|
|
|
if ([subview isKindOfClass:[RCTShadowText class]]) {
|
|
|
|
shadowText = subview;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
UIEdgeInsets padding = shadowText.paddingAsInsets;
|
|
|
|
CGFloat width = shadowText.frame.size.width - (padding.left + padding.right);
|
2016-04-12 07:01:31 -07:00
|
|
|
|
2016-04-21 03:56:41 -07:00
|
|
|
NSTextStorage *textStorage = [shadowText buildTextStorageForWidth:width widthMode:CSS_MEASURE_MODE_EXACTLY];
|
2015-11-14 10:25:00 -08:00
|
|
|
[uiBlocks addObject:^(RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTTextView *> *viewRegistry) {
|
2015-11-04 04:04:37 -08:00
|
|
|
RCTTextView *textView = viewRegistry[reactTag];
|
|
|
|
RCTText *text;
|
|
|
|
for (RCTText *subview in textView.reactSubviews) {
|
|
|
|
if ([subview isKindOfClass:[RCTText class]]) {
|
|
|
|
text = subview;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
text.textStorage = textStorage;
|
|
|
|
[textView performTextUpdate];
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-11-14 10:25:00 -08:00
|
|
|
return ^(RCTUIManager *uiManager, NSDictionary<NSNumber *, UIView *> *viewRegistry) {
|
2015-11-04 04:04:37 -08:00
|
|
|
for (RCTViewManagerUIBlock uiBlock in uiBlocks) {
|
|
|
|
uiBlock(uiManager, viewRegistry);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
return nil;
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
2015-04-07 02:19:49 -07:00
|
|
|
- (RCTViewManagerUIBlock)uiBlockToAmendWithShadowView:(RCTShadowText *)shadowView
|
2015-03-27 09:36:33 -07:00
|
|
|
{
|
|
|
|
NSNumber *reactTag = shadowView.reactTag;
|
|
|
|
UIEdgeInsets padding = shadowView.paddingAsInsets;
|
2015-04-07 02:19:49 -07:00
|
|
|
|
2015-11-14 10:25:00 -08:00
|
|
|
return ^(RCTUIManager *uiManager, NSDictionary<NSNumber *, RCTText *> *viewRegistry) {
|
2015-04-29 01:29:00 -07:00
|
|
|
RCTText *text = viewRegistry[reactTag];
|
2015-04-07 02:19:49 -07:00
|
|
|
text.contentInset = padding;
|
2015-03-27 09:36:33 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
@end
|