Peter 9bc996374f Add 'submodules/AsyncDisplayKit/' from commit '02bedc12816e251ad71777f9d2578329b6d2bef6'
git-subtree-dir: submodules/AsyncDisplayKit
git-subtree-mainline: d06f423e0ed3df1fed9bd10d79ee312a9179b632
git-subtree-split: 02bedc12816e251ad71777f9d2578329b6d2bef6
2019-06-11 18:42:43 +01:00

274 lines
12 KiB
Plaintext

//
// ASDelegateProxy.mm
// Texture
//
// Copyright (c) Facebook, Inc. and its affiliates. All rights reserved.
// Changes after 4/13/2017 are: Copyright (c) Pinterest, Inc. All rights reserved.
// Licensed under Apache 2.0: http://www.apache.org/licenses/LICENSE-2.0
//
#ifndef MINIMAL_ASDK
#import <AsyncDisplayKit/ASDelegateProxy.h>
#import <AsyncDisplayKit/ASTableNode.h>
#import <AsyncDisplayKit/ASCollectionNode.h>
#import <AsyncDisplayKit/ASAssert.h>
// UIKit performs a class check for UIDataSourceModelAssociation protocol conformance rather than an instance check, so
// the implementation of conformsToProtocol: below never gets called. We need to declare the two as conforming to the protocol here, then
// we need to implement dummy methods to get rid of a compiler warning about not conforming to the protocol.
@interface ASTableViewProxy () <UIDataSourceModelAssociation>
@end
@interface ASCollectionViewProxy () <UIDataSourceModelAssociation>
@end
@interface ASDelegateProxy (UIDataSourceModelAssociationPrivate)
- (nullable NSString *)_modelIdentifierForElementAtIndexPath:(NSIndexPath *)indexPath inView:(UIView *)view;
- (nullable NSIndexPath *)_indexPathForElementWithModelIdentifier:(NSString *)identifier inView:(UIView *)view;
@end
@implementation ASTableViewProxy
- (BOOL)interceptsSelector:(SEL)selector
{
return (
// handled by ASTableView node<->cell machinery
selector == @selector(tableView:cellForRowAtIndexPath:) ||
selector == @selector(tableView:heightForRowAtIndexPath:) ||
// Selection, highlighting, menu
selector == @selector(tableView:willSelectRowAtIndexPath:) ||
selector == @selector(tableView:didSelectRowAtIndexPath:) ||
selector == @selector(tableView:willDeselectRowAtIndexPath:) ||
selector == @selector(tableView:didDeselectRowAtIndexPath:) ||
selector == @selector(tableView:shouldHighlightRowAtIndexPath:) ||
selector == @selector(tableView:didHighlightRowAtIndexPath:) ||
selector == @selector(tableView:didUnhighlightRowAtIndexPath:) ||
selector == @selector(tableView:shouldShowMenuForRowAtIndexPath:) ||
selector == @selector(tableView:canPerformAction:forRowAtIndexPath:withSender:) ||
selector == @selector(tableView:performAction:forRowAtIndexPath:withSender:) ||
// handled by ASRangeController
selector == @selector(numberOfSectionsInTableView:) ||
selector == @selector(tableView:numberOfRowsInSection:) ||
// reordering support
selector == @selector(tableView:canMoveRowAtIndexPath:) ||
selector == @selector(tableView:moveRowAtIndexPath:toIndexPath:) ||
// used for ASCellNode visibility
selector == @selector(scrollViewDidScroll:) ||
// used for ASCellNode user interaction
selector == @selector(scrollViewWillBeginDragging:) ||
selector == @selector(scrollViewDidEndDragging:willDecelerate:) ||
// used for ASRangeController visibility updates
selector == @selector(tableView:willDisplayCell:forRowAtIndexPath:) ||
selector == @selector(tableView:didEndDisplayingCell:forRowAtIndexPath:) ||
// used for batch fetching API
selector == @selector(scrollViewWillEndDragging:withVelocity:targetContentOffset:) ||
selector == @selector(scrollViewDidEndDecelerating:) ||
// UIDataSourceModelAssociation
selector == @selector(modelIdentifierForElementAtIndexPath:inView:) ||
selector == @selector(indexPathForElementWithModelIdentifier:inView:)
);
}
- (nullable NSString *)modelIdentifierForElementAtIndexPath:(NSIndexPath *)indexPath inView:(UIView *)view {
return [self _modelIdentifierForElementAtIndexPath:indexPath inView:view];
}
- (nullable NSIndexPath *)indexPathForElementWithModelIdentifier:(NSString *)identifier inView:(UIView *)view {
return [self _indexPathForElementWithModelIdentifier:identifier inView:view];
}
@end
@implementation ASCollectionViewProxy
- (BOOL)interceptsSelector:(SEL)selector
{
return (
// handled by ASCollectionView node<->cell machinery
selector == @selector(collectionView:cellForItemAtIndexPath:) ||
selector == @selector(collectionView:layout:sizeForItemAtIndexPath:) ||
selector == @selector(collectionView:layout:insetForSectionAtIndex:) ||
selector == @selector(collectionView:layout:minimumLineSpacingForSectionAtIndex:) ||
selector == @selector(collectionView:layout:minimumInteritemSpacingForSectionAtIndex:) ||
selector == @selector(collectionView:layout:referenceSizeForHeaderInSection:) ||
selector == @selector(collectionView:layout:referenceSizeForFooterInSection:) ||
selector == @selector(collectionView:viewForSupplementaryElementOfKind:atIndexPath:) ||
// Selection, highlighting, menu
selector == @selector(collectionView:shouldSelectItemAtIndexPath:) ||
selector == @selector(collectionView:didSelectItemAtIndexPath:) ||
selector == @selector(collectionView:shouldDeselectItemAtIndexPath:) ||
selector == @selector(collectionView:didDeselectItemAtIndexPath:) ||
selector == @selector(collectionView:shouldHighlightItemAtIndexPath:) ||
selector == @selector(collectionView:didHighlightItemAtIndexPath:) ||
selector == @selector(collectionView:didUnhighlightItemAtIndexPath:) ||
selector == @selector(collectionView:shouldShowMenuForItemAtIndexPath:) ||
selector == @selector(collectionView:canPerformAction:forItemAtIndexPath:withSender:) ||
selector == @selector(collectionView:performAction:forItemAtIndexPath:withSender:) ||
// Item counts
selector == @selector(numberOfSectionsInCollectionView:) ||
selector == @selector(collectionView:numberOfItemsInSection:) ||
// Element appearance callbacks
selector == @selector(collectionView:willDisplayCell:forItemAtIndexPath:) ||
selector == @selector(collectionView:didEndDisplayingCell:forItemAtIndexPath:) ||
selector == @selector(collectionView:willDisplaySupplementaryView:forElementKind:atIndexPath:) ||
selector == @selector(collectionView:didEndDisplayingSupplementaryView:forElementOfKind:atIndexPath:) ||
// used for batch fetching API
selector == @selector(scrollViewWillEndDragging:withVelocity:targetContentOffset:) ||
selector == @selector(scrollViewDidEndDecelerating:) ||
// used for ASCellNode visibility
selector == @selector(scrollViewDidScroll:) ||
// used for ASCellNode user interaction
selector == @selector(scrollViewWillBeginDragging:) ||
selector == @selector(scrollViewDidEndDragging:willDecelerate:) ||
// intercepted due to not being supported by ASCollectionView (prevent bugs caused by usage)
selector == @selector(collectionView:canMoveItemAtIndexPath:) ||
selector == @selector(collectionView:moveItemAtIndexPath:toIndexPath:) ||
// UIDataSourceModelAssociation
selector == @selector(modelIdentifierForElementAtIndexPath:inView:) ||
selector == @selector(indexPathForElementWithModelIdentifier:inView:)
);
}
- (nullable NSString *)modelIdentifierForElementAtIndexPath:(NSIndexPath *)indexPath inView:(UIView *)view {
return [self _modelIdentifierForElementAtIndexPath:indexPath inView:view];
}
- (nullable NSIndexPath *)indexPathForElementWithModelIdentifier:(NSString *)identifier inView:(UIView *)view {
return [self _indexPathForElementWithModelIdentifier:identifier inView:view];
}
@end
@implementation ASPagerNodeProxy
- (BOOL)interceptsSelector:(SEL)selector
{
return (
// handled by ASPagerDataSource node<->cell machinery
selector == @selector(collectionNode:nodeForItemAtIndexPath:) ||
selector == @selector(collectionNode:nodeBlockForItemAtIndexPath:) ||
selector == @selector(collectionNode:numberOfItemsInSection:) ||
selector == @selector(collectionNode:constrainedSizeForItemAtIndexPath:)
);
}
@end
@implementation ASDelegateProxy {
id <ASDelegateProxyInterceptor> __weak _interceptor;
id __weak _target;
}
- (instancetype)initWithTarget:(id)target interceptor:(id <ASDelegateProxyInterceptor>)interceptor
{
ASDisplayNodeAssert(interceptor, @"interceptor must not be nil");
_target = target ? : [NSNull null];
_interceptor = interceptor;
return self;
}
- (BOOL)conformsToProtocol:(Protocol *)aProtocol
{
id target = _target;
if (target) {
return [target conformsToProtocol:aProtocol];
} else {
return [super conformsToProtocol:aProtocol];
}
}
- (BOOL)respondsToSelector:(SEL)aSelector
{
if ([self interceptsSelector:aSelector]) {
return [_interceptor respondsToSelector:aSelector];
} else {
// Also return NO if _target has become nil due to zeroing weak reference (or placeholder initialization).
return [_target respondsToSelector:aSelector];
}
}
- (id)forwardingTargetForSelector:(SEL)aSelector
{
if ([self interceptsSelector:aSelector]) {
return _interceptor;
} else {
id target = _target;
if (target) {
return [target respondsToSelector:aSelector] ? target : nil;
} else {
// The _interceptor needs to be nilled out in this scenario. For that a strong reference needs to be created
// to be able to nil out the _interceptor but still let it know that the proxy target has deallocated
// We have to hold a strong reference to the interceptor as we have to nil it out and call the proxyTargetHasDeallocated
// The reason that the interceptor needs to be nilled out is that there maybe a change of a infinite loop, for example
// if a method will be called in the proxyTargetHasDeallocated: that again would trigger a whole new forwarding cycle
id <ASDelegateProxyInterceptor> interceptor = _interceptor;
_interceptor = nil;
[interceptor proxyTargetHasDeallocated:self];
return nil;
}
}
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
// Check for a compiled definition for the selector
NSMethodSignature *methodSignature = nil;
if ([self interceptsSelector:aSelector]) {
methodSignature = [[_interceptor class] instanceMethodSignatureForSelector:aSelector];
} else {
methodSignature = [[_target class] instanceMethodSignatureForSelector:aSelector];
}
// Unfortunately, in order to get this object to work properly, the use of a method which creates an NSMethodSignature
// from a C string. -methodSignatureForSelector is called when a compiled definition for the selector cannot be found.
// This is the place where we have to create our own dud NSMethodSignature. This is necessary because if this method
// returns nil, a selector not found exception is raised. The string argument to -signatureWithObjCTypes: outlines
// the return type and arguments to the message. To return a dud NSMethodSignature, pretty much any signature will
// suffice. Since the -forwardInvocation call will do nothing if the delegate does not respond to the selector,
// the dud NSMethodSignature simply gets us around the exception.
return methodSignature ?: [NSMethodSignature signatureWithObjCTypes:"@^v^c"];
}
- (void)forwardInvocation:(NSInvocation *)invocation
{
// If we are down here this means _interceptor and _target where nil. Just don't do anything to prevent a crash
}
- (BOOL)interceptsSelector:(SEL)selector
{
ASDisplayNodeAssert(NO, @"This method must be overridden by subclasses.");
return NO;
}
- (nullable NSString *)_modelIdentifierForElementAtIndexPath:(NSIndexPath *)indexPath inView:(UIView *)view {
return [(id)_interceptor modelIdentifierForElementAtIndexPath:indexPath inView:view];
}
- (nullable NSIndexPath *)_indexPathForElementWithModelIdentifier:(NSString *)identifier inView:(UIView *)view {
return [(id)_interceptor indexPathForElementWithModelIdentifier:identifier inView:view];
}
@end
#endif