/* 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