mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-09-05 20:22:15 +00:00
- Introduce thread-safe ASEventLog - ASCollectionNode and ASTableNode share their event log with their ASDataController. The controller uses it to log change set submitting and finishing events. - ASCollectionNode and ASTableNode print their data source and delegate in their debug description.
759 lines
29 KiB
Objective-C
759 lines
29 KiB
Objective-C
//
|
|
// ASTableViewTests.m
|
|
// AsyncDisplayKit
|
|
//
|
|
// 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 <XCTest/XCTest.h>
|
|
|
|
#import "ASTableView.h"
|
|
#import "ASTableViewInternal.h"
|
|
#import "ASDisplayNode+Subclasses.h"
|
|
#import "ASChangeSetDataController.h"
|
|
#import "ASCellNode.h"
|
|
#import "ASTableNode.h"
|
|
#import "ASTableView+Undeprecated.h"
|
|
#import <JGMethodSwizzler/JGMethodSwizzler.h>
|
|
#import "ASXCTExtensions.h"
|
|
|
|
#define NumberOfSections 10
|
|
#define NumberOfRowsPerSection 20
|
|
#define NumberOfReloadIterations 50
|
|
|
|
@interface ASTestDataController : ASChangeSetDataController
|
|
@property (nonatomic) int numberOfAllNodesRelayouts;
|
|
@end
|
|
|
|
@implementation ASTestDataController
|
|
|
|
- (void)relayoutAllNodes
|
|
{
|
|
_numberOfAllNodesRelayouts++;
|
|
[super relayoutAllNodes];
|
|
}
|
|
|
|
@end
|
|
|
|
@interface UITableView (Testing)
|
|
// This will start recording all editing calls to UITableView
|
|
// into the provided array.
|
|
// Make sure to call [UITableView deswizzleInstanceMethods] to reset this.
|
|
+ (void)as_recordEditingCallsIntoArray:(NSMutableArray<NSString *> *)selectors;
|
|
@end
|
|
|
|
@interface ASTestTableView : ASTableView
|
|
@property (nonatomic, copy) void (^willDeallocBlock)(ASTableView *tableView);
|
|
@end
|
|
|
|
@implementation ASTestTableView
|
|
|
|
- (instancetype)__initWithFrame:(CGRect)frame style:(UITableViewStyle)style
|
|
{
|
|
|
|
return [super _initWithFrame:frame style:style dataControllerClass:[ASTestDataController class] eventLog:[[ASEventLog alloc] init]];
|
|
}
|
|
|
|
- (ASTestDataController *)testDataController
|
|
{
|
|
return (ASTestDataController *)self.dataController;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
if (_willDeallocBlock) {
|
|
_willDeallocBlock(self);
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@interface ASTableViewTestDelegate : NSObject <ASTableDataSource, ASTableDelegate>
|
|
@property (nonatomic, copy) void (^willDeallocBlock)(ASTableViewTestDelegate *delegate);
|
|
@end
|
|
|
|
@implementation ASTableViewTestDelegate
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
- (ASCellNode *)tableView:(ASTableView *)tableView nodeForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
- (ASCellNodeBlock)tableView:(ASTableView *)tableView nodeBlockForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
if (_willDeallocBlock) {
|
|
_willDeallocBlock(self);
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@interface ASTestTextCellNode : ASTextCellNode
|
|
/** Calculated by counting how many times -layoutSpecThatFits: is called on the main thread. */
|
|
@property (nonatomic) int numberOfLayoutsOnMainThread;
|
|
@end
|
|
|
|
@implementation ASTestTextCellNode
|
|
|
|
- (ASLayoutSpec *)layoutSpecThatFits:(ASSizeRange)constrainedSize
|
|
{
|
|
if ([NSThread isMainThread]) {
|
|
_numberOfLayoutsOnMainThread++;
|
|
}
|
|
return [super layoutSpecThatFits:constrainedSize];
|
|
}
|
|
|
|
@end
|
|
|
|
@interface ASTableViewFilledDataSource : NSObject <ASTableDataSource, ASTableDelegate>
|
|
@property (nonatomic) BOOL usesSectionIndex;
|
|
@end
|
|
|
|
@implementation ASTableViewFilledDataSource
|
|
|
|
- (BOOL)respondsToSelector:(SEL)aSelector
|
|
{
|
|
if (aSelector == @selector(sectionIndexTitlesForTableView:) || aSelector == @selector(tableView:sectionForSectionIndexTitle:atIndex:)) {
|
|
return _usesSectionIndex;
|
|
} else {
|
|
return [super respondsToSelector:aSelector];
|
|
}
|
|
}
|
|
|
|
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
|
|
{
|
|
return NumberOfSections;
|
|
}
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
|
|
{
|
|
return NumberOfRowsPerSection;
|
|
}
|
|
|
|
- (ASCellNode *)tableView:(ASTableView *)tableView nodeForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
ASTestTextCellNode *textCellNode = [ASTestTextCellNode new];
|
|
textCellNode.text = indexPath.description;
|
|
|
|
return textCellNode;
|
|
}
|
|
|
|
- (ASCellNodeBlock)tableView:(ASTableView *)tableView nodeBlockForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
return ^{
|
|
ASTestTextCellNode *textCellNode = [ASTestTextCellNode new];
|
|
textCellNode.text = indexPath.description;
|
|
return textCellNode;
|
|
};
|
|
}
|
|
|
|
- (nullable NSArray<NSString *> *)sectionIndexTitlesForTableView:(UITableView *)tableView
|
|
{
|
|
return @[ @"A", @"B", @"C" ];
|
|
}
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView sectionForSectionIndexTitle:(NSString *)title atIndex:(NSInteger)index
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
@end
|
|
|
|
@interface ASTableViewFilledDelegate : NSObject <ASTableDelegate>
|
|
@end
|
|
|
|
@implementation ASTableViewFilledDelegate
|
|
|
|
- (ASSizeRange)tableView:(ASTableView *)tableView constrainedSizeForRowAtIndexPath:(NSIndexPath *)indexPath
|
|
{
|
|
return ASSizeRangeMake(CGSizeMake(10, 42));
|
|
}
|
|
|
|
@end
|
|
|
|
@interface ASTableViewTests : XCTestCase
|
|
@property (nonatomic, retain) ASTableView *testTableView;
|
|
@end
|
|
|
|
@implementation ASTableViewTests
|
|
|
|
- (void)testDataSourceImplementsNecessaryMethods
|
|
{
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, 100, 400)
|
|
style:UITableViewStylePlain];
|
|
|
|
|
|
|
|
ASTableViewFilledDataSource *dataSource = (ASTableViewFilledDataSource *)[NSObject new];
|
|
XCTAssertThrows((tableView.asyncDataSource = dataSource));
|
|
|
|
dataSource = [ASTableViewFilledDataSource new];
|
|
XCTAssertNoThrow((tableView.asyncDataSource = dataSource));
|
|
}
|
|
|
|
- (void)testConstrainedSizeForRowAtIndexPath
|
|
{
|
|
// Initial width of the table view is non-zero and all nodes are measured with this size.
|
|
// Any subsequence size change must trigger a relayout.
|
|
// Width and height are swapped so that a later size change will simulate a rotation
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, 100, 400)
|
|
style:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDelegate *delegate = [ASTableViewFilledDelegate new];
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = delegate;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
[tableView reloadDataImmediately];
|
|
[tableView setNeedsLayout];
|
|
[tableView layoutIfNeeded];
|
|
|
|
for (int section = 0; section < NumberOfSections; section++) {
|
|
for (int row = 0; row < NumberOfRowsPerSection; row++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
CGRect rect = [tableView rectForRowAtIndexPath:indexPath];
|
|
XCTAssertEqual(rect.size.width, 100); // specified width should be ignored for table
|
|
XCTAssertEqual(rect.size.height, 42);
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: Convert this to ARC.
|
|
- (void)DISABLED_testTableViewDoesNotRetainItselfAndDelegate
|
|
{
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectZero style:UITableViewStylePlain];
|
|
|
|
__block BOOL tableViewDidDealloc = NO;
|
|
tableView.willDeallocBlock = ^(ASTableView *v){
|
|
tableViewDidDealloc = YES;
|
|
};
|
|
|
|
ASTableViewTestDelegate *delegate = [[ASTableViewTestDelegate alloc] init];
|
|
|
|
__block BOOL delegateDidDealloc = NO;
|
|
delegate.willDeallocBlock = ^(ASTableViewTestDelegate *d){
|
|
delegateDidDealloc = YES;
|
|
};
|
|
|
|
tableView.asyncDataSource = delegate;
|
|
tableView.asyncDelegate = delegate;
|
|
|
|
// [delegate release];
|
|
XCTAssertTrue(delegateDidDealloc, @"unexpected delegate lifetime:%@", delegate);
|
|
|
|
// XCTAssertNoThrow([tableView release], @"unexpected exception when deallocating table view:%@", tableView);
|
|
XCTAssertTrue(tableViewDidDealloc, @"unexpected table view lifetime:%@", tableView);
|
|
}
|
|
|
|
- (NSIndexSet *)randomIndexSet
|
|
{
|
|
NSInteger randA = arc4random_uniform(NumberOfSections - 1);
|
|
NSInteger randB = arc4random_uniform(NumberOfSections - 1);
|
|
|
|
return [NSIndexSet indexSetWithIndexesInRange:NSMakeRange(MIN(randA, randB), MAX(randA, randB) - MIN(randA, randB))];
|
|
}
|
|
|
|
- (NSArray *)randomIndexPathsExisting:(BOOL)existing
|
|
{
|
|
NSMutableArray *indexPaths = [NSMutableArray array];
|
|
[[self randomIndexSet] enumerateIndexesUsingBlock:^(NSUInteger idx, BOOL *stop) {
|
|
NSUInteger rowNum = NumberOfRowsPerSection;
|
|
NSIndexPath *sectionIndex = [[NSIndexPath alloc] initWithIndex:idx];
|
|
for (NSUInteger i = (existing ? 0 : rowNum); i < (existing ? rowNum : rowNum * 2); i++) {
|
|
// Maximize evility by sporadically skipping indicies 1/3rd of the time, but only if reloading existing rows
|
|
if (existing && arc4random_uniform(2) == 0) {
|
|
continue;
|
|
}
|
|
|
|
NSIndexPath *indexPath = [sectionIndex indexPathByAddingIndex:i];
|
|
[indexPaths addObject:indexPath];
|
|
}
|
|
}];
|
|
return indexPaths;
|
|
}
|
|
|
|
- (void)DISABLED_testReloadData
|
|
{
|
|
// Keep the viewport moderately sized so that new cells are loaded on scrolling
|
|
ASTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, 100, 500)
|
|
style:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
XCTestExpectation *reloadDataExpectation = [self expectationWithDescription:@"reloadData"];
|
|
|
|
[tableView reloadDataWithCompletion:^{
|
|
NSLog(@"*** Reload Complete ***");
|
|
[reloadDataExpectation fulfill];
|
|
}];
|
|
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
|
|
for (int i = 0; i < NumberOfReloadIterations; ++i) {
|
|
UITableViewRowAnimation rowAnimation = (arc4random_uniform(2) == 0 ? UITableViewRowAnimationMiddle : UITableViewRowAnimationNone);
|
|
BOOL animatedScroll = (arc4random_uniform(2) == 0 ? YES : NO);
|
|
BOOL reloadRowsInsteadOfSections = (arc4random_uniform(2) == 0 ? YES : NO);
|
|
NSTimeInterval runLoopDelay = ((arc4random_uniform(2) == 0) ? (1.0 / (1 + arc4random_uniform(500))) : 0);
|
|
BOOL useBeginEndUpdates = (arc4random_uniform(3) == 0 ? YES : NO);
|
|
|
|
// instrument our instrumentation ;)
|
|
//NSLog(@"Iteration %03d: %@|%@|%@|%@|%g", i, (rowAnimation == UITableViewRowAnimationNone) ? @"NONE " : @"MIDDLE", animatedScroll ? @"ASCR" : @" ", reloadRowsInsteadOfSections ? @"ROWS" : @"SECS", useBeginEndUpdates ? @"BEGEND" : @" ", runLoopDelay);
|
|
|
|
if (useBeginEndUpdates) {
|
|
[tableView beginUpdates];
|
|
}
|
|
|
|
if (reloadRowsInsteadOfSections) {
|
|
NSArray *indexPaths = [self randomIndexPathsExisting:YES];
|
|
//NSLog(@"reloading rows: %@", indexPaths);
|
|
[tableView reloadRowsAtIndexPaths:indexPaths withRowAnimation:rowAnimation];
|
|
} else {
|
|
NSIndexSet *sections = [self randomIndexSet];
|
|
//NSLog(@"reloading sections: %@", sections);
|
|
[tableView reloadSections:sections withRowAnimation:rowAnimation];
|
|
}
|
|
|
|
[tableView setContentOffset:CGPointMake(0, arc4random_uniform(tableView.contentSize.height - tableView.bounds.size.height)) animated:animatedScroll];
|
|
|
|
if (runLoopDelay > 0) {
|
|
// Run other stuff on the main queue for between 2ms and 1000ms.
|
|
[[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:runLoopDelay]];
|
|
}
|
|
|
|
if (useBeginEndUpdates) {
|
|
[tableView endUpdates];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)testRelayoutAllNodesWithNonZeroSizeInitially
|
|
{
|
|
// Initial width of the table view is non-zero and all nodes are measured with this size.
|
|
// Any subsequence size change must trigger a relayout.
|
|
CGSize tableViewFinalSize = CGSizeMake(100, 500);
|
|
// Width and height are swapped so that a later size change will simulate a rotation
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, tableViewFinalSize.height, tableViewFinalSize.width)
|
|
style:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
[tableView layoutIfNeeded];
|
|
|
|
XCTAssertEqual(tableView.testDataController.numberOfAllNodesRelayouts, 0);
|
|
[self triggerSizeChangeAndAssertRelayoutAllNodesForTableView:tableView newSize:tableViewFinalSize];
|
|
}
|
|
|
|
- (void)testRelayoutAllNodesWithZeroSizeInitially
|
|
{
|
|
// Initial width of the table view is 0. The first size change is part of the initial config.
|
|
// Any subsequence size change after that must trigger a relayout.
|
|
CGSize tableViewFinalSize = CGSizeMake(100, 500);
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectZero
|
|
style:UITableViewStylePlain];
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
// Initial configuration
|
|
UIView *superview = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 500, 500)];
|
|
[superview addSubview:tableView];
|
|
// Width and height are swapped so that a later size change will simulate a rotation
|
|
tableView.frame = CGRectMake(0, 0, tableViewFinalSize.height, tableViewFinalSize.width);
|
|
[tableView layoutIfNeeded];
|
|
|
|
XCTAssertEqual(tableView.testDataController.numberOfAllNodesRelayouts, 0);
|
|
[self triggerSizeChangeAndAssertRelayoutAllNodesForTableView:tableView newSize:tableViewFinalSize];
|
|
}
|
|
|
|
- (void)testRelayoutVisibleRowsWhenEditingModeIsChanged
|
|
{
|
|
CGSize tableViewSize = CGSizeMake(100, 500);
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, tableViewSize.width, tableViewSize.height)
|
|
style:UITableViewStylePlain];
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
[self triggerFirstLayoutMeasurementForTableView:tableView];
|
|
|
|
NSArray *visibleNodes = [tableView visibleNodes];
|
|
XCTAssertGreaterThan(visibleNodes.count, 0);
|
|
|
|
// Cause table view to enter editing mode.
|
|
// Visibile nodes should be re-measured on main thread with the new (smaller) content view width.
|
|
// Other nodes are untouched.
|
|
XCTestExpectation *relayoutAfterEnablingEditingExpectation = [self expectationWithDescription:@"relayoutAfterEnablingEditing"];
|
|
[tableView beginUpdates];
|
|
[tableView setEditing:YES];
|
|
[tableView endUpdatesAnimated:YES completion:^(BOOL completed) {
|
|
for (int section = 0; section < NumberOfSections; section++) {
|
|
for (int row = 0; row < NumberOfRowsPerSection; row++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
ASTestTextCellNode *node = (ASTestTextCellNode *)[tableView nodeForRowAtIndexPath:indexPath];
|
|
if ([visibleNodes containsObject:node]) {
|
|
XCTAssertEqual(node.numberOfLayoutsOnMainThread, 1);
|
|
XCTAssertLessThan(node.constrainedSizeForCalculatedLayout.max.width, tableViewSize.width);
|
|
} else {
|
|
XCTAssertEqual(node.numberOfLayoutsOnMainThread, 0);
|
|
XCTAssertEqual(node.constrainedSizeForCalculatedLayout.max.width, tableViewSize.width);
|
|
}
|
|
}
|
|
}
|
|
[relayoutAfterEnablingEditingExpectation fulfill];
|
|
}];
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
|
|
// Cause table view to leave editing mode.
|
|
// Visibile nodes should be re-measured again.
|
|
// All nodes should have max constrained width equals to the table view width.
|
|
XCTestExpectation *relayoutAfterDisablingEditingExpectation = [self expectationWithDescription:@"relayoutAfterDisablingEditing"];
|
|
[tableView beginUpdates];
|
|
[tableView setEditing:NO];
|
|
[tableView endUpdatesAnimated:YES completion:^(BOOL completed) {
|
|
for (int section = 0; section < NumberOfSections; section++) {
|
|
for (int row = 0; row < NumberOfRowsPerSection; row++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
ASTestTextCellNode *node = (ASTestTextCellNode *)[tableView nodeForRowAtIndexPath:indexPath];
|
|
BOOL visible = [visibleNodes containsObject:node];
|
|
XCTAssertEqual(node.numberOfLayoutsOnMainThread, visible ? 2: 0);
|
|
XCTAssertEqual(node.constrainedSizeForCalculatedLayout.max.width, tableViewSize.width);
|
|
}
|
|
}
|
|
[relayoutAfterDisablingEditingExpectation fulfill];
|
|
}];
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (void)DISABLED_testRelayoutRowsAfterEditingModeIsChangedAndTheyBecomeVisible
|
|
{
|
|
CGSize tableViewSize = CGSizeMake(100, 500);
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] __initWithFrame:CGRectMake(0, 0, tableViewSize.width, tableViewSize.height)
|
|
style:UITableViewStylePlain];
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
[self triggerFirstLayoutMeasurementForTableView:tableView];
|
|
|
|
// Cause table view to enter editing mode and then scroll to the bottom.
|
|
// The last node should be re-measured on main thread with the new (smaller) content view width.
|
|
NSIndexPath *lastRowIndexPath = [NSIndexPath indexPathForRow:(NumberOfRowsPerSection - 1) inSection:(NumberOfSections - 1)];
|
|
XCTestExpectation *relayoutExpectation = [self expectationWithDescription:@"relayout"];
|
|
[tableView beginUpdates];
|
|
[tableView setEditing:YES];
|
|
[tableView setContentOffset:CGPointMake(0, CGFLOAT_MAX) animated:YES];
|
|
[tableView endUpdatesAnimated:YES completion:^(BOOL completed) {
|
|
ASTestTextCellNode *node = (ASTestTextCellNode *)[tableView nodeForRowAtIndexPath:lastRowIndexPath];
|
|
XCTAssertEqual(node.numberOfLayoutsOnMainThread, 1);
|
|
XCTAssertLessThan(node.constrainedSizeForCalculatedLayout.max.width, tableViewSize.width);
|
|
[relayoutExpectation fulfill];
|
|
}];
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (void)testIndexPathForNode
|
|
{
|
|
CGSize tableViewSize = CGSizeMake(100, 500);
|
|
ASTestTableView *tableView = [[ASTestTableView alloc] initWithFrame:CGRectMake(0, 0, tableViewSize.width, tableViewSize.height)
|
|
style:UITableViewStylePlain];
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
|
|
tableView.asyncDelegate = dataSource;
|
|
tableView.asyncDataSource = dataSource;
|
|
|
|
[tableView reloadDataWithCompletion:^{
|
|
for (NSUInteger i = 0; i < NumberOfSections; i++) {
|
|
for (NSUInteger j = 0; j < NumberOfRowsPerSection; j++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:j inSection:i];
|
|
ASCellNode *cellNode = [tableView nodeForRowAtIndexPath:indexPath];
|
|
NSIndexPath *reportedIndexPath = [tableView indexPathForNode:cellNode];
|
|
XCTAssertEqual(indexPath.row, reportedIndexPath.row);
|
|
}
|
|
}
|
|
self.testTableView = nil;
|
|
}];
|
|
}
|
|
|
|
- (void)triggerFirstLayoutMeasurementForTableView:(ASTableView *)tableView{
|
|
XCTestExpectation *reloadDataExpectation = [self expectationWithDescription:@"reloadData"];
|
|
[tableView reloadDataWithCompletion:^{
|
|
for (int section = 0; section < NumberOfSections; section++) {
|
|
for (int row = 0; row < NumberOfRowsPerSection; row++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
ASTestTextCellNode *node = (ASTestTextCellNode *)[tableView nodeForRowAtIndexPath:indexPath];
|
|
XCTAssertEqual(node.numberOfLayoutsOnMainThread, 0);
|
|
XCTAssertEqual(node.constrainedSizeForCalculatedLayout.max.width, tableView.frame.size.width);
|
|
}
|
|
}
|
|
[reloadDataExpectation fulfill];
|
|
}];
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (void)triggerSizeChangeAndAssertRelayoutAllNodesForTableView:(ASTestTableView *)tableView newSize:(CGSize)newSize
|
|
{
|
|
XCTestExpectation *nodesMeasuredUsingNewConstrainedSizeExpectation = [self expectationWithDescription:@"nodesMeasuredUsingNewConstrainedSize"];
|
|
|
|
[tableView beginUpdates];
|
|
|
|
CGRect frame = tableView.frame;
|
|
frame.size = newSize;
|
|
tableView.frame = frame;
|
|
[tableView layoutIfNeeded];
|
|
|
|
[tableView endUpdatesAnimated:NO completion:^(BOOL completed) {
|
|
XCTAssertEqual(tableView.testDataController.numberOfAllNodesRelayouts, 1);
|
|
|
|
for (int section = 0; section < NumberOfSections; section++) {
|
|
for (int row = 0; row < NumberOfRowsPerSection; row++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:section];
|
|
ASTestTextCellNode *node = (ASTestTextCellNode *)[tableView nodeForRowAtIndexPath:indexPath];
|
|
XCTAssertLessThanOrEqual(node.numberOfLayoutsOnMainThread, 1);
|
|
XCTAssertEqual(node.constrainedSizeForCalculatedLayout.max.width, newSize.width);
|
|
}
|
|
}
|
|
[nodesMeasuredUsingNewConstrainedSizeExpectation fulfill];
|
|
}];
|
|
[self waitForExpectationsWithTimeout:5 handler:^(NSError *error) {
|
|
if (error) {
|
|
XCTFail(@"Expectation failed: %@", error);
|
|
}
|
|
}];
|
|
}
|
|
|
|
/**
|
|
* This may seem silly, but we had issues where the runtime sometimes wouldn't correctly report
|
|
* conformances declared on categories.
|
|
*/
|
|
- (void)testThatTableNodeConformsToExpectedProtocols
|
|
{
|
|
ASTableNode *node = [[ASTableNode alloc] initWithStyle:UITableViewStylePlain];
|
|
XCTAssert([node conformsToProtocol:@protocol(ASRangeControllerUpdateRangeProtocol)]);
|
|
}
|
|
|
|
- (void)testThatInitialDataLoadHappensInOneShot
|
|
{
|
|
NSMutableArray *selectors = [NSMutableArray array];
|
|
ASTableNode *node = [[ASTableNode alloc] initWithStyle:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
node.frame = CGRectMake(0, 0, 100, 100);
|
|
|
|
node.dataSource = dataSource;
|
|
node.delegate = dataSource;
|
|
|
|
[UITableView as_recordEditingCallsIntoArray:selectors];
|
|
XCTAssertGreaterThan(node.numberOfSections, 0);
|
|
[node waitUntilAllUpdatesAreCommitted];
|
|
XCTAssertGreaterThan(node.view.numberOfSections, 0);
|
|
|
|
// Assert that the beginning of the call pattern is correct.
|
|
// There is currently noise that comes after that we will allow for this test.
|
|
NSArray *expectedSelectors = @[ NSStringFromSelector(@selector(reloadData)),
|
|
NSStringFromSelector(@selector(insertSections:withRowAnimation:)),
|
|
NSStringFromSelector(@selector(insertRowsAtIndexPaths:withRowAnimation:))];
|
|
NSArray *firstSelectors = [selectors subarrayWithRange:NSMakeRange(0, expectedSelectors.count)];
|
|
XCTAssertEqualObjects(firstSelectors, expectedSelectors);
|
|
|
|
[UITableView deswizzleAllInstanceMethods];
|
|
}
|
|
|
|
- (void)testThatReloadDataHappensInOneShot
|
|
{
|
|
NSMutableArray *selectors = [NSMutableArray array];
|
|
ASTableNode *node = [[ASTableNode alloc] initWithStyle:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
node.frame = CGRectMake(0, 0, 100, 100);
|
|
|
|
node.dataSource = dataSource;
|
|
node.delegate = dataSource;
|
|
|
|
// Load initial data.
|
|
XCTAssertGreaterThan(node.numberOfSections, 0);
|
|
[node waitUntilAllUpdatesAreCommitted];
|
|
XCTAssertGreaterThan(node.view.numberOfSections, 0);
|
|
|
|
// Reload data.
|
|
[UITableView as_recordEditingCallsIntoArray:selectors];
|
|
[node reloadData];
|
|
[node waitUntilAllUpdatesAreCommitted];
|
|
|
|
// Assert that the beginning of the call pattern is correct.
|
|
// There is currently noise that comes after that we will allow for this test.
|
|
NSArray *expectedSelectors = @[NSStringFromSelector(@selector(reloadData)),
|
|
NSStringFromSelector(@selector(beginUpdates)),
|
|
NSStringFromSelector(@selector(deleteSections:withRowAnimation:)),
|
|
NSStringFromSelector(@selector(insertSections:withRowAnimation:)),
|
|
NSStringFromSelector(@selector(endUpdates)),
|
|
NSStringFromSelector(@selector(insertRowsAtIndexPaths:withRowAnimation:))];
|
|
NSArray *firstSelectors = [selectors subarrayWithRange:NSMakeRange(0, expectedSelectors.count)];
|
|
XCTAssertEqualObjects(firstSelectors, expectedSelectors);
|
|
|
|
[UITableView deswizzleAllInstanceMethods];
|
|
}
|
|
|
|
/**
|
|
* This tests an issue where, if the table is loaded before the first layout pass,
|
|
* the nodes are first measured with a constrained width of 0 which isn't ideal.
|
|
*/
|
|
- (void)testThatNodeConstrainedSizesAreCorrectIfReloadIsPreempted
|
|
{
|
|
ASTableNode *node = [[ASTableNode alloc] initWithStyle:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
CGFloat cellWidth = 320;
|
|
node.frame = CGRectMake(0, 0, cellWidth, 480);
|
|
|
|
node.dataSource = dataSource;
|
|
node.delegate = dataSource;
|
|
|
|
// Trigger data load BEFORE first layout pass, to ensure constrained size is correct.
|
|
XCTAssertGreaterThan(node.numberOfSections, 0);
|
|
[node waitUntilAllUpdatesAreCommitted];
|
|
|
|
ASSizeRange expectedSizeRange = ASSizeRangeMake(CGSizeMake(cellWidth, 0));
|
|
expectedSizeRange.max.height = CGFLOAT_MAX;
|
|
|
|
for (NSInteger i = 0; i < node.numberOfSections; i++) {
|
|
for (NSInteger j = 0; j < [node numberOfRowsInSection:i]; j++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForItem:j inSection:i];
|
|
ASTestTextCellNode *cellNode = (id)[node nodeForRowAtIndexPath:indexPath];
|
|
ASXCTAssertEqualSizeRanges(cellNode.constrainedSizeForCalculatedLayout, expectedSizeRange);
|
|
XCTAssertEqual(cellNode.numberOfLayoutsOnMainThread, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)testSectionIndexHandling
|
|
{
|
|
ASTableNode *node = [[ASTableNode alloc] initWithStyle:UITableViewStylePlain];
|
|
|
|
ASTableViewFilledDataSource *dataSource = [ASTableViewFilledDataSource new];
|
|
dataSource.usesSectionIndex = YES;
|
|
node.frame = CGRectMake(0, 0, 320, 480);
|
|
|
|
node.dataSource = dataSource;
|
|
node.delegate = dataSource;
|
|
|
|
// Trigger data load
|
|
XCTAssertGreaterThan(node.numberOfSections, 0);
|
|
XCTAssertGreaterThan([node numberOfRowsInSection:0], 0);
|
|
[node waitUntilAllUpdatesAreCommitted];
|
|
|
|
UITableViewCell *cell = [node.view cellForRowAtIndexPath:[NSIndexPath indexPathForItem:0 inSection:0]];
|
|
XCTAssertNotNil(cell);
|
|
|
|
CGFloat cellWidth = cell.contentView.frame.size.width;
|
|
XCTAssert(cellWidth > 0 && cellWidth < 320, @"Expected cell width to be about 305. Width: %@", @(cellWidth));
|
|
|
|
ASSizeRange expectedSizeRange = ASSizeRangeMake(CGSizeMake(cellWidth, 0));
|
|
expectedSizeRange.max.height = CGFLOAT_MAX;
|
|
|
|
for (NSInteger i = 0; i < node.numberOfSections; i++) {
|
|
for (NSInteger j = 0; j < [node numberOfRowsInSection:i]; j++) {
|
|
NSIndexPath *indexPath = [NSIndexPath indexPathForItem:j inSection:i];
|
|
ASTestTextCellNode *cellNode = (id)[node nodeForRowAtIndexPath:indexPath];
|
|
ASXCTAssertEqualSizeRanges(cellNode.constrainedSizeForCalculatedLayout, expectedSizeRange);
|
|
// We will have to accept a relayout on main thread, since the index bar won't show
|
|
// up until some of the cells are inserted.
|
|
XCTAssertLessThanOrEqual(cellNode.numberOfLayoutsOnMainThread, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation UITableView (Testing)
|
|
|
|
+ (void)as_recordEditingCallsIntoArray:(NSMutableArray<NSString *> *)selectors
|
|
{
|
|
[UITableView swizzleInstanceMethod:@selector(reloadData) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *) {
|
|
JGOriginalImplementation(void);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(beginUpdates) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *) {
|
|
JGOriginalImplementation(void);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(endUpdates) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *) {
|
|
JGOriginalImplementation(void);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(insertRowsAtIndexPaths:withRowAnimation:) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *, NSArray *indexPaths, UITableViewRowAnimation anim) {
|
|
JGOriginalImplementation(void, indexPaths, anim);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(deleteRowsAtIndexPaths:withRowAnimation:) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *, NSArray *indexPaths, UITableViewRowAnimation anim) {
|
|
JGOriginalImplementation(void, indexPaths, anim);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(insertSections:withRowAnimation:) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *, NSIndexSet *indexes, UITableViewRowAnimation anim) {
|
|
JGOriginalImplementation(void, indexes, anim);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
[UITableView swizzleInstanceMethod:@selector(deleteSections:withRowAnimation:) withReplacement:JGMethodReplacementProviderBlock {
|
|
return JGMethodReplacement(void, UITableView *, NSIndexSet *indexes, UITableViewRowAnimation anim) {
|
|
JGOriginalImplementation(void, indexes, anim);
|
|
[selectors addObject:NSStringFromSelector(_cmd)];
|
|
};
|
|
}];
|
|
}
|
|
|
|
@end
|