Swiftgram/AsyncDisplayKit/Details/Transactions/_ASAsyncTransactionContainer.m
2014-06-26 22:32:55 -07:00

120 lines
4.4 KiB
Objective-C

/* 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 "_ASAsyncTransactionContainer+Private.h"
#import "_ASAsyncTransaction.h"
#import "_ASAsyncTransactionGroup.h"
@implementation CALayer (ASAsyncTransactionContainerTransactions)
@dynamic asyncdisplaykit_asyncLayerTransactions;
@dynamic asyncdisplaykit_currentAsyncLayerTransaction;
// No-ops in the base class. Mostly exposed for testing.
- (void)asyncdisplaykit_asyncTransactionContainerWillBeginTransaction:(_ASAsyncTransaction *)transaction {}
- (void)asyncdisplaykit_asyncTransactionContainerDidCompleteTransaction:(_ASAsyncTransaction *)transaction {}
@end
@implementation CALayer (ASDisplayNodeAsyncTransactionContainer)
@dynamic asyncdisplaykit_asyncTransactionContainer;
- (ASAsyncTransactionContainerState)asyncdisplaykit_asyncTransactionContainerState
{
return ([self.asyncdisplaykit_asyncLayerTransactions count] == 0) ? ASAsyncTransactionContainerStateNoTransactions : ASAsyncTransactionContainerStatePendingTransactions;
}
- (void)asyncdisplaykit_cancelAsyncTransactions
{
// If there was an open transaction, commit and clear the current transaction. Otherwise:
// (1) The run loop observer will try to commit a canceled transaction which is not allowed
// (2) We leave the canceled transaction attached to the layer, dooming future operations
_ASAsyncTransaction *currentTransaction = self.asyncdisplaykit_currentAsyncLayerTransaction;
[currentTransaction commit];
self.asyncdisplaykit_currentAsyncLayerTransaction = nil;
for (_ASAsyncTransaction *transaction in [self.asyncdisplaykit_asyncLayerTransactions copy]) {
[transaction cancel];
}
}
- (void)asyncdisplaykit_asyncTransactionContainerStateDidChange
{
id delegate = self.delegate;
if ([delegate respondsToSelector:@selector(asyncdisplaykit_asyncTransactionContainerStateDidChange)]) {
[delegate asyncdisplaykit_asyncTransactionContainerStateDidChange];
}
}
- (_ASAsyncTransaction *)asyncdisplaykit_asyncTransaction
{
_ASAsyncTransaction *transaction = self.asyncdisplaykit_currentAsyncLayerTransaction;
if (transaction == nil) {
NSHashTable *transactions = self.asyncdisplaykit_asyncLayerTransactions;
if (transactions == nil) {
transactions = [NSHashTable hashTableWithOptions:NSPointerFunctionsObjectPointerPersonality];
self.asyncdisplaykit_asyncLayerTransactions = transactions;
}
transaction = [[_ASAsyncTransaction alloc] initWithCallbackQueue:dispatch_get_main_queue() completionBlock:^(_ASAsyncTransaction *completedTransaction, BOOL cancelled) {
[transactions removeObject:completedTransaction];
[self asyncdisplaykit_asyncTransactionContainerDidCompleteTransaction:completedTransaction];
if ([transactions count] == 0) {
[self asyncdisplaykit_asyncTransactionContainerStateDidChange];
}
}];
[transactions addObject:transaction];
self.asyncdisplaykit_currentAsyncLayerTransaction = transaction;
[self asyncdisplaykit_asyncTransactionContainerWillBeginTransaction:transaction];
if ([transactions count] == 1) {
[self asyncdisplaykit_asyncTransactionContainerStateDidChange];
}
}
[[_ASAsyncTransactionGroup mainTransactionGroup] addTransactionContainer:self];
return transaction;
}
- (CALayer *)asyncdisplaykit_parentTransactionContainer
{
CALayer *containerLayer = self;
while (containerLayer && !containerLayer.asyncdisplaykit_isAsyncTransactionContainer) {
containerLayer = containerLayer.superlayer;
}
return containerLayer;
}
@end
@implementation UIView (ASDisplayNodeAsyncTransactionContainer)
- (BOOL)asyncdisplaykit_isAsyncTransactionContainer
{
return self.layer.asyncdisplaykit_isAsyncTransactionContainer;
}
- (void)asyncdisplaykit_setAsyncTransactionContainer:(BOOL)asyncTransactionContainer
{
self.layer.asyncdisplaykit_asyncTransactionContainer = asyncTransactionContainer;
}
- (ASAsyncTransactionContainerState)asyncdisplaykit_asyncTransactionContainerState
{
return self.layer.asyncdisplaykit_asyncTransactionContainerState;
}
- (void)asyncdisplaykit_cancelAsyncTransactions
{
[self.layer asyncdisplaykit_cancelAsyncTransactions];
}
- (void)asyncdisplaykit_asyncTransactionContainerStateDidChange
{
// No-op in the base class.
}
@end