mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-09-15 16:36:36 +00:00
* Separate dataSource & UIKit index spaces Beef up our supplementary node support Make the API way better Go nuts Add a unit test for UICollectionView's handling of reloadData inside batch updates Wrap indexPathForNode: in a cache Convert index paths in delegate methods Go back on table view Put collection view back Switch up the API Move most ASCollectionView API to ASCollectionNode Move most table logic over to ASTableNode Do the things More conversion work Keep on keepin' on Get table view delegate API done More porting Simplify Clear the delegate More cleanup Move more stuff around Remove pointless file Re-add some API Put back more API Use the right flag * Some cleanup * Remove incorrect comment * Tweak the API * Put back a couple methods * update example projects (note: ASCollectionView deprecation warnings expected) * change reloadDataWithCompletion:nil --> reloadData * Clean up rebase * Make deprecated numberOfItemsInSection methods optional * Use the right flag * Address nits * update ASDKTube, ASDKgram & ASViewController examples
412 lines
11 KiB
Plaintext
412 lines
11 KiB
Plaintext
//
|
|
// ASCollectionNode.mm
|
|
// AsyncDisplayKit
|
|
//
|
|
// Created by Scott Goodson on 9/5/15.
|
|
//
|
|
// 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 root directory of this source tree. An additional grant
|
|
// of patent rights can be found in the PATENTS file in the same directory.
|
|
//
|
|
|
|
#import "ASCollectionInternal.h"
|
|
#import "ASCollectionViewLayoutFacilitatorProtocol.h"
|
|
#import "ASCollectionNode.h"
|
|
#import "ASDisplayNode+Subclasses.h"
|
|
#import "ASEnvironmentInternal.h"
|
|
#import "ASInternalHelpers.h"
|
|
#import "ASCellNode+Internal.h"
|
|
#import "AsyncDisplayKit+Debug.h"
|
|
#import "ASSectionContext.h"
|
|
#import "ASCollectionDataController.h"
|
|
#import "ASCollectionView+Undeprecated.h"
|
|
|
|
#pragma mark - _ASCollectionPendingState
|
|
|
|
@interface _ASCollectionPendingState : NSObject
|
|
@property (weak, nonatomic) id <ASCollectionDelegate> delegate;
|
|
@property (weak, nonatomic) id <ASCollectionDataSource> dataSource;
|
|
@property (assign, nonatomic) ASLayoutRangeMode rangeMode;
|
|
@end
|
|
|
|
@implementation _ASCollectionPendingState
|
|
|
|
- (instancetype)init
|
|
{
|
|
self = [super init];
|
|
if (self) {
|
|
_rangeMode = ASLayoutRangeModeCount;
|
|
}
|
|
return self;
|
|
}
|
|
@end
|
|
|
|
// TODO: Add support for tuning parameters in the pending state
|
|
#if 0 // This is not used yet, but will provide a way to avoid creating the view to set range values.
|
|
@implementation _ASCollectionPendingState {
|
|
std::vector<std::vector<ASRangeTuningParameters>> _tuningParameters;
|
|
}
|
|
|
|
- (instancetype)init
|
|
{
|
|
self = [super init];
|
|
if (self) {
|
|
_tuningParameters = std::vector<std::vector<ASRangeTuningParameters>> (ASLayoutRangeModeCount, std::vector<ASRangeTuningParameters> (ASLayoutRangeTypeCount));
|
|
_rangeMode = ASLayoutRangeModeCount;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (ASRangeTuningParameters)tuningParametersForRangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
return [self tuningParametersForRangeMode:ASLayoutRangeModeFull rangeType:rangeType];
|
|
}
|
|
|
|
- (void)setTuningParameters:(ASRangeTuningParameters)tuningParameters forRangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
return [self setTuningParameters:tuningParameters forRangeMode:ASLayoutRangeModeFull rangeType:rangeType];
|
|
}
|
|
|
|
- (ASRangeTuningParameters)tuningParametersForRangeMode:(ASLayoutRangeMode)rangeMode rangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
ASDisplayNodeAssert(rangeMode < _tuningParameters.size() && rangeType < _tuningParameters[rangeMode].size(), @"Requesting a range that is OOB for the configured tuning parameters");
|
|
return _tuningParameters[rangeMode][rangeType];
|
|
}
|
|
|
|
- (void)setTuningParameters:(ASRangeTuningParameters)tuningParameters forRangeMode:(ASLayoutRangeMode)rangeMode rangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
ASDisplayNodeAssert(rangeMode < _tuningParameters.size() && rangeType < _tuningParameters[rangeMode].size(), @"Setting a range that is OOB for the configured tuning parameters");
|
|
_tuningParameters[rangeMode][rangeType] = tuningParameters;
|
|
}
|
|
|
|
@end
|
|
#endif
|
|
|
|
#pragma mark - ASCollectionNode
|
|
|
|
@interface ASCollectionNode ()
|
|
{
|
|
ASDN::RecursiveMutex _environmentStateLock;
|
|
}
|
|
@property (nonatomic) _ASCollectionPendingState *pendingState;
|
|
@end
|
|
|
|
@implementation ASCollectionNode
|
|
|
|
#pragma mark Lifecycle
|
|
|
|
- (instancetype)init
|
|
{
|
|
ASDISPLAYNODE_NOT_DESIGNATED_INITIALIZER();
|
|
UICollectionViewLayout *nilLayout = nil;
|
|
self = [self initWithCollectionViewLayout:nilLayout]; // Will throw an exception for lacking a UICV Layout.
|
|
return nil;
|
|
}
|
|
|
|
- (instancetype)initWithCollectionViewLayout:(UICollectionViewLayout *)layout
|
|
{
|
|
return [self initWithFrame:CGRectZero collectionViewLayout:layout];
|
|
}
|
|
|
|
- (instancetype)initWithFrame:(CGRect)frame collectionViewLayout:(UICollectionViewLayout *)layout
|
|
{
|
|
return [self initWithFrame:frame collectionViewLayout:layout layoutFacilitator:nil];
|
|
}
|
|
|
|
- (instancetype)initWithFrame:(CGRect)frame collectionViewLayout:(UICollectionViewLayout *)layout layoutFacilitator:(id<ASCollectionViewLayoutFacilitatorProtocol>)layoutFacilitator
|
|
{
|
|
ASDisplayNodeViewBlock collectionViewBlock = ^UIView *{
|
|
return [[ASCollectionView alloc] _initWithFrame:frame collectionViewLayout:layout layoutFacilitator:layoutFacilitator];
|
|
};
|
|
|
|
if (self = [super initWithViewBlock:collectionViewBlock]) {
|
|
return self;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
self.delegate = nil;
|
|
self.dataSource = nil;
|
|
}
|
|
|
|
#pragma mark ASDisplayNode
|
|
|
|
- (void)didLoad
|
|
{
|
|
[super didLoad];
|
|
|
|
ASCollectionView *view = self.view;
|
|
view.collectionNode = self;
|
|
|
|
if (_pendingState) {
|
|
_ASCollectionPendingState *pendingState = _pendingState;
|
|
self.pendingState = nil;
|
|
view.asyncDelegate = pendingState.delegate;
|
|
view.asyncDataSource = pendingState.dataSource;
|
|
|
|
if (pendingState.rangeMode != ASLayoutRangeModeCount) {
|
|
[view.rangeController updateCurrentRangeWithMode:pendingState.rangeMode];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (ASCollectionView *)view
|
|
{
|
|
return (ASCollectionView *)[super view];
|
|
}
|
|
|
|
- (void)clearContents
|
|
{
|
|
[super clearContents];
|
|
[self.rangeController clearContents];
|
|
}
|
|
|
|
- (void)clearFetchedData
|
|
{
|
|
[super clearFetchedData];
|
|
[self.rangeController clearFetchedData];
|
|
}
|
|
|
|
- (void)interfaceStateDidChange:(ASInterfaceState)newState fromState:(ASInterfaceState)oldState
|
|
{
|
|
[super interfaceStateDidChange:newState fromState:oldState];
|
|
[ASRangeController layoutDebugOverlayIfNeeded];
|
|
}
|
|
|
|
#if ASRangeControllerLoggingEnabled
|
|
- (void)didEnterVisibleState
|
|
{
|
|
[super didEnterVisibleState];
|
|
NSLog(@"%@ - visible: YES", self);
|
|
}
|
|
|
|
- (void)didExitVisibleState
|
|
{
|
|
[super didExitVisibleState];
|
|
NSLog(@"%@ - visible: NO", self);
|
|
}
|
|
#endif
|
|
|
|
#pragma mark Setter / Getter
|
|
|
|
// TODO: Implement this without the view.
|
|
- (ASCollectionDataController *)dataController
|
|
{
|
|
return (ASCollectionDataController *)self.view.dataController;
|
|
}
|
|
|
|
// TODO: Implement this without the view.
|
|
- (ASRangeController *)rangeController
|
|
{
|
|
return self.view.rangeController;
|
|
}
|
|
|
|
- (_ASCollectionPendingState *)pendingState
|
|
{
|
|
if (!_pendingState && ![self isNodeLoaded]) {
|
|
self.pendingState = [[_ASCollectionPendingState alloc] init];
|
|
}
|
|
ASDisplayNodeAssert(![self isNodeLoaded] || !_pendingState, @"ASCollectionNode should not have a pendingState once it is loaded");
|
|
return _pendingState;
|
|
}
|
|
|
|
- (void)setDelegate:(id <ASCollectionDelegate>)delegate
|
|
{
|
|
if ([self pendingState]) {
|
|
_pendingState.delegate = delegate;
|
|
} else {
|
|
ASDisplayNodeAssert([self isNodeLoaded], @"ASCollectionNode should be loaded if pendingState doesn't exist");
|
|
self.view.asyncDelegate = delegate;
|
|
}
|
|
}
|
|
|
|
- (id <ASCollectionDelegate>)delegate
|
|
{
|
|
if ([self pendingState]) {
|
|
return _pendingState.delegate;
|
|
} else {
|
|
return self.view.asyncDelegate;
|
|
}
|
|
}
|
|
|
|
- (void)setDataSource:(id <ASCollectionDataSource>)dataSource
|
|
{
|
|
if ([self pendingState]) {
|
|
_pendingState.dataSource = dataSource;
|
|
} else {
|
|
ASDisplayNodeAssert([self isNodeLoaded], @"ASCollectionNode should be loaded if pendingState doesn't exist");
|
|
self.view.asyncDataSource = dataSource;
|
|
}
|
|
}
|
|
|
|
- (id <ASCollectionDataSource>)dataSource
|
|
{
|
|
if ([self pendingState]) {
|
|
return _pendingState.dataSource;
|
|
} else {
|
|
return self.view.asyncDataSource;
|
|
}
|
|
}
|
|
|
|
#pragma mark - Range Tuning
|
|
|
|
- (ASRangeTuningParameters)tuningParametersForRangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
return [self.rangeController tuningParametersForRangeMode:ASLayoutRangeModeFull rangeType:rangeType];
|
|
}
|
|
|
|
- (void)setTuningParameters:(ASRangeTuningParameters)tuningParameters forRangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
[self.rangeController setTuningParameters:tuningParameters forRangeMode:ASLayoutRangeModeFull rangeType:rangeType];
|
|
}
|
|
|
|
- (ASRangeTuningParameters)tuningParametersForRangeMode:(ASLayoutRangeMode)rangeMode rangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
return [self.rangeController tuningParametersForRangeMode:rangeMode rangeType:rangeType];
|
|
}
|
|
|
|
- (void)setTuningParameters:(ASRangeTuningParameters)tuningParameters forRangeMode:(ASLayoutRangeMode)rangeMode rangeType:(ASLayoutRangeType)rangeType
|
|
{
|
|
return [self.rangeController setTuningParameters:tuningParameters forRangeMode:rangeMode rangeType:rangeType];
|
|
}
|
|
|
|
#pragma mark - Querying Data
|
|
|
|
- (NSInteger)numberOfItemsInSection:(NSInteger)section
|
|
{
|
|
return [self.dataController numberOfRowsInSection:section];
|
|
}
|
|
|
|
- (NSInteger)numberOfSections
|
|
{
|
|
return [self.dataController numberOfSections];
|
|
}
|
|
|
|
- (NSArray<__kindof ASCellNode *> *)visibleNodes
|
|
{
|
|
return [self.view visibleNodes];
|
|
}
|
|
|
|
- (NSIndexPath *)indexPathForNode:(ASCellNode *)cellNode
|
|
{
|
|
return [self.dataController indexPathForNode:cellNode];
|
|
}
|
|
|
|
- (ASCellNode *)nodeForItemAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
return [self.dataController nodeAtIndexPath:indexPath];
|
|
}
|
|
|
|
- (id<ASSectionContext>)contextForSection:(NSInteger)section
|
|
{
|
|
ASDisplayNodeAssertMainThread();
|
|
return [self.dataController contextForSection:section];
|
|
}
|
|
|
|
#pragma mark - Editing
|
|
|
|
- (void)registerSupplementaryNodeOfKind:(NSString *)elementKind
|
|
{
|
|
[self.view registerSupplementaryNodeOfKind:elementKind];
|
|
}
|
|
|
|
- (void)performBatchAnimated:(BOOL)animated updates:(void (^)())updates completion:(void (^)(BOOL))completion
|
|
{
|
|
[self.view performBatchAnimated:animated updates:updates completion:completion];
|
|
}
|
|
|
|
- (void)performBatchUpdates:(void (^)())updates completion:(void (^)(BOOL))completion
|
|
{
|
|
[self.view performBatchUpdates:updates completion:completion];
|
|
}
|
|
|
|
- (void)reloadDataWithCompletion:(void (^)())completion
|
|
{
|
|
[self.view reloadDataWithCompletion:completion];
|
|
}
|
|
|
|
- (void)reloadData
|
|
{
|
|
[self.view reloadData];
|
|
}
|
|
|
|
- (void)reloadDataImmediately
|
|
{
|
|
[self.view reloadDataImmediately];
|
|
}
|
|
|
|
- (void)beginUpdates
|
|
{
|
|
[self.dataController beginUpdates];
|
|
}
|
|
|
|
- (void)endUpdatesAnimated:(BOOL)animated
|
|
{
|
|
[self endUpdatesAnimated:animated completion:nil];
|
|
}
|
|
|
|
- (void)endUpdatesAnimated:(BOOL)animated completion:(void (^)(BOOL))completion
|
|
{
|
|
[self.dataController endUpdatesAnimated:animated completion:completion];
|
|
}
|
|
|
|
- (void)insertSections:(NSIndexSet *)sections
|
|
{
|
|
[self.view insertSections:sections];
|
|
}
|
|
|
|
- (void)deleteSections:(NSIndexSet *)sections
|
|
{
|
|
[self.view deleteSections:sections];
|
|
}
|
|
|
|
- (void)reloadSections:(NSIndexSet *)sections
|
|
{
|
|
[self.view reloadSections:sections];
|
|
}
|
|
|
|
- (void)moveSection:(NSInteger)section toSection:(NSInteger)newSection
|
|
{
|
|
[self.view moveSection:section toSection:newSection];
|
|
}
|
|
|
|
- (void)insertItemsAtIndexPaths:(NSArray *)indexPaths
|
|
{
|
|
[self.view insertItemsAtIndexPaths:indexPaths];
|
|
}
|
|
|
|
- (void)deleteItemsAtIndexPaths:(NSArray *)indexPaths
|
|
{
|
|
[self.view deleteItemsAtIndexPaths:indexPaths];
|
|
}
|
|
|
|
- (void)reloadItemsAtIndexPaths:(NSArray *)indexPaths
|
|
{
|
|
[self.view reloadItemsAtIndexPaths:indexPaths];
|
|
}
|
|
|
|
- (void)moveItemAtIndexPath:(NSIndexPath *)indexPath toIndexPath:(NSIndexPath *)newIndexPath
|
|
{
|
|
[self.view moveItemAtIndexPath:indexPath toIndexPath:newIndexPath];
|
|
}
|
|
|
|
#pragma mark - ASRangeControllerUpdateRangeProtocol
|
|
|
|
- (void)updateCurrentRangeWithMode:(ASLayoutRangeMode)rangeMode;
|
|
{
|
|
if ([self pendingState]) {
|
|
_pendingState.rangeMode = rangeMode;
|
|
} else {
|
|
[self.rangeController updateCurrentRangeWithMode:rangeMode];
|
|
}
|
|
}
|
|
|
|
#pragma mark ASEnvironment
|
|
|
|
ASEnvironmentCollectionTableSetEnvironmentState(_environmentStateLock)
|
|
|
|
@end
|