react-native/Libraries/NativeAnimation/RCTViewPropertyMapper.m
Ryan Gomba eb96b7fabc Maintain transform order
Summary:
This diff addresses the issues raised by kmagiera in https://github.com/facebook/react-native/pull/7884. Transforms should be applied in the order they are defined, just like in `processTransform.js`. A scale applied before a translation, for instance, should give a different result than a translation applied before a scale.

We leverage CATransform3D to do the heavy lifting. A concatenated transform is passed all the way to `RCTViewPropertyMapper`. It is compared with the transform currently applied to the view, and if different, applied. The same approach is used for opacity.

I think it makes the most sense to do this diffing in `RCTViewPropertyMapper`, as opposed to creating and cleaning up an `_updatedPropsDictionary` each frame in `RCTTransformAnimatedNode` and `RCTStyleAnimatedNode`. The node should keep its full value; applying a minimal set of altered props is an optimization.  The higher up this optimization is implemented, the more assumptions it makes. e.g. that there will only ever be a sing
Closes https://github.com/facebook/react-native/pull/9050

Differential Revision: D3658139

fbshipit-source-id: ad6286762ef734084cbdf83c9bd9241190302d34
2016-08-02 14:28:29 -07:00

61 lines
1.4 KiB
Objective-C

/**
* Copyright (c) 2015-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 "RCTViewPropertyMapper.h"
#import <UIKit/UIKit.h>
#import "RCTBridge.h"
#import "RCTConvert.h"
#import "RCTUIManager.h"
#import "RCTNativeAnimatedModule.h"
@implementation RCTViewPropertyMapper
{
RCTNativeAnimatedModule *_animationModule;
}
- (instancetype)initWithViewTag:(NSNumber *)viewTag
animationModule:(RCTNativeAnimatedModule *)animationModule
{
if ((self = [super init])) {
_animationModule = animationModule;
_viewTag = viewTag;
_animationModule = animationModule;
}
return self;
}
RCT_NOT_IMPLEMENTED(- (instancetype)init)
- (void)updateViewWithDictionary:(NSDictionary<NSString *, NSObject *> *)updates
{
if (!updates.count) {
return;
}
UIView *view = [_animationModule.bridge.uiManager viewForReactTag:_viewTag];
if (!view) {
return;
}
NSNumber *opacity = [RCTConvert NSNumber:updates[@"opacity"]];
if (opacity) {
view.alpha = opacity.floatValue;
}
NSObject *transform = updates[@"transform"];
if ([transform isKindOfClass:[NSValue class]]) {
view.layer.allowsEdgeAntialiasing = YES;
view.layer.transform = ((NSValue *)transform).CATransform3DValue;
}
}
@end