mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00

git-subtree-dir: submodules/AsyncDisplayKit git-subtree-mainline: d06f423e0ed3df1fed9bd10d79ee312a9179b632 git-subtree-split: 02bedc12816e251ad71777f9d2578329b6d2bef6
1334 lines
45 KiB
Plaintext
1334 lines
45 KiB
Plaintext
//
|
|
// ASTextNode2.mm
|
|
// Texture
|
|
//
|
|
// Copyright (c) Pinterest, Inc. All rights reserved.
|
|
// Licensed under Apache 2.0: http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
|
|
#import <AsyncDisplayKit/ASTextNode2.h>
|
|
#import <AsyncDisplayKit/ASTextNode.h> // Definition of ASTextNodeDelegate
|
|
|
|
#import <tgmath.h>
|
|
#import <deque>
|
|
|
|
#import <AsyncDisplayKit/_ASDisplayLayer.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+FrameworkPrivate.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+Subclasses.h>
|
|
#import <AsyncDisplayKit/ASDisplayNodeExtras.h>
|
|
#import <AsyncDisplayKit/ASDisplayNodeInternal.h>
|
|
#import <AsyncDisplayKit/ASHighlightOverlayLayer.h>
|
|
|
|
#import <AsyncDisplayKit/ASTextKitRenderer+Positioning.h>
|
|
#import <AsyncDisplayKit/ASTextKitShadower.h>
|
|
#import <AsyncDisplayKit/ASEqualityHelpers.h>
|
|
|
|
#import <AsyncDisplayKit/ASInternalHelpers.h>
|
|
|
|
#import <AsyncDisplayKit/CoreGraphics+ASConvenience.h>
|
|
#import <AsyncDisplayKit/ASObjectDescriptionHelpers.h>
|
|
#import <AsyncDisplayKit/ASTextLayout.h>
|
|
#import <AsyncDisplayKit/ASThread.h>
|
|
|
|
@interface ASTextCacheValue : NSObject {
|
|
@package
|
|
AS::Mutex _m;
|
|
std::deque<std::tuple<CGSize, ASTextLayout *>> _layouts;
|
|
}
|
|
@end
|
|
@implementation ASTextCacheValue
|
|
@end
|
|
|
|
/**
|
|
* If set, we will record all values set to attributedText into an array
|
|
* and once we get 2000, we'll write them all out into a plist file.
|
|
*
|
|
* This is useful for gathering realistic text data sets from apps for performance
|
|
* testing.
|
|
*/
|
|
#define AS_TEXTNODE2_RECORD_ATTRIBUTED_STRINGS 0
|
|
|
|
/**
|
|
* If it can't find a compatible layout, this method creates one.
|
|
*
|
|
* NOTE: Be careful to copy `text` if needed.
|
|
*/
|
|
static NS_RETURNS_RETAINED ASTextLayout *ASTextNodeCompatibleLayoutWithContainerAndText(ASTextContainer *container, NSAttributedString *text) {
|
|
static dispatch_once_t onceToken;
|
|
static AS::Mutex *layoutCacheLock;
|
|
static NSCache<NSAttributedString *, ASTextCacheValue *> *textLayoutCache;
|
|
dispatch_once(&onceToken, ^{
|
|
layoutCacheLock = new AS::Mutex();
|
|
textLayoutCache = [[NSCache alloc] init];
|
|
});
|
|
|
|
layoutCacheLock->lock();
|
|
|
|
ASTextCacheValue *cacheValue = [textLayoutCache objectForKey:text];
|
|
if (cacheValue == nil) {
|
|
cacheValue = [[ASTextCacheValue alloc] init];
|
|
[textLayoutCache setObject:cacheValue forKey:[text copy]];
|
|
}
|
|
|
|
// Lock the cache item for the rest of the method. Only after acquiring can we release the NSCache.
|
|
AS::MutexLocker lock(cacheValue->_m);
|
|
layoutCacheLock->unlock();
|
|
|
|
CGRect containerBounds = (CGRect){ .size = container.size };
|
|
{
|
|
for (const auto &t : cacheValue->_layouts) {
|
|
CGSize constrainedSize = std::get<0>(t);
|
|
ASTextLayout *layout = std::get<1>(t);
|
|
|
|
CGSize layoutSize = layout.textBoundingSize;
|
|
// 1. CoreText can return frames that are narrower than the constrained width, for obvious reasons.
|
|
// 2. CoreText can return frames that are slightly wider than the constrained width, for some reason.
|
|
// We have to trust that somehow it's OK to try and draw within our size constraint, despite the return value.
|
|
// 3. Thus, those two values (constrained width & returned width) form a range, where
|
|
// intermediate values in that range will be snapped. Thus, we can use a given layout as long as our
|
|
// width is in that range, between the min and max of those two values.
|
|
CGRect minRect = CGRectMake(0, 0, MIN(layoutSize.width, constrainedSize.width), MIN(layoutSize.height, constrainedSize.height));
|
|
if (!CGRectContainsRect(containerBounds, minRect)) {
|
|
continue;
|
|
}
|
|
CGRect maxRect = CGRectMake(0, 0, MAX(layoutSize.width, constrainedSize.width), MAX(layoutSize.height, constrainedSize.height));
|
|
if (!CGRectContainsRect(maxRect, containerBounds)) {
|
|
continue;
|
|
}
|
|
if (!CGSizeEqualToSize(container.size, constrainedSize)) {
|
|
continue;
|
|
}
|
|
|
|
// Now check container params.
|
|
ASTextContainer *otherContainer = layout.container;
|
|
if (!UIEdgeInsetsEqualToEdgeInsets(container.insets, otherContainer.insets)) {
|
|
continue;
|
|
}
|
|
if (!ASObjectIsEqual(container.exclusionPaths, otherContainer.exclusionPaths)) {
|
|
continue;
|
|
}
|
|
if (container.maximumNumberOfRows != otherContainer.maximumNumberOfRows) {
|
|
continue;
|
|
}
|
|
if (container.truncationType != otherContainer.truncationType) {
|
|
continue;
|
|
}
|
|
if (!ASObjectIsEqual(container.truncationToken, otherContainer.truncationToken)) {
|
|
continue;
|
|
}
|
|
// TODO: When we get a cache hit, move this entry to the front (LRU).
|
|
return layout;
|
|
}
|
|
}
|
|
|
|
// Cache Miss. Compute the text layout.
|
|
ASTextLayout *layout = [ASTextLayout layoutWithContainer:container text:text];
|
|
|
|
// Store the result in the cache.
|
|
{
|
|
// This is a critical section. However we also must hold the lock until this point, in case
|
|
// another thread requests this cache item while a layout is being calculated, so they don't race.
|
|
cacheValue->_layouts.push_front(std::make_tuple(container.size, layout));
|
|
if (cacheValue->_layouts.size() > 3) {
|
|
cacheValue->_layouts.pop_back();
|
|
}
|
|
}
|
|
|
|
return layout;
|
|
}
|
|
|
|
static const CGFloat ASTextNodeHighlightLightOpacity = 0.11;
|
|
static const CGFloat ASTextNodeHighlightDarkOpacity = 0.22;
|
|
static NSString *ASTextNodeTruncationTokenAttributeName = @"ASTextNodeTruncationAttribute";
|
|
|
|
#if AS_ENABLE_TEXTNODE
|
|
@interface ASTextNode2 () <UIGestureRecognizerDelegate>
|
|
#else
|
|
@interface ASTextNode () <UIGestureRecognizerDelegate>
|
|
#endif
|
|
|
|
@end
|
|
|
|
#if AS_ENABLE_TEXTNODE
|
|
@implementation ASTextNode2 {
|
|
#else
|
|
@implementation ASTextNode {
|
|
#endif
|
|
ASTextContainer *_textContainer;
|
|
|
|
CGSize _shadowOffset;
|
|
CGColorRef _shadowColor;
|
|
CGFloat _shadowOpacity;
|
|
CGFloat _shadowRadius;
|
|
|
|
NSAttributedString *_attributedText;
|
|
NSAttributedString *_truncationAttributedText;
|
|
NSAttributedString *_additionalTruncationMessage;
|
|
NSAttributedString *_composedTruncationText;
|
|
NSArray<NSNumber *> *_pointSizeScaleFactors;
|
|
NSLineBreakMode _truncationMode;
|
|
|
|
NSString *_highlightedLinkAttributeName;
|
|
id _highlightedLinkAttributeValue;
|
|
ASTextNodeHighlightStyle _highlightStyle;
|
|
NSRange _highlightRange;
|
|
ASHighlightOverlayLayer *_activeHighlightLayer;
|
|
UIColor *_placeholderColor;
|
|
|
|
UILongPressGestureRecognizer *_longPressGestureRecognizer;
|
|
}
|
|
@dynamic placeholderEnabled;
|
|
|
|
static NSArray *DefaultLinkAttributeNames() {
|
|
static NSArray *names;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
names = @[ NSLinkAttributeName ];
|
|
});
|
|
return names;
|
|
}
|
|
|
|
- (instancetype)init
|
|
{
|
|
if (self = [super init]) {
|
|
_textContainer = [[ASTextContainer alloc] init];
|
|
// Load default values from superclass.
|
|
_shadowOffset = [super shadowOffset];
|
|
_shadowColor = CGColorRetain([super shadowColor]);
|
|
_shadowOpacity = [super shadowOpacity];
|
|
_shadowRadius = [super shadowRadius];
|
|
|
|
// Disable user interaction for text node by default.
|
|
self.userInteractionEnabled = NO;
|
|
self.needsDisplayOnBoundsChange = YES;
|
|
|
|
_textContainer.truncationType = ASTextTruncationTypeEnd;
|
|
|
|
// The common case is for a text node to be non-opaque and blended over some background.
|
|
self.opaque = NO;
|
|
self.backgroundColor = [UIColor clearColor];
|
|
|
|
self.linkAttributeNames = DefaultLinkAttributeNames();
|
|
|
|
// Accessibility
|
|
self.isAccessibilityElement = YES;
|
|
self.accessibilityTraits = self.defaultAccessibilityTraits;
|
|
|
|
// Placeholders
|
|
// Disabled by default in ASDisplayNode, but add a few options for those who toggle
|
|
// on the special placeholder behavior of ASTextNode.
|
|
_placeholderColor = ASDisplayNodeDefaultPlaceholderColor();
|
|
_placeholderInsets = UIEdgeInsetsMake(1.0, 0.0, 1.0, 0.0);
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
CGColorRelease(_shadowColor);
|
|
|
|
if (_longPressGestureRecognizer) {
|
|
_longPressGestureRecognizer.delegate = nil;
|
|
[_longPressGestureRecognizer removeTarget:nil action:NULL];
|
|
[self.view removeGestureRecognizer:_longPressGestureRecognizer];
|
|
}
|
|
}
|
|
|
|
#pragma mark - Description
|
|
|
|
- (NSString *)_plainStringForDescription
|
|
{
|
|
NSString *plainString = [[self.attributedText string] stringByTrimmingCharactersInSet:[NSCharacterSet newlineCharacterSet]];
|
|
if (plainString.length > 50) {
|
|
plainString = [[plainString substringToIndex:50] stringByAppendingString:@"…"];
|
|
}
|
|
return plainString;
|
|
}
|
|
|
|
- (NSMutableArray<NSDictionary *> *)propertiesForDescription
|
|
{
|
|
NSMutableArray *result = [super propertiesForDescription];
|
|
NSString *plainString = [self _plainStringForDescription];
|
|
if (plainString.length > 0) {
|
|
[result insertObject:@{ @"text" : ASStringWithQuotesIfMultiword(plainString) } atIndex:0];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
- (NSMutableArray<NSDictionary *> *)propertiesForDebugDescription
|
|
{
|
|
NSMutableArray *result = [super propertiesForDebugDescription];
|
|
NSString *plainString = [self _plainStringForDescription];
|
|
if (plainString.length > 0) {
|
|
[result insertObject:@{ @"text" : ASStringWithQuotesIfMultiword(plainString) } atIndex:0];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#pragma mark - ASDisplayNode
|
|
|
|
- (void)didLoad
|
|
{
|
|
[super didLoad];
|
|
|
|
// If we are view-backed and the delegate cares, support the long-press callback.
|
|
// Locking is not needed, as all instance variables used are main-thread-only.
|
|
SEL longPressCallback = @selector(textNode:longPressedLinkAttribute:value:atPoint:textRange:);
|
|
if (!self.isLayerBacked && [self.delegate respondsToSelector:longPressCallback]) {
|
|
_longPressGestureRecognizer = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(_handleLongPress:)];
|
|
_longPressGestureRecognizer.cancelsTouchesInView = self.longPressCancelsTouches;
|
|
_longPressGestureRecognizer.delegate = self;
|
|
[self.view addGestureRecognizer:_longPressGestureRecognizer];
|
|
}
|
|
}
|
|
|
|
- (BOOL)supportsLayerBacking
|
|
{
|
|
if (!super.supportsLayerBacking) {
|
|
return NO;
|
|
}
|
|
|
|
ASLockScopeSelf();
|
|
// If the text contains any links, return NO.
|
|
NSAttributedString *attributedText = _attributedText;
|
|
NSRange range = NSMakeRange(0, attributedText.length);
|
|
for (NSString *linkAttributeName in _linkAttributeNames) {
|
|
__block BOOL hasLink = NO;
|
|
[attributedText enumerateAttribute:linkAttributeName inRange:range options:NSAttributedStringEnumerationLongestEffectiveRangeNotRequired usingBlock:^(id _Nullable value, NSRange range, BOOL * _Nonnull stop) {
|
|
hasLink = (value != nil);
|
|
*stop = YES;
|
|
}];
|
|
if (hasLink) {
|
|
return NO;
|
|
}
|
|
}
|
|
return YES;
|
|
}
|
|
|
|
- (NSString *)defaultAccessibilityLabel
|
|
{
|
|
ASLockScopeSelf();
|
|
return _attributedText.string;
|
|
}
|
|
|
|
- (UIAccessibilityTraits)defaultAccessibilityTraits
|
|
{
|
|
return UIAccessibilityTraitStaticText;
|
|
}
|
|
|
|
#pragma mark - Layout and Sizing
|
|
|
|
- (void)setTextContainerInset:(UIEdgeInsets)textContainerInset
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCustom(_textContainer.insets, textContainerInset, UIEdgeInsetsEqualToEdgeInsets)) {
|
|
[self setNeedsLayout];
|
|
}
|
|
}
|
|
|
|
- (UIEdgeInsets)textContainerInset
|
|
{
|
|
// textContainer is invariant and has an atomic accessor.
|
|
return _textContainer.insets;
|
|
}
|
|
|
|
- (void)setTextContainerLinePositionModifier:(id<ASTextLinePositionModifier>)modifier
|
|
{
|
|
ASLockedSelfCompareAssignObjects(_textContainer.linePositionModifier, modifier);
|
|
}
|
|
|
|
- (id<ASTextLinePositionModifier>)textContainerLinePositionModifier
|
|
{
|
|
ASLockScopeSelf();
|
|
return _textContainer.linePositionModifier;
|
|
}
|
|
|
|
- (CGSize)calculateSizeThatFits:(CGSize)constrainedSize
|
|
{
|
|
ASDisplayNodeAssert(constrainedSize.width >= 0, @"Constrained width for text (%f) is too narrow", constrainedSize.width);
|
|
ASDisplayNodeAssert(constrainedSize.height >= 0, @"Constrained height for text (%f) is too short", constrainedSize.height);
|
|
|
|
ASLockScopeSelf();
|
|
|
|
_textContainer.size = constrainedSize;
|
|
[self _ensureTruncationText];
|
|
|
|
// If the constrained size has a max/inf value on the text's forward direction, the text node is calculating its intrinsic size.
|
|
// Need to consider both width and height when determining if it is calculating instrinsic size. Even the constrained width is provided, the height can be inf
|
|
// it may provide a text that is longer than the width and require a wordWrapping line break mode and looking for the height to be calculated.
|
|
BOOL isCalculatingIntrinsicSize = (_textContainer.size.width >= ASTextContainerMaxSize.width) || (_textContainer.size.height >= ASTextContainerMaxSize.height);
|
|
|
|
NSMutableAttributedString *mutableText = [_attributedText mutableCopy];
|
|
[self prepareAttributedString:mutableText isForIntrinsicSize:isCalculatingIntrinsicSize];
|
|
ASTextLayout *layout = ASTextNodeCompatibleLayoutWithContainerAndText(_textContainer, mutableText);
|
|
if (layout.truncatedLine != nil && layout.truncatedLine.size.width > layout.textBoundingSize.width) {
|
|
return (CGSize) {MIN(constrainedSize.width, layout.truncatedLine.size.width), layout.textBoundingSize.height};
|
|
}
|
|
|
|
return layout.textBoundingSize;
|
|
}
|
|
|
|
#pragma mark - Modifying User Text
|
|
|
|
// Returns the ascender of the first character in attributedString by also including the line height if specified in paragraph style.
|
|
+ (CGFloat)ascenderWithAttributedString:(NSAttributedString *)attributedString
|
|
{
|
|
UIFont *font = [attributedString attribute:NSFontAttributeName atIndex:0 effectiveRange:NULL];
|
|
NSParagraphStyle *paragraphStyle = [attributedString attribute:NSParagraphStyleAttributeName atIndex:0 effectiveRange:NULL];
|
|
if (!paragraphStyle) {
|
|
return font.ascender;
|
|
}
|
|
CGFloat lineHeight = MAX(font.lineHeight, paragraphStyle.minimumLineHeight);
|
|
if (paragraphStyle.maximumLineHeight > 0) {
|
|
lineHeight = MIN(lineHeight, paragraphStyle.maximumLineHeight);
|
|
}
|
|
return lineHeight + font.descender;
|
|
}
|
|
|
|
- (NSAttributedString *)attributedText
|
|
{
|
|
ASLockScopeSelf();
|
|
return _attributedText;
|
|
}
|
|
|
|
- (void)setAttributedText:(NSAttributedString *)attributedText
|
|
{
|
|
if (attributedText == nil) {
|
|
attributedText = [[NSAttributedString alloc] initWithString:@"" attributes:nil];
|
|
}
|
|
|
|
// Many accessors in this method will acquire the lock (including ASDisplayNode methods).
|
|
// Holding it for the duration of the method is more efficient in this case.
|
|
ASLockScopeSelf();
|
|
|
|
if (!ASCompareAssignCopy(_attributedText, attributedText)) {
|
|
return;
|
|
}
|
|
|
|
// Since truncation text matches style of attributedText, invalidate it now.
|
|
[self _locked_invalidateTruncationText];
|
|
|
|
NSUInteger length = attributedText.length;
|
|
if (length > 0) {
|
|
ASLayoutElementStyle *style = [self _locked_style];
|
|
style.ascender = [[self class] ascenderWithAttributedString:attributedText];
|
|
style.descender = [[attributedText attribute:NSFontAttributeName atIndex:attributedText.length - 1 effectiveRange:NULL] descender];
|
|
}
|
|
|
|
// Tell the display node superclasses that the cached layout is incorrect now
|
|
[self setNeedsLayout];
|
|
|
|
// Force display to create renderer with new size and redisplay with new string
|
|
[self setNeedsDisplay];
|
|
|
|
// Accessiblity
|
|
self.accessibilityLabel = self.defaultAccessibilityLabel;
|
|
self.isAccessibilityElement = (length != 0); // We're an accessibility element by default if there is a string.
|
|
|
|
#if AS_TEXTNODE2_RECORD_ATTRIBUTED_STRINGS
|
|
[ASTextNode _registerAttributedText:_attributedText];
|
|
#endif
|
|
}
|
|
|
|
#pragma mark - Text Layout
|
|
|
|
- (void)setExclusionPaths:(NSArray *)exclusionPaths
|
|
{
|
|
ASLockScopeSelf();
|
|
_textContainer.exclusionPaths = exclusionPaths;
|
|
|
|
[self setNeedsLayout];
|
|
[self setNeedsDisplay];
|
|
}
|
|
|
|
- (NSArray *)exclusionPaths
|
|
{
|
|
ASLockScopeSelf();
|
|
return _textContainer.exclusionPaths;
|
|
}
|
|
|
|
- (void)prepareAttributedString:(NSMutableAttributedString *)attributedString isForIntrinsicSize:(BOOL)isForIntrinsicSize
|
|
{
|
|
ASLockScopeSelf();
|
|
NSLineBreakMode innerMode;
|
|
switch (_truncationMode) {
|
|
case NSLineBreakByWordWrapping:
|
|
case NSLineBreakByCharWrapping:
|
|
case NSLineBreakByClipping:
|
|
innerMode = _truncationMode;
|
|
break;
|
|
default:
|
|
innerMode = NSLineBreakByWordWrapping;
|
|
}
|
|
|
|
// Apply/Fix paragraph style if needed
|
|
[attributedString enumerateAttribute:NSParagraphStyleAttributeName inRange:NSMakeRange(0, attributedString.length) options:kNilOptions usingBlock:^(NSParagraphStyle *style, NSRange range, BOOL * _Nonnull stop) {
|
|
|
|
BOOL applyTruncationMode = YES;
|
|
NSMutableParagraphStyle *paragraphStyle = nil;
|
|
// Only "left" and "justified" alignments are supported while calculating intrinsic size.
|
|
// Other alignments like "right", "center" and "natural" cause the size to be bigger than needed and thus should be ignored/overridden.
|
|
const BOOL forceLeftAlignment = (style != nil
|
|
&& isForIntrinsicSize
|
|
&& style.alignment != NSTextAlignmentLeft
|
|
&& style.alignment != NSTextAlignmentJustified);
|
|
if (style != nil) {
|
|
if (innerMode == style.lineBreakMode) {
|
|
applyTruncationMode = NO;
|
|
}
|
|
paragraphStyle = [style mutableCopy];
|
|
} else {
|
|
if (innerMode == NSLineBreakByWordWrapping) {
|
|
applyTruncationMode = NO;
|
|
}
|
|
paragraphStyle = [NSMutableParagraphStyle new];
|
|
}
|
|
if (!applyTruncationMode && !forceLeftAlignment) {
|
|
return;
|
|
}
|
|
paragraphStyle.lineBreakMode = innerMode;
|
|
|
|
if (applyTruncationMode) {
|
|
paragraphStyle.lineBreakMode = _truncationMode;
|
|
}
|
|
if (forceLeftAlignment) {
|
|
paragraphStyle.alignment = NSTextAlignmentLeft;
|
|
}
|
|
[attributedString addAttribute:NSParagraphStyleAttributeName value:paragraphStyle range:range];
|
|
}];
|
|
|
|
// Apply shadow if needed
|
|
if (_shadowOpacity > 0 && (_shadowRadius != 0 || !CGSizeEqualToSize(_shadowOffset, CGSizeZero)) && CGColorGetAlpha(_shadowColor) > 0) {
|
|
NSShadow *shadow = [[NSShadow alloc] init];
|
|
if (_shadowOpacity != 1) {
|
|
CGColorRef shadowColorRef = CGColorCreateCopyWithAlpha(_shadowColor, _shadowOpacity * CGColorGetAlpha(_shadowColor));
|
|
shadow.shadowColor = [UIColor colorWithCGColor:shadowColorRef];
|
|
CGColorRelease(shadowColorRef);
|
|
} else {
|
|
shadow.shadowColor = [UIColor colorWithCGColor:_shadowColor];
|
|
}
|
|
shadow.shadowOffset = _shadowOffset;
|
|
shadow.shadowBlurRadius = _shadowRadius;
|
|
[attributedString addAttribute:NSShadowAttributeName value:shadow range:NSMakeRange(0, attributedString.length)];
|
|
}
|
|
}
|
|
|
|
#pragma mark - Drawing
|
|
|
|
- (NSObject *)drawParametersForAsyncLayer:(_ASDisplayLayer *)layer
|
|
{
|
|
ASLockScopeSelf();
|
|
[self _ensureTruncationText];
|
|
|
|
// Unlike layout, here we must copy the container since drawing is asynchronous.
|
|
ASTextContainer *copiedContainer = [_textContainer copy];
|
|
copiedContainer.size = self.bounds.size;
|
|
[copiedContainer makeImmutable];
|
|
NSMutableAttributedString *mutableText = [_attributedText mutableCopy] ?: [[NSMutableAttributedString alloc] init];
|
|
|
|
[self prepareAttributedString:mutableText isForIntrinsicSize:NO];
|
|
|
|
return @{
|
|
@"container": copiedContainer,
|
|
@"text": mutableText,
|
|
@"bgColor": self.backgroundColor ?: [NSNull null]
|
|
};
|
|
}
|
|
|
|
+ (void)drawRect:(CGRect)bounds withParameters:(NSDictionary *)layoutDict isCancelled:(NS_NOESCAPE asdisplaynode_iscancelled_block_t)isCancelledBlock isRasterizing:(BOOL)isRasterizing
|
|
{
|
|
ASTextContainer *container = layoutDict[@"container"];
|
|
NSAttributedString *text = layoutDict[@"text"];
|
|
UIColor *bgColor = layoutDict[@"bgColor"];
|
|
ASTextLayout *layout = ASTextNodeCompatibleLayoutWithContainerAndText(container, text);
|
|
|
|
if (isCancelledBlock()) {
|
|
return;
|
|
}
|
|
|
|
// Fill background color.
|
|
if (bgColor == (id)[NSNull null]) {
|
|
bgColor = nil;
|
|
}
|
|
|
|
// They may have already drawn into this context in the pre-context block
|
|
// so unfortunately we have to use the normal blend mode, not copy.
|
|
if (bgColor && CGColorGetAlpha(bgColor.CGColor) > 0) {
|
|
[bgColor setFill];
|
|
UIRectFillUsingBlendMode(bounds, kCGBlendModeNormal);
|
|
}
|
|
|
|
CGContextRef context = UIGraphicsGetCurrentContext();
|
|
ASDisplayNodeAssert(context, @"This is no good without a context.");
|
|
|
|
[layout drawInContext:context size:bounds.size point:bounds.origin view:nil layer:nil debug:[ASTextDebugOption sharedDebugOption] cancel:isCancelledBlock];
|
|
}
|
|
|
|
#pragma mark - Attributes
|
|
|
|
- (id)linkAttributeValueAtPoint:(CGPoint)point
|
|
attributeName:(out NSString **)attributeNameOut
|
|
range:(out NSRange *)rangeOut
|
|
{
|
|
return [self _linkAttributeValueAtPoint:point
|
|
attributeName:attributeNameOut
|
|
range:rangeOut
|
|
inAdditionalTruncationMessage:NULL
|
|
forHighlighting:NO];
|
|
}
|
|
|
|
- (id)_linkAttributeValueAtPoint:(CGPoint)point
|
|
attributeName:(out NSString **)attributeNameOut
|
|
range:(out NSRange *)rangeOut
|
|
inAdditionalTruncationMessage:(out BOOL *)inAdditionalTruncationMessageOut
|
|
forHighlighting:(BOOL)highlighting
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
// TODO: The copy and application of size shouldn't be required, but it is currently.
|
|
// See discussion in https://github.com/TextureGroup/Texture/pull/396
|
|
ASTextContainer *containerCopy = [_textContainer copy];
|
|
containerCopy.size = self.calculatedSize;
|
|
ASTextLayout *layout = ASTextNodeCompatibleLayoutWithContainerAndText(containerCopy, _attributedText);
|
|
|
|
if ([self _locked_pointInsideAdditionalTruncationMessage:point withLayout:layout]) {
|
|
if (inAdditionalTruncationMessageOut != NULL) {
|
|
*inAdditionalTruncationMessageOut = YES;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
NSRange visibleRange = layout.visibleRange;
|
|
NSRange clampedRange = NSIntersectionRange(visibleRange, NSMakeRange(0, _attributedText.length));
|
|
|
|
// Search the 9 points of a 44x44 square around the touch until we find a link.
|
|
// Start from center, then do sides, then do top/bottom, then do corners.
|
|
static constexpr CGSize kRectOffsets[9] = {
|
|
{ 0, 0 },
|
|
{ -22, 0 }, { 22, 0 },
|
|
{ 0, -22 }, { 0, 22 },
|
|
{ -22, -22 }, { -22, 22 },
|
|
{ 22, -22 }, { 22, 22 }
|
|
};
|
|
|
|
for (const CGSize &offset : kRectOffsets) {
|
|
const CGPoint testPoint = CGPointMake(point.x + offset.width,
|
|
point.y + offset.height);
|
|
ASTextPosition *pos = [layout closestPositionToPoint:testPoint];
|
|
if (!pos || !NSLocationInRange(pos.offset, clampedRange)) {
|
|
continue;
|
|
}
|
|
for (NSString *attributeName in _linkAttributeNames) {
|
|
NSRange effectiveRange = NSMakeRange(0, 0);
|
|
id value = [_attributedText attribute:attributeName atIndex:pos.offset
|
|
longestEffectiveRange:&effectiveRange inRange:clampedRange];
|
|
if (value == nil) {
|
|
// Didn't find any links specified with this attribute.
|
|
continue;
|
|
}
|
|
|
|
// If highlighting, check with delegate first. If not implemented, assume YES.
|
|
if (highlighting
|
|
&& [_delegate respondsToSelector:@selector(textNode:shouldHighlightLinkAttribute:value:atPoint:)]
|
|
&& ![_delegate textNode:(ASTextNode *)self shouldHighlightLinkAttribute:attributeName
|
|
value:value atPoint:point]) {
|
|
continue;
|
|
}
|
|
|
|
*rangeOut = NSIntersectionRange(visibleRange, effectiveRange);
|
|
|
|
if (attributeNameOut != NULL) {
|
|
*attributeNameOut = attributeName;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
}
|
|
|
|
return nil;
|
|
}
|
|
|
|
- (BOOL)_locked_pointInsideAdditionalTruncationMessage:(CGPoint)point withLayout:(ASTextLayout *)layout
|
|
{
|
|
// Check if the range is within the additional truncation range
|
|
BOOL inAdditionalTruncationMessage = NO;
|
|
|
|
CTLineRef truncatedCTLine = layout.truncatedLine.CTLine;
|
|
if (truncatedCTLine != NULL && _additionalTruncationMessage != nil) {
|
|
CFIndex stringIndexForPosition = CTLineGetStringIndexForPosition(truncatedCTLine, point);
|
|
if (stringIndexForPosition != kCFNotFound) {
|
|
CFIndex truncatedCTLineGlyphCount = CTLineGetGlyphCount(truncatedCTLine);
|
|
|
|
CTLineRef truncationTokenLine = CTLineCreateWithAttributedString((CFAttributedStringRef)_truncationAttributedText);
|
|
CFIndex truncationTokenLineGlyphCount = truncationTokenLine ? CTLineGetGlyphCount(truncationTokenLine) : 0;
|
|
CFRelease(truncationTokenLine);
|
|
|
|
CTLineRef additionalTruncationTokenLine = CTLineCreateWithAttributedString((CFAttributedStringRef)_additionalTruncationMessage);
|
|
CFIndex additionalTruncationTokenLineGlyphCount = additionalTruncationTokenLine ? CTLineGetGlyphCount(additionalTruncationTokenLine) : 0;
|
|
CFRelease(additionalTruncationTokenLine);
|
|
|
|
switch (_textContainer.truncationType) {
|
|
case ASTextTruncationTypeStart: {
|
|
CFIndex composedTruncationTextLineGlyphCount = truncationTokenLineGlyphCount + additionalTruncationTokenLineGlyphCount;
|
|
if (stringIndexForPosition > truncationTokenLineGlyphCount &&
|
|
stringIndexForPosition < composedTruncationTextLineGlyphCount) {
|
|
inAdditionalTruncationMessage = YES;
|
|
}
|
|
break;
|
|
}
|
|
case ASTextTruncationTypeMiddle: {
|
|
CFIndex composedTruncationTextLineGlyphCount = truncationTokenLineGlyphCount + additionalTruncationTokenLineGlyphCount;
|
|
CFIndex firstTruncatedTokenIndex = (truncatedCTLineGlyphCount - composedTruncationTextLineGlyphCount) / 2.0;
|
|
if ((firstTruncatedTokenIndex + truncationTokenLineGlyphCount) < stringIndexForPosition &&
|
|
stringIndexForPosition < (firstTruncatedTokenIndex + composedTruncationTextLineGlyphCount)) {
|
|
inAdditionalTruncationMessage = YES;
|
|
}
|
|
break;
|
|
}
|
|
case ASTextTruncationTypeEnd: {
|
|
if (stringIndexForPosition > (truncatedCTLineGlyphCount - additionalTruncationTokenLineGlyphCount)) {
|
|
inAdditionalTruncationMessage = YES;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
// For now, assume that a tap inside this text, but outside the text range is a tap on the
|
|
// truncation token.
|
|
if (![layout textRangeAtPoint:point]) {
|
|
inAdditionalTruncationMessage = YES;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return inAdditionalTruncationMessage;
|
|
}
|
|
|
|
#pragma mark - UIGestureRecognizerDelegate
|
|
|
|
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
ASLockScopeSelf(); // Protect usage of _highlight* ivars.
|
|
|
|
if (gestureRecognizer == _longPressGestureRecognizer) {
|
|
// Don't allow long press on truncation message
|
|
if ([self _pendingTruncationTap]) {
|
|
return NO;
|
|
}
|
|
|
|
// Ask our delegate if a long-press on an attribute is relevant
|
|
id<ASTextNodeDelegate> delegate = self.delegate;
|
|
if ([delegate respondsToSelector:@selector(textNode:shouldLongPressLinkAttribute:value:atPoint:)]) {
|
|
return [delegate textNode:(ASTextNode *)self
|
|
shouldLongPressLinkAttribute:_highlightedLinkAttributeName
|
|
value:_highlightedLinkAttributeValue
|
|
atPoint:[gestureRecognizer locationInView:self.view]];
|
|
}
|
|
|
|
// Otherwise we are good to go.
|
|
return YES;
|
|
}
|
|
|
|
if (([self _pendingLinkTap] || [self _pendingTruncationTap])
|
|
&& [gestureRecognizer isKindOfClass:[UITapGestureRecognizer class]]
|
|
&& CGRectContainsPoint(self.threadSafeBounds, [gestureRecognizer locationInView:self.view])) {
|
|
return NO;
|
|
}
|
|
|
|
return [super gestureRecognizerShouldBegin:gestureRecognizer];
|
|
}
|
|
|
|
#pragma mark - Highlighting
|
|
|
|
- (ASTextNodeHighlightStyle)highlightStyle
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
return _highlightStyle;
|
|
}
|
|
|
|
- (void)setHighlightStyle:(ASTextNodeHighlightStyle)highlightStyle
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
_highlightStyle = highlightStyle;
|
|
}
|
|
|
|
- (NSRange)highlightRange
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
return _highlightRange;
|
|
}
|
|
|
|
- (void)setHighlightRange:(NSRange)highlightRange
|
|
{
|
|
[self setHighlightRange:highlightRange animated:NO];
|
|
}
|
|
|
|
- (void)setHighlightRange:(NSRange)highlightRange animated:(BOOL)animated
|
|
{
|
|
[self _setHighlightRange:highlightRange forAttributeName:nil value:nil animated:animated];
|
|
}
|
|
|
|
- (void)_setHighlightRange:(NSRange)highlightRange forAttributeName:(NSString *)highlightedAttributeName value:(id)highlightedAttributeValue animated:(BOOL)animated
|
|
{
|
|
ASLockScopeSelf(); // Protect usage of _highlight* ivars.
|
|
|
|
// Set these so that link tapping works.
|
|
_highlightedLinkAttributeName = highlightedAttributeName;
|
|
_highlightedLinkAttributeValue = highlightedAttributeValue;
|
|
_highlightRange = highlightRange;
|
|
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
// Much of the code from original ASTextNode is probably usable here.
|
|
|
|
return;
|
|
}
|
|
|
|
- (void)_clearHighlightIfNecessary
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
|
|
if ([self _pendingLinkTap] || [self _pendingTruncationTap]) {
|
|
[self setHighlightRange:NSMakeRange(0, 0) animated:YES];
|
|
}
|
|
}
|
|
|
|
+ (CGColorRef)_highlightColorForStyle:(ASTextNodeHighlightStyle)style
|
|
{
|
|
return [UIColor colorWithWhite:(style == ASTextNodeHighlightStyleLight ? 0.0 : 1.0) alpha:1.0].CGColor;
|
|
}
|
|
|
|
+ (CGFloat)_highlightOpacityForStyle:(ASTextNodeHighlightStyle)style
|
|
{
|
|
return (style == ASTextNodeHighlightStyleLight) ? ASTextNodeHighlightLightOpacity : ASTextNodeHighlightDarkOpacity;
|
|
}
|
|
|
|
#pragma mark - Text rects
|
|
|
|
- (NSArray *)rectsForTextRange:(NSRange)textRange
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return @[];
|
|
}
|
|
|
|
- (NSArray *)highlightRectsForTextRange:(NSRange)textRange
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return @[];
|
|
}
|
|
|
|
- (CGRect)trailingRect
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return CGRectZero;
|
|
}
|
|
|
|
- (CGRect)frameForTextRange:(NSRange)textRange
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return CGRectZero;
|
|
}
|
|
|
|
#pragma mark - Placeholders
|
|
|
|
- (UIColor *)placeholderColor
|
|
{
|
|
return ASLockedSelf(_placeholderColor);
|
|
}
|
|
|
|
- (void)setPlaceholderColor:(UIColor *)placeholderColor
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCopy(_placeholderColor, placeholderColor)) {
|
|
self.placeholderEnabled = CGColorGetAlpha(placeholderColor.CGColor) > 0;
|
|
}
|
|
}
|
|
|
|
- (UIImage *)placeholderImage
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return nil;
|
|
}
|
|
|
|
#pragma mark - Touch Handling
|
|
|
|
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
|
|
if (!_passthroughNonlinkTouches) {
|
|
return [super pointInside:point withEvent:event];
|
|
}
|
|
|
|
NSRange range = NSMakeRange(0, 0);
|
|
NSString *linkAttributeName = nil;
|
|
BOOL inAdditionalTruncationMessage = NO;
|
|
|
|
id linkAttributeValue = [self _linkAttributeValueAtPoint:point
|
|
attributeName:&linkAttributeName
|
|
range:&range
|
|
inAdditionalTruncationMessage:&inAdditionalTruncationMessage
|
|
forHighlighting:YES];
|
|
|
|
NSUInteger lastCharIndex = NSIntegerMax;
|
|
BOOL linkCrossesVisibleRange = (lastCharIndex > range.location) && (lastCharIndex < NSMaxRange(range) - 1);
|
|
|
|
if (range.length > 0 && !linkCrossesVisibleRange && linkAttributeValue != nil && linkAttributeName != nil) {
|
|
return YES;
|
|
} else {
|
|
return NO;
|
|
}
|
|
}
|
|
|
|
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
|
|
[super touchesBegan:touches withEvent:event];
|
|
|
|
CGPoint point = [[touches anyObject] locationInView:self.view];
|
|
|
|
NSRange range = NSMakeRange(0, 0);
|
|
NSString *linkAttributeName = nil;
|
|
BOOL inAdditionalTruncationMessage = NO;
|
|
|
|
id linkAttributeValue = [self _linkAttributeValueAtPoint:point
|
|
attributeName:&linkAttributeName
|
|
range:&range
|
|
inAdditionalTruncationMessage:&inAdditionalTruncationMessage
|
|
forHighlighting:YES];
|
|
|
|
NSUInteger lastCharIndex = NSIntegerMax;
|
|
BOOL linkCrossesVisibleRange = (lastCharIndex > range.location) && (lastCharIndex < NSMaxRange(range) - 1);
|
|
|
|
if (inAdditionalTruncationMessage) {
|
|
NSRange visibleRange = NSMakeRange(0, 0);
|
|
{
|
|
ASLockScopeSelf();
|
|
// TODO: The copy and application of size shouldn't be required, but it is currently.
|
|
// See discussion in https://github.com/TextureGroup/Texture/pull/396
|
|
ASTextContainer *containerCopy = [_textContainer copy];
|
|
containerCopy.size = self.calculatedSize;
|
|
ASTextLayout *layout = ASTextNodeCompatibleLayoutWithContainerAndText(containerCopy, _attributedText);
|
|
visibleRange = layout.visibleRange;
|
|
}
|
|
NSRange truncationMessageRange = [self _additionalTruncationMessageRangeWithVisibleRange:visibleRange];
|
|
[self _setHighlightRange:truncationMessageRange forAttributeName:ASTextNodeTruncationTokenAttributeName value:nil animated:YES];
|
|
} else if (range.length > 0 && !linkCrossesVisibleRange && linkAttributeValue != nil && linkAttributeName != nil) {
|
|
[self _setHighlightRange:range forAttributeName:linkAttributeName value:linkAttributeValue animated:YES];
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
[super touchesCancelled:touches withEvent:event];
|
|
|
|
[self _clearHighlightIfNecessary];
|
|
}
|
|
|
|
- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
[super touchesEnded:touches withEvent:event];
|
|
|
|
ASLockScopeSelf(); // Protect usage of _highlight* ivars.
|
|
id<ASTextNodeDelegate> delegate = self.delegate;
|
|
if ([self _pendingLinkTap] && [delegate respondsToSelector:@selector(textNode:tappedLinkAttribute:value:atPoint:textRange:)]) {
|
|
CGPoint point = [[touches anyObject] locationInView:self.view];
|
|
[delegate textNode:(ASTextNode *)self tappedLinkAttribute:_highlightedLinkAttributeName value:_highlightedLinkAttributeValue atPoint:point textRange:_highlightRange];
|
|
}
|
|
|
|
if ([self _pendingTruncationTap]) {
|
|
if ([delegate respondsToSelector:@selector(textNodeTappedTruncationToken:)]) {
|
|
[delegate textNodeTappedTruncationToken:(ASTextNode *)self];
|
|
}
|
|
}
|
|
|
|
[self _clearHighlightIfNecessary];
|
|
}
|
|
|
|
- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
[super touchesMoved:touches withEvent:event];
|
|
|
|
ASLockScopeSelf(); // Protect usage of _highlight* ivars.
|
|
UITouch *touch = [touches anyObject];
|
|
CGPoint locationInView = [touch locationInView:self.view];
|
|
// on 3D Touch enabled phones, this gets fired with changes in force, and usually will get fired immediately after touchesBegan:withEvent:
|
|
if (CGPointEqualToPoint([touch previousLocationInView:self.view], locationInView))
|
|
return;
|
|
|
|
// If touch has moved out of the current highlight range, clear the highlight.
|
|
if (_highlightRange.length > 0) {
|
|
NSRange range = NSMakeRange(0, 0);
|
|
[self _linkAttributeValueAtPoint:locationInView
|
|
attributeName:NULL
|
|
range:&range
|
|
inAdditionalTruncationMessage:NULL
|
|
forHighlighting:YES];
|
|
|
|
if (!NSEqualRanges(_highlightRange, range)) {
|
|
[self _clearHighlightIfNecessary];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)_handleLongPress:(UILongPressGestureRecognizer *)longPressRecognizer
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
|
|
// Respond to long-press when it begins, not when it ends.
|
|
if (longPressRecognizer.state == UIGestureRecognizerStateBegan) {
|
|
id<ASTextNodeDelegate> delegate = self.delegate;
|
|
if ([delegate respondsToSelector:@selector(textNode:longPressedLinkAttribute:value:atPoint:textRange:)]) {
|
|
ASLockScopeSelf(); // Protect usage of _highlight* ivars.
|
|
CGPoint touchPoint = [_longPressGestureRecognizer locationInView:self.view];
|
|
[delegate textNode:(ASTextNode *)self longPressedLinkAttribute:_highlightedLinkAttributeName value:_highlightedLinkAttributeValue atPoint:touchPoint textRange:_highlightRange];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (BOOL)_pendingLinkTap
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
return (_highlightedLinkAttributeValue != nil && ![self _pendingTruncationTap]) && self.delegate != nil;
|
|
}
|
|
|
|
- (BOOL)_pendingTruncationTap
|
|
{
|
|
return [ASLockedSelf(_highlightedLinkAttributeName) isEqualToString:ASTextNodeTruncationTokenAttributeName];
|
|
}
|
|
|
|
#pragma mark - Shadow Properties
|
|
|
|
/**
|
|
* Note about shadowed text:
|
|
*
|
|
* Shadowed text is pretty rare, and we are a framework that targets serious developers.
|
|
* We should probably ignore these properties and tell developers to set the shadow into their attributed text instead.
|
|
*/
|
|
- (CGColorRef)shadowColor
|
|
{
|
|
return ASLockedSelf(_shadowColor);
|
|
}
|
|
|
|
- (void)setShadowColor:(CGColorRef)shadowColor
|
|
{
|
|
ASLockScopeSelf();
|
|
if (_shadowColor != shadowColor && CGColorEqualToColor(shadowColor, _shadowColor) == NO) {
|
|
CGColorRelease(_shadowColor);
|
|
_shadowColor = CGColorRetain(shadowColor);
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (CGSize)shadowOffset
|
|
{
|
|
return ASLockedSelf(_shadowOffset);
|
|
}
|
|
|
|
- (void)setShadowOffset:(CGSize)shadowOffset
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCustom(_shadowOffset, shadowOffset, CGSizeEqualToSize)) {
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (CGFloat)shadowOpacity
|
|
{
|
|
return ASLockedSelf(_shadowOpacity);
|
|
}
|
|
|
|
- (void)setShadowOpacity:(CGFloat)shadowOpacity
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssign(_shadowOpacity, shadowOpacity)) {
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (CGFloat)shadowRadius
|
|
{
|
|
return ASLockedSelf(_shadowRadius);
|
|
}
|
|
|
|
- (void)setShadowRadius:(CGFloat)shadowRadius
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssign(_shadowRadius, shadowRadius)) {
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (UIEdgeInsets)shadowPadding
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return UIEdgeInsetsZero;
|
|
}
|
|
|
|
- (void)setPointSizeScaleFactors:(NSArray<NSNumber *> *)scaleFactors
|
|
{
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCopy(_pointSizeScaleFactors, scaleFactors)) {
|
|
[self setNeedsLayout];
|
|
}
|
|
}
|
|
|
|
- (NSArray<NSNumber *> *)pointSizeScaleFactors
|
|
{
|
|
return ASLockedSelf(_pointSizeScaleFactors);
|
|
}
|
|
|
|
#pragma mark - Truncation Message
|
|
|
|
static NSAttributedString *DefaultTruncationAttributedString()
|
|
{
|
|
static NSAttributedString *defaultTruncationAttributedString;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
defaultTruncationAttributedString = [[NSAttributedString alloc] initWithString:NSLocalizedString(@"\u2026", @"Default truncation string")];
|
|
});
|
|
return defaultTruncationAttributedString;
|
|
}
|
|
|
|
- (void)_ensureTruncationText
|
|
{
|
|
ASLockScopeSelf();
|
|
if (_textContainer.truncationToken == nil) {
|
|
_textContainer.truncationToken = [self _locked_composedTruncationText];
|
|
}
|
|
}
|
|
|
|
- (NSAttributedString *)truncationAttributedText
|
|
{
|
|
return ASLockedSelf(_truncationAttributedText);
|
|
}
|
|
|
|
- (void)setTruncationAttributedText:(NSAttributedString *)truncationAttributedText
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCopy(_truncationAttributedText, truncationAttributedText)) {
|
|
[self _invalidateTruncationText];
|
|
}
|
|
}
|
|
|
|
- (NSAttributedString *)additionalTruncationMessage
|
|
{
|
|
return ASLockedSelf(_additionalTruncationMessage);
|
|
}
|
|
|
|
- (void)setAdditionalTruncationMessage:(NSAttributedString *)additionalTruncationMessage
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssignCopy(_additionalTruncationMessage, additionalTruncationMessage)) {
|
|
[self _invalidateTruncationText];
|
|
}
|
|
}
|
|
|
|
- (NSLineBreakMode)truncationMode
|
|
{
|
|
return ASLockedSelf(_truncationMode);
|
|
}
|
|
|
|
- (void)setTruncationMode:(NSLineBreakMode)truncationMode
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssign(_truncationMode, truncationMode)) {
|
|
ASTextTruncationType truncationType;
|
|
switch (truncationMode) {
|
|
case NSLineBreakByTruncatingHead:
|
|
truncationType = ASTextTruncationTypeStart;
|
|
break;
|
|
case NSLineBreakByTruncatingTail:
|
|
truncationType = ASTextTruncationTypeEnd;
|
|
break;
|
|
case NSLineBreakByTruncatingMiddle:
|
|
truncationType = ASTextTruncationTypeMiddle;
|
|
break;
|
|
default:
|
|
truncationType = ASTextTruncationTypeNone;
|
|
}
|
|
|
|
_textContainer.truncationType = truncationType;
|
|
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (BOOL)isTruncated
|
|
{
|
|
return ASLockedSelf([self locked_textLayoutForSize:[self _locked_threadSafeBounds].size].truncatedLine != nil);
|
|
}
|
|
|
|
- (BOOL)shouldTruncateForConstrainedSize:(ASSizeRange)constrainedSize
|
|
{
|
|
return ASLockedSelf([self locked_textLayoutForSize:constrainedSize.max].truncatedLine != nil);
|
|
}
|
|
|
|
- (ASTextLayout *)locked_textLayoutForSize:(CGSize)size
|
|
{
|
|
ASTextContainer *container = [_textContainer copy];
|
|
container.size = size;
|
|
return ASTextNodeCompatibleLayoutWithContainerAndText(container, _attributedText);
|
|
}
|
|
|
|
- (NSUInteger)maximumNumberOfLines
|
|
{
|
|
// _textContainer is invariant and this is just atomic access.
|
|
return _textContainer.maximumNumberOfRows;
|
|
}
|
|
|
|
- (void)setMaximumNumberOfLines:(NSUInteger)maximumNumberOfLines
|
|
{
|
|
ASLockScopeSelf();
|
|
if (ASCompareAssign(_textContainer.maximumNumberOfRows, maximumNumberOfLines)) {
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
- (NSUInteger)lineCount
|
|
{
|
|
ASLockScopeSelf();
|
|
AS_TEXT_ALERT_UNIMPLEMENTED_FEATURE();
|
|
return 0;
|
|
}
|
|
|
|
#pragma mark - Truncation Message
|
|
|
|
- (void)_invalidateTruncationText
|
|
{
|
|
ASLockScopeSelf();
|
|
[self _locked_invalidateTruncationText];
|
|
[self setNeedsDisplay];
|
|
}
|
|
|
|
- (void)_locked_invalidateTruncationText
|
|
{
|
|
_textContainer.truncationToken = nil;
|
|
}
|
|
|
|
/**
|
|
* @return the additional truncation message range within the as-rendered text.
|
|
* Must be called from main thread
|
|
*/
|
|
- (NSRange)_additionalTruncationMessageRangeWithVisibleRange:(NSRange)visibleRange
|
|
{
|
|
ASLockScopeSelf();
|
|
|
|
// Check if we even have an additional truncation message.
|
|
if (!_additionalTruncationMessage) {
|
|
return NSMakeRange(NSNotFound, 0);
|
|
}
|
|
|
|
// Character location of the unicode ellipsis (the first index after the visible range)
|
|
NSInteger truncationTokenIndex = NSMaxRange(visibleRange);
|
|
|
|
NSUInteger additionalTruncationMessageLength = _additionalTruncationMessage.length;
|
|
// We get the location of the truncation token, then add the length of the
|
|
// truncation attributed string +1 for the space between.
|
|
return NSMakeRange(truncationTokenIndex + _truncationAttributedText.length + 1, additionalTruncationMessageLength);
|
|
}
|
|
|
|
/**
|
|
* @return the truncation message for the string. If there are both an
|
|
* additional truncation message and a truncation attributed string, they will
|
|
* be properly composed.
|
|
*/
|
|
- (NSAttributedString *)_locked_composedTruncationText
|
|
{
|
|
ASAssertLocked(__instanceLock__);
|
|
if (_composedTruncationText == nil) {
|
|
if (_truncationAttributedText != nil && _additionalTruncationMessage != nil) {
|
|
NSMutableAttributedString *newComposedTruncationString = [[NSMutableAttributedString alloc] initWithAttributedString:_truncationAttributedText];
|
|
[newComposedTruncationString.mutableString appendString:@" "];
|
|
[newComposedTruncationString appendAttributedString:_additionalTruncationMessage];
|
|
_composedTruncationText = newComposedTruncationString;
|
|
} else if (_truncationAttributedText != nil) {
|
|
_composedTruncationText = _truncationAttributedText;
|
|
} else if (_additionalTruncationMessage != nil) {
|
|
_composedTruncationText = _additionalTruncationMessage;
|
|
} else {
|
|
_composedTruncationText = DefaultTruncationAttributedString();
|
|
}
|
|
_composedTruncationText = [self _locked_prepareTruncationStringForDrawing:_composedTruncationText];
|
|
}
|
|
return _composedTruncationText;
|
|
}
|
|
|
|
/**
|
|
* - cleanses it of core text attributes so TextKit doesn't crash
|
|
* - Adds whole-string attributes so the truncation message matches the styling
|
|
* of the body text
|
|
*/
|
|
- (NSAttributedString *)_locked_prepareTruncationStringForDrawing:(NSAttributedString *)truncationString
|
|
{
|
|
ASAssertLocked(__instanceLock__);
|
|
NSMutableAttributedString *truncationMutableString = [truncationString mutableCopy];
|
|
// Grab the attributes from the full string
|
|
if (_attributedText.length > 0) {
|
|
NSAttributedString *originalString = _attributedText;
|
|
NSInteger originalStringLength = _attributedText.length;
|
|
// Add any of the original string's attributes to the truncation string,
|
|
// but don't overwrite any of the truncation string's attributes
|
|
NSDictionary *originalStringAttributes = [originalString attributesAtIndex:originalStringLength-1 effectiveRange:NULL];
|
|
[truncationString enumerateAttributesInRange:NSMakeRange(0, truncationString.length) options:0 usingBlock:
|
|
^(NSDictionary *attributes, NSRange range, BOOL *stop) {
|
|
NSMutableDictionary *futureTruncationAttributes = [originalStringAttributes mutableCopy];
|
|
[futureTruncationAttributes addEntriesFromDictionary:attributes];
|
|
[truncationMutableString setAttributes:futureTruncationAttributes range:range];
|
|
}];
|
|
}
|
|
return truncationMutableString;
|
|
}
|
|
|
|
#if AS_TEXTNODE2_RECORD_ATTRIBUTED_STRINGS
|
|
+ (void)_registerAttributedText:(NSAttributedString *)str
|
|
{
|
|
static NSMutableArray *array;
|
|
static NSLock *lock;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
lock = [NSLock new];
|
|
array = [NSMutableArray new];
|
|
});
|
|
[lock lock];
|
|
[array addObject:str];
|
|
if (array.count % 20 == 0) {
|
|
NSLog(@"Got %d strings", (int)array.count);
|
|
}
|
|
if (array.count == 2000) {
|
|
NSString *path = [NSTemporaryDirectory() stringByAppendingPathComponent:@"AttributedStrings.plist"];
|
|
NSAssert([NSKeyedArchiver archiveRootObject:array toFile:path], nil);
|
|
NSLog(@"Saved to %@", path);
|
|
}
|
|
[lock unlock];
|
|
}
|
|
#endif
|
|
|
|
+ (void)enableDebugging
|
|
{
|
|
ASTextDebugOption *debugOption = [[ASTextDebugOption alloc] init];
|
|
debugOption.CTLineFillColor = [UIColor colorWithRed:0 green:0.3 blue:1 alpha:0.1];
|
|
[ASTextDebugOption setSharedDebugOption:debugOption];
|
|
}
|
|
|
|
- (BOOL)usingExperiment
|
|
{
|
|
return YES;
|
|
}
|
|
|
|
@end
|