mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
7b59c5a47e
Summary: Main change is to the property diffing - we now use the last known props set on the view rather than the default props to compute the diff. This requires exposing a `getProps` method on all view components which should be fine I think. I also realized that in more complex animations with multiple nodes, the node that the animation starts on might not be connected to a view, so we don't know if it's fabric just based on that, so we have to do a recursive search through the children to find if there are any that are associated with a fabric view to decide we should start the animation immediately. Unfortunately there can still be a timing gap here since the animated API is async and the uimanager API is sync - I'll need to change the animated API to be sync to completely fix this. Reviewed By: shergin Differential Revision: D14732028 fbshipit-source-id: 882c056b0b63aa576f8e42439be405cf7fb3147a
129 lines
2.5 KiB
Objective-C
129 lines
2.5 KiB
Objective-C
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#import "RCTAnimatedNode.h"
|
|
|
|
#import <React/RCTDefines.h>
|
|
|
|
@implementation RCTAnimatedNode
|
|
{
|
|
NSMapTable<NSNumber *, RCTAnimatedNode *> *_childNodes;
|
|
NSMapTable<NSNumber *, RCTAnimatedNode *> *_parentNodes;
|
|
}
|
|
|
|
- (instancetype)initWithTag:(NSNumber *)tag
|
|
config:(NSDictionary<NSString *, id> *)config
|
|
{
|
|
if ((self = [super init])) {
|
|
_nodeTag = tag;
|
|
_config = [config copy];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
RCT_NOT_IMPLEMENTED(- (instancetype)init)
|
|
|
|
- (NSMapTable<NSNumber *, RCTAnimatedNode *> *)childNodes
|
|
{
|
|
return _childNodes;
|
|
}
|
|
|
|
- (NSMapTable<NSNumber *, RCTAnimatedNode *> *)parentNodes
|
|
{
|
|
return _parentNodes;
|
|
}
|
|
|
|
- (void)addChild:(RCTAnimatedNode *)child
|
|
{
|
|
if (!_childNodes) {
|
|
_childNodes = [NSMapTable strongToWeakObjectsMapTable];
|
|
}
|
|
if (child) {
|
|
[_childNodes setObject:child forKey:child.nodeTag];
|
|
[child onAttachedToNode:self];
|
|
}
|
|
}
|
|
|
|
- (void)removeChild:(RCTAnimatedNode *)child
|
|
{
|
|
if (!_childNodes) {
|
|
return;
|
|
}
|
|
if (child) {
|
|
[_childNodes removeObjectForKey:child.nodeTag];
|
|
[child onDetachedFromNode:self];
|
|
}
|
|
}
|
|
|
|
- (void)onAttachedToNode:(RCTAnimatedNode *)parent
|
|
{
|
|
if (!_parentNodes) {
|
|
_parentNodes = [NSMapTable strongToWeakObjectsMapTable];
|
|
}
|
|
if (parent) {
|
|
[_parentNodes setObject:parent forKey:parent.nodeTag];
|
|
}
|
|
}
|
|
|
|
- (void)onDetachedFromNode:(RCTAnimatedNode *)parent
|
|
{
|
|
if (!_parentNodes) {
|
|
return;
|
|
}
|
|
if (parent) {
|
|
[_parentNodes removeObjectForKey:parent.nodeTag];
|
|
}
|
|
}
|
|
|
|
- (void)detachNode
|
|
{
|
|
for (RCTAnimatedNode *parent in _parentNodes.objectEnumerator) {
|
|
[parent removeChild:self];
|
|
}
|
|
for (RCTAnimatedNode *child in _childNodes.objectEnumerator) {
|
|
[self removeChild:child];
|
|
}
|
|
}
|
|
|
|
- (void)setNeedsUpdate
|
|
{
|
|
_needsUpdate = YES;
|
|
for (RCTAnimatedNode *child in _childNodes.objectEnumerator) {
|
|
[child setNeedsUpdate];
|
|
}
|
|
}
|
|
|
|
- (void)updateNodeIfNecessary
|
|
{
|
|
if (_needsUpdate) {
|
|
for (RCTAnimatedNode *parent in _parentNodes.objectEnumerator) {
|
|
[parent updateNodeIfNecessary];
|
|
}
|
|
[self performUpdate];
|
|
}
|
|
}
|
|
|
|
- (void)performUpdate
|
|
{
|
|
_needsUpdate = NO;
|
|
// To be overidden by subclasses
|
|
// This method is called on a node only if it has been marked for update
|
|
// during the current update loop
|
|
}
|
|
|
|
- (BOOL)isManagedByFabric
|
|
{
|
|
for (RCTAnimatedNode *child in _childNodes.objectEnumerator) {
|
|
if ([child isManagedByFabric]) {
|
|
return YES;
|
|
}
|
|
}
|
|
return NO;
|
|
}
|
|
|
|
@end
|