mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-08-31 18:02:20 +00:00
* Rename -[ASCellNode viewModel] -> -[ASCellNode nodeViewModel] to avoid breaking existing code * Update test * Update the changelog
435 lines
13 KiB
Plaintext
435 lines
13 KiB
Plaintext
//
|
|
// ASCellNode.mm
|
|
// Texture
|
|
//
|
|
// Copyright (c) 2014-present, Facebook, Inc. All rights reserved.
|
|
// This source code is licensed under the BSD-style license found in the
|
|
// LICENSE file in the /ASDK-Licenses directory of this source tree. An additional
|
|
// grant of patent rights can be found in the PATENTS file in the same directory.
|
|
//
|
|
// Modifications to this file made after 4/13/2017 are: Copyright (c) 2017-present,
|
|
// Pinterest, Inc. Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
|
|
#import <AsyncDisplayKit/ASCellNode+Internal.h>
|
|
|
|
#import <AsyncDisplayKit/ASEqualityHelpers.h>
|
|
#import <AsyncDisplayKit/ASInternalHelpers.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+FrameworkPrivate.h>
|
|
#import <AsyncDisplayKit/ASCollectionView+Undeprecated.h>
|
|
#import <AsyncDisplayKit/ASCollectionElement.h>
|
|
#import <AsyncDisplayKit/ASTableView+Undeprecated.h>
|
|
#import <AsyncDisplayKit/_ASDisplayView.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+Subclasses.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+Beta.h>
|
|
#import <AsyncDisplayKit/ASTextNode.h>
|
|
#import <AsyncDisplayKit/ASCollectionNode.h>
|
|
#import <AsyncDisplayKit/ASTableNode.h>
|
|
|
|
#import <AsyncDisplayKit/ASViewController.h>
|
|
#import <AsyncDisplayKit/ASInsetLayoutSpec.h>
|
|
#import <AsyncDisplayKit/ASDisplayNodeInternal.h>
|
|
|
|
#pragma mark -
|
|
#pragma mark ASCellNode
|
|
|
|
@interface ASCellNode ()
|
|
{
|
|
ASDisplayNodeViewControllerBlock _viewControllerBlock;
|
|
ASDisplayNodeDidLoadBlock _viewControllerDidLoadBlock;
|
|
ASDisplayNode *_viewControllerNode;
|
|
UIViewController *_viewController;
|
|
BOOL _suspendInteractionDelegate;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation ASCellNode
|
|
@synthesize interactionDelegate = _interactionDelegate;
|
|
|
|
- (instancetype)init
|
|
{
|
|
if (!(self = [super init]))
|
|
return nil;
|
|
|
|
// Use UITableViewCell defaults
|
|
_selectionStyle = UITableViewCellSelectionStyleDefault;
|
|
self.clipsToBounds = YES;
|
|
|
|
return self;
|
|
}
|
|
|
|
- (instancetype)initWithViewControllerBlock:(ASDisplayNodeViewControllerBlock)viewControllerBlock didLoadBlock:(ASDisplayNodeDidLoadBlock)didLoadBlock
|
|
{
|
|
if (!(self = [super init]))
|
|
return nil;
|
|
|
|
ASDisplayNodeAssertNotNil(viewControllerBlock, @"should initialize with a valid block that returns a UIViewController");
|
|
_viewControllerBlock = viewControllerBlock;
|
|
_viewControllerDidLoadBlock = didLoadBlock;
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)didLoad
|
|
{
|
|
[super didLoad];
|
|
|
|
if (_viewControllerBlock != nil) {
|
|
|
|
_viewController = _viewControllerBlock();
|
|
_viewControllerBlock = nil;
|
|
|
|
if ([_viewController isKindOfClass:[ASViewController class]]) {
|
|
ASViewController *asViewController = (ASViewController *)_viewController;
|
|
_viewControllerNode = asViewController.node;
|
|
[_viewController view];
|
|
} else {
|
|
// Careful to avoid retain cycle
|
|
UIViewController *viewController = _viewController;
|
|
_viewControllerNode = [[ASDisplayNode alloc] initWithViewBlock:^{
|
|
return viewController.view;
|
|
}];
|
|
}
|
|
[self addSubnode:_viewControllerNode];
|
|
|
|
// Since we just loaded our node, and added _viewControllerNode as a subnode,
|
|
// _viewControllerNode must have just loaded its view, so now is an appropriate
|
|
// time to execute our didLoadBlock, if we were given one.
|
|
if (_viewControllerDidLoadBlock != nil) {
|
|
_viewControllerDidLoadBlock(self);
|
|
_viewControllerDidLoadBlock = nil;
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)layout
|
|
{
|
|
[super layout];
|
|
|
|
_viewControllerNode.frame = self.bounds;
|
|
}
|
|
|
|
- (void)_rootNodeDidInvalidateSize
|
|
{
|
|
if (_interactionDelegate != nil) {
|
|
[_interactionDelegate nodeDidInvalidateSize:self];
|
|
} else {
|
|
[super _rootNodeDidInvalidateSize];
|
|
}
|
|
}
|
|
|
|
- (void)_layoutTransitionMeasurementDidFinish
|
|
{
|
|
if (_interactionDelegate != nil) {
|
|
[_interactionDelegate nodeDidInvalidateSize:self];
|
|
} else {
|
|
[super _layoutTransitionMeasurementDidFinish];
|
|
}
|
|
}
|
|
|
|
- (void)setSelected:(BOOL)selected
|
|
{
|
|
if (_selected != selected) {
|
|
_selected = selected;
|
|
if (!_suspendInteractionDelegate) {
|
|
[_interactionDelegate nodeSelectedStateDidChange:self];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)setHighlighted:(BOOL)highlighted
|
|
{
|
|
if (_highlighted != highlighted) {
|
|
_highlighted = highlighted;
|
|
if (!_suspendInteractionDelegate) {
|
|
[_interactionDelegate nodeHighlightedStateDidChange:self];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)__setSelectedFromUIKit:(BOOL)selected;
|
|
{
|
|
if (selected != _selected) {
|
|
_suspendInteractionDelegate = YES;
|
|
self.selected = selected;
|
|
_suspendInteractionDelegate = NO;
|
|
}
|
|
}
|
|
|
|
- (void)__setHighlightedFromUIKit:(BOOL)highlighted;
|
|
{
|
|
if (highlighted != _highlighted) {
|
|
_suspendInteractionDelegate = YES;
|
|
self.highlighted = highlighted;
|
|
_suspendInteractionDelegate = NO;
|
|
}
|
|
}
|
|
|
|
- (BOOL)canUpdateToViewModel:(id)viewModel
|
|
{
|
|
return [self.nodeViewModel class] == [viewModel class];
|
|
}
|
|
|
|
- (NSIndexPath *)indexPath
|
|
{
|
|
return [self.owningNode indexPathForNode:self];
|
|
}
|
|
|
|
- (UIViewController *)viewController
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
// Force the view to load so that we will create the
|
|
// view controller if we haven't already.
|
|
if (self.isNodeLoaded == NO) {
|
|
[self view];
|
|
}
|
|
return _viewController;
|
|
}
|
|
|
|
#pragma clang diagnostic push
|
|
#pragma clang diagnostic ignored "-Wobjc-missing-super-calls"
|
|
|
|
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
ASDisplayNodeAssert([self.view isKindOfClass:_ASDisplayView.class], @"ASCellNode views must be of type _ASDisplayView");
|
|
[(_ASDisplayView *)self.view __forwardTouchesBegan:touches withEvent:event];
|
|
}
|
|
|
|
- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
ASDisplayNodeAssert([self.view isKindOfClass:_ASDisplayView.class], @"ASCellNode views must be of type _ASDisplayView");
|
|
[(_ASDisplayView *)self.view __forwardTouchesMoved:touches withEvent:event];
|
|
}
|
|
|
|
- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
ASDisplayNodeAssert([self.view isKindOfClass:_ASDisplayView.class], @"ASCellNode views must be of type _ASDisplayView");
|
|
[(_ASDisplayView *)self.view __forwardTouchesEnded:touches withEvent:event];
|
|
}
|
|
|
|
- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
ASDisplayNodeAssert([self.view isKindOfClass:_ASDisplayView.class], @"ASCellNode views must be of type _ASDisplayView");
|
|
[(_ASDisplayView *)self.view __forwardTouchesCancelled:touches withEvent:event];
|
|
}
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
- (void)setLayoutAttributes:(UICollectionViewLayoutAttributes *)layoutAttributes
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
if (ASObjectIsEqual(layoutAttributes, _layoutAttributes) == NO) {
|
|
_layoutAttributes = layoutAttributes;
|
|
if (layoutAttributes != nil) {
|
|
[self applyLayoutAttributes:layoutAttributes];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)applyLayoutAttributes:(UICollectionViewLayoutAttributes *)layoutAttributes
|
|
{
|
|
// To be overriden by subclasses
|
|
}
|
|
|
|
- (void)cellNodeVisibilityEvent:(ASCellNodeVisibilityEvent)event inScrollView:(UIScrollView *)scrollView withCellFrame:(CGRect)cellFrame
|
|
{
|
|
// To be overriden by subclasses
|
|
}
|
|
|
|
- (void)didEnterVisibleState
|
|
{
|
|
[super didEnterVisibleState];
|
|
if (self.neverShowPlaceholders) {
|
|
[self recursivelyEnsureDisplaySynchronously:YES];
|
|
}
|
|
[self handleVisibilityChange:YES];
|
|
}
|
|
|
|
- (void)didExitVisibleState
|
|
{
|
|
[super didExitVisibleState];
|
|
[self handleVisibilityChange:NO];
|
|
}
|
|
|
|
+ (BOOL)requestsVisibilityNotifications
|
|
{
|
|
static NSCache<Class, NSNumber *> *cache;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
cache = [[NSCache alloc] init];
|
|
});
|
|
NSNumber *result = [cache objectForKey:self];
|
|
if (result == nil) {
|
|
BOOL overrides = ASSubclassOverridesSelector([ASCellNode class], self, @selector(cellNodeVisibilityEvent:inScrollView:withCellFrame:));
|
|
result = overrides ? (NSNumber *)kCFBooleanTrue : (NSNumber *)kCFBooleanFalse;
|
|
[cache setObject:result forKey:self];
|
|
}
|
|
return (result == (NSNumber *)kCFBooleanTrue);
|
|
}
|
|
|
|
- (void)handleVisibilityChange:(BOOL)isVisible
|
|
{
|
|
if ([self.class requestsVisibilityNotifications] == NO) {
|
|
return; // The work below is expensive, and only valuable for subclasses watching visibility events.
|
|
}
|
|
|
|
// NOTE: This assertion is failing in some apps and will be enabled soon.
|
|
// ASDisplayNodeAssert(self.isNodeLoaded, @"Node should be loaded in order for it to become visible or invisible. If not in this situation, we shouldn't trigger creating the view.");
|
|
|
|
UIView *view = self.view;
|
|
CGRect cellFrame = CGRectZero;
|
|
|
|
// Ensure our _scrollView is still valid before converting. It's also possible that we have already been removed from the _scrollView,
|
|
// in which case it is not valid to perform a convertRect (this actually crashes on iOS 8).
|
|
UIScrollView *scrollView = (_scrollView != nil && view.superview != nil && [view isDescendantOfView:_scrollView]) ? _scrollView : nil;
|
|
if (scrollView) {
|
|
cellFrame = [view convertRect:view.bounds toView:_scrollView];
|
|
}
|
|
|
|
// If we did not convert, we'll pass along CGRectZero and a nil scrollView. The EventInvisible call is thus equivalent to
|
|
// didExitVisibileState, but is more convenient for the developer than implementing multiple methods.
|
|
[self cellNodeVisibilityEvent:isVisible ? ASCellNodeVisibilityEventVisible
|
|
: ASCellNodeVisibilityEventInvisible
|
|
inScrollView:scrollView
|
|
withCellFrame:cellFrame];
|
|
}
|
|
|
|
- (NSMutableArray<NSDictionary *> *)propertiesForDebugDescription
|
|
{
|
|
NSMutableArray *result = [super propertiesForDebugDescription];
|
|
|
|
UIScrollView *scrollView = self.scrollView;
|
|
|
|
ASDisplayNode *owningNode = scrollView.asyncdisplaykit_node;
|
|
if ([owningNode isKindOfClass:[ASCollectionNode class]]) {
|
|
NSIndexPath *ip = [(ASCollectionNode *)owningNode indexPathForNode:self];
|
|
if (ip != nil) {
|
|
[result addObject:@{ @"indexPath" : ip }];
|
|
}
|
|
[result addObject:@{ @"collectionNode" : owningNode }];
|
|
} else if ([owningNode isKindOfClass:[ASTableNode class]]) {
|
|
NSIndexPath *ip = [(ASTableNode *)owningNode indexPathForNode:self];
|
|
if (ip != nil) {
|
|
[result addObject:@{ @"indexPath" : ip }];
|
|
}
|
|
[result addObject:@{ @"tableNode" : owningNode }];
|
|
|
|
} else if ([scrollView isKindOfClass:[ASCollectionView class]]) {
|
|
NSIndexPath *ip = [(ASCollectionView *)scrollView indexPathForNode:self];
|
|
if (ip != nil) {
|
|
[result addObject:@{ @"indexPath" : ip }];
|
|
}
|
|
[result addObject:@{ @"collectionView" : ASObjectDescriptionMakeTiny(scrollView) }];
|
|
|
|
} else if ([scrollView isKindOfClass:[ASTableView class]]) {
|
|
NSIndexPath *ip = [(ASTableView *)scrollView indexPathForNode:self];
|
|
if (ip != nil) {
|
|
[result addObject:@{ @"indexPath" : ip }];
|
|
}
|
|
[result addObject:@{ @"tableView" : ASObjectDescriptionMakeTiny(scrollView) }];
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
- (NSString *)supplementaryElementKind
|
|
{
|
|
return self.collectionElement.supplementaryElementKind;
|
|
}
|
|
|
|
- (BOOL)supportsLayerBacking
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
#pragma mark -
|
|
#pragma mark ASTextCellNode
|
|
|
|
@implementation ASTextCellNode
|
|
|
|
static const CGFloat kASTextCellNodeDefaultFontSize = 18.0f;
|
|
static const CGFloat kASTextCellNodeDefaultHorizontalPadding = 15.0f;
|
|
static const CGFloat kASTextCellNodeDefaultVerticalPadding = 11.0f;
|
|
|
|
- (instancetype)init
|
|
{
|
|
return [self initWithAttributes:[ASTextCellNode defaultTextAttributes] insets:[ASTextCellNode defaultTextInsets]];
|
|
}
|
|
|
|
- (instancetype)initWithAttributes:(NSDictionary *)textAttributes insets:(UIEdgeInsets)textInsets
|
|
{
|
|
self = [super init];
|
|
if (self) {
|
|
_textInsets = textInsets;
|
|
_textAttributes = [textAttributes copy];
|
|
_textNode = [[ASTextNode alloc] init];
|
|
self.automaticallyManagesSubnodes = YES;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (ASLayoutSpec *)layoutSpecThatFits:(ASSizeRange)constrainedSize
|
|
{
|
|
return [ASInsetLayoutSpec insetLayoutSpecWithInsets:self.textInsets child:self.textNode];
|
|
}
|
|
|
|
+ (NSDictionary *)defaultTextAttributes
|
|
{
|
|
return @{NSFontAttributeName : [UIFont systemFontOfSize:kASTextCellNodeDefaultFontSize]};
|
|
}
|
|
|
|
+ (UIEdgeInsets)defaultTextInsets
|
|
{
|
|
return UIEdgeInsetsMake(kASTextCellNodeDefaultVerticalPadding, kASTextCellNodeDefaultHorizontalPadding, kASTextCellNodeDefaultVerticalPadding, kASTextCellNodeDefaultHorizontalPadding);
|
|
}
|
|
|
|
- (void)setTextAttributes:(NSDictionary *)textAttributes
|
|
{
|
|
ASDisplayNodeAssertNotNil(textAttributes, @"Invalid text attributes");
|
|
|
|
_textAttributes = [textAttributes copy];
|
|
|
|
[self updateAttributedText];
|
|
}
|
|
|
|
- (void)setTextInsets:(UIEdgeInsets)textInsets
|
|
{
|
|
_textInsets = textInsets;
|
|
|
|
[self setNeedsLayout];
|
|
}
|
|
|
|
- (void)setText:(NSString *)text
|
|
{
|
|
if (ASObjectIsEqual(_text, text)) return;
|
|
|
|
_text = [text copy];
|
|
|
|
[self updateAttributedText];
|
|
}
|
|
|
|
- (void)updateAttributedText
|
|
{
|
|
if (_text == nil) {
|
|
_textNode.attributedText = nil;
|
|
return;
|
|
}
|
|
|
|
_textNode.attributedText = [[NSAttributedString alloc] initWithString:self.text attributes:self.textAttributes];
|
|
[self setNeedsLayout];
|
|
}
|
|
|
|
@end
|