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 "RCTText.h"
|
|
|
|
|
|
|
|
#import "RCTShadowText.h"
|
|
|
|
#import "RCTUtils.h"
|
2015-03-26 02:58:06 -07:00
|
|
|
#import "UIView+React.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
@implementation RCTText
|
|
|
|
{
|
|
|
|
NSLayoutManager *_layoutManager;
|
|
|
|
NSTextStorage *_textStorage;
|
|
|
|
NSTextContainer *_textContainer;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)initWithFrame:(CGRect)frame
|
|
|
|
{
|
|
|
|
if ((self = [super initWithFrame:frame])) {
|
|
|
|
_textStorage = [[NSTextStorage alloc] init];
|
|
|
|
|
2015-04-21 19:13:40 -07:00
|
|
|
self.opaque = NO;
|
2015-02-19 20:10:52 -08:00
|
|
|
self.contentMode = UIViewContentModeRedraw;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSAttributedString *)attributedText
|
|
|
|
{
|
|
|
|
return [_textStorage copy];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)setAttributedText:(NSAttributedString *)attributedText
|
|
|
|
{
|
2015-04-09 07:40:18 -07:00
|
|
|
for (NSLayoutManager *existingLayoutManager in _textStorage.layoutManagers) {
|
|
|
|
[_textStorage removeLayoutManager:existingLayoutManager];
|
|
|
|
}
|
|
|
|
|
|
|
|
_textStorage = [[NSTextStorage alloc] initWithAttributedString:attributedText];
|
|
|
|
|
|
|
|
if (_layoutManager) {
|
|
|
|
[_textStorage addLayoutManager:_layoutManager];
|
|
|
|
}
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
[self setNeedsDisplay];
|
|
|
|
}
|
|
|
|
|
2015-04-07 02:19:49 -07:00
|
|
|
- (void)setTextContainer:(NSTextContainer *)textContainer
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-04-09 07:40:18 -07:00
|
|
|
if ([_textContainer isEqual:textContainer]) {
|
|
|
|
return;
|
|
|
|
}
|
2015-04-07 02:19:49 -07:00
|
|
|
|
|
|
|
_textContainer = textContainer;
|
|
|
|
|
|
|
|
for (NSInteger i = _layoutManager.textContainers.count - 1; i >= 0; i--) {
|
|
|
|
[_layoutManager removeTextContainerAtIndex:i];
|
|
|
|
}
|
2015-04-09 07:40:18 -07:00
|
|
|
|
|
|
|
if (_textContainer) {
|
|
|
|
[_layoutManager addTextContainer:_textContainer];
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
[self setNeedsDisplay];
|
|
|
|
}
|
|
|
|
|
2015-04-07 02:19:49 -07:00
|
|
|
- (void)setLayoutManager:(NSLayoutManager *)layoutManager
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-04-09 07:40:18 -07:00
|
|
|
if ([_layoutManager isEqual:layoutManager]) {
|
|
|
|
return;
|
|
|
|
}
|
2015-04-07 02:19:49 -07:00
|
|
|
|
|
|
|
_layoutManager = layoutManager;
|
|
|
|
|
2015-04-07 13:43:08 -07:00
|
|
|
for (NSLayoutManager *existingLayoutManager in _textStorage.layoutManagers) {
|
|
|
|
[_textStorage removeLayoutManager:existingLayoutManager];
|
2015-04-07 02:19:49 -07:00
|
|
|
}
|
2015-04-09 07:40:18 -07:00
|
|
|
|
|
|
|
if (_layoutManager) {
|
|
|
|
[_textStorage addLayoutManager:_layoutManager];
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
[self setNeedsDisplay];
|
|
|
|
}
|
|
|
|
|
2015-03-27 09:36:33 -07:00
|
|
|
- (CGRect)textFrame
|
|
|
|
{
|
|
|
|
return UIEdgeInsetsInsetRect(self.bounds, _contentInset);
|
|
|
|
}
|
|
|
|
|
2015-04-23 16:00:34 -07:00
|
|
|
- (void)drawRect:(CGRect)rect
|
2015-02-19 20:10:52 -08:00
|
|
|
{
|
2015-04-23 16:00:34 -07:00
|
|
|
CGRect textFrame = [self textFrame];
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2015-04-23 16:00:34 -07:00
|
|
|
// We reset the text container size every time because RCTShadowText's
|
|
|
|
// RCTMeasure overrides it. The header comment for `size` says that a height
|
|
|
|
// of 0.0 should be enough, but it isn't.
|
|
|
|
_textContainer.size = CGSizeMake(textFrame.size.width, CGFLOAT_MAX);
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
NSRange glyphRange = [_layoutManager glyphRangeForTextContainer:_textContainer];
|
2015-04-23 16:00:34 -07:00
|
|
|
[_layoutManager drawBackgroundForGlyphRange:glyphRange atPoint:textFrame.origin];
|
|
|
|
[_layoutManager drawGlyphsForGlyphRange:glyphRange atPoint:textFrame.origin];
|
2015-02-19 20:10:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
- (NSNumber *)reactTagAtPoint:(CGPoint)point
|
|
|
|
{
|
|
|
|
CGFloat fraction;
|
2015-04-14 17:51:28 -07:00
|
|
|
NSUInteger characterIndex = [_layoutManager characterIndexForPoint:point
|
|
|
|
inTextContainer:_textContainer
|
|
|
|
fractionOfDistanceBetweenInsertionPoints:&fraction];
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
NSNumber *reactTag = nil;
|
|
|
|
|
|
|
|
// If the point is not before (fraction == 0.0) the first character and not
|
|
|
|
// after (fraction == 1.0) the last character, then the attribute is valid.
|
|
|
|
if (_textStorage.length > 0 && (fraction > 0 || characterIndex > 0) && (fraction < 1 || characterIndex < _textStorage.length - 1)) {
|
|
|
|
reactTag = [_textStorage attribute:RCTReactTagAttributeName atIndex:characterIndex effectiveRange:NULL];
|
|
|
|
}
|
|
|
|
|
|
|
|
return reactTag ?: self.reactTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|