2016-03-24 13:18:39 +00:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* @providesModule NativeAnimatedHelper
|
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-08-12 01:10:16 +00:00
|
|
|
const NativeAnimatedModule = require('NativeModules').NativeAnimatedModule;
|
|
|
|
const NativeEventEmitter = require('NativeEventEmitter');
|
2016-03-24 13:18:39 +00:00
|
|
|
|
2016-08-12 01:10:16 +00:00
|
|
|
const invariant = require('fbjs/lib/invariant');
|
2016-03-24 13:18:39 +00:00
|
|
|
|
2016-08-12 01:10:16 +00:00
|
|
|
let __nativeAnimatedNodeTagCount = 1; /* used for animated nodes */
|
|
|
|
let __nativeAnimationIdCount = 1; /* used for started animations */
|
2016-03-24 13:18:39 +00:00
|
|
|
|
2016-11-02 06:55:21 +00:00
|
|
|
type EndResult = {finished: boolean};
|
2016-03-24 13:18:39 +00:00
|
|
|
type EndCallback = (result: EndResult) => void;
|
2016-09-19 11:07:36 +00:00
|
|
|
type EventMapping = {
|
2016-11-02 06:55:21 +00:00
|
|
|
nativeEventPath: Array<string>,
|
|
|
|
animatedValueTag: number,
|
2016-09-19 11:07:36 +00:00
|
|
|
};
|
2016-03-24 13:18:39 +00:00
|
|
|
|
2016-08-12 01:10:16 +00:00
|
|
|
let nativeEventEmitter;
|
|
|
|
|
2016-03-24 13:18:39 +00:00
|
|
|
/**
|
2016-08-12 01:10:16 +00:00
|
|
|
* Simple wrappers around NativeAnimatedModule to provide flow and autocmplete support for
|
2016-03-24 13:18:39 +00:00
|
|
|
* the native module methods
|
|
|
|
*/
|
2016-08-12 01:10:16 +00:00
|
|
|
const API = {
|
2016-03-24 13:18:39 +00:00
|
|
|
createAnimatedNode: function(tag: number, config: Object): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.createAnimatedNode(tag, config);
|
|
|
|
},
|
2016-08-04 20:11:37 +00:00
|
|
|
startListeningToAnimatedNodeValue: function(tag: number) {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.startListeningToAnimatedNodeValue(tag);
|
|
|
|
},
|
|
|
|
stopListeningToAnimatedNodeValue: function(tag: number) {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.stopListeningToAnimatedNodeValue(tag);
|
|
|
|
},
|
2016-03-24 13:18:39 +00:00
|
|
|
connectAnimatedNodes: function(parentTag: number, childTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.connectAnimatedNodes(parentTag, childTag);
|
|
|
|
},
|
|
|
|
disconnectAnimatedNodes: function(parentTag: number, childTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.disconnectAnimatedNodes(parentTag, childTag);
|
|
|
|
},
|
2016-04-22 07:01:55 +00:00
|
|
|
startAnimatingNode: function(animationId: number, nodeTag: number, config: Object, endCallback: EndCallback): void {
|
2016-03-24 13:18:39 +00:00
|
|
|
assertNativeAnimatedModule();
|
2016-04-22 07:01:55 +00:00
|
|
|
NativeAnimatedModule.startAnimatingNode(animationId, nodeTag, config, endCallback);
|
|
|
|
},
|
|
|
|
stopAnimation: function(animationId: number) {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.stopAnimation(animationId);
|
2016-03-24 13:18:39 +00:00
|
|
|
},
|
|
|
|
setAnimatedNodeValue: function(nodeTag: number, value: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.setAnimatedNodeValue(nodeTag, value);
|
|
|
|
},
|
2016-09-26 17:17:29 +00:00
|
|
|
setAnimatedNodeOffset: function(nodeTag: number, offset: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.setAnimatedNodeOffset(nodeTag, offset);
|
|
|
|
},
|
|
|
|
flattenAnimatedNodeOffset: function(nodeTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.flattenAnimatedNodeOffset(nodeTag);
|
|
|
|
},
|
2016-11-08 04:36:52 +00:00
|
|
|
extractAnimatedNodeOffset: function(nodeTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.extractAnimatedNodeOffset(nodeTag);
|
|
|
|
},
|
2016-03-24 13:18:39 +00:00
|
|
|
connectAnimatedNodeToView: function(nodeTag: number, viewTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.connectAnimatedNodeToView(nodeTag, viewTag);
|
|
|
|
},
|
|
|
|
disconnectAnimatedNodeFromView: function(nodeTag: number, viewTag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.disconnectAnimatedNodeFromView(nodeTag, viewTag);
|
|
|
|
},
|
|
|
|
dropAnimatedNode: function(tag: number): void {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.dropAnimatedNode(tag);
|
|
|
|
},
|
2016-09-19 11:07:36 +00:00
|
|
|
addAnimatedEventToView: function(viewTag: number, eventName: string, eventMapping: EventMapping) {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.addAnimatedEventToView(viewTag, eventName, eventMapping);
|
|
|
|
},
|
|
|
|
removeAnimatedEventFromView(viewTag: number, eventName: string) {
|
|
|
|
assertNativeAnimatedModule();
|
|
|
|
NativeAnimatedModule.removeAnimatedEventFromView(viewTag, eventName);
|
|
|
|
}
|
2016-03-24 13:18:39 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Properties allowed by the native animated implementation.
|
|
|
|
*
|
|
|
|
* In general native animated implementation should support any numeric property that doesn't need
|
|
|
|
* to be updated through the shadow view hierarchy (all non-layout properties). This list is limited
|
|
|
|
* to the properties that will perform best when animated off the JS thread.
|
|
|
|
*/
|
2016-08-12 01:10:16 +00:00
|
|
|
const PROPS_WHITELIST = {
|
2016-03-24 13:18:39 +00:00
|
|
|
style: {
|
|
|
|
opacity: true,
|
2016-06-09 17:34:41 +00:00
|
|
|
transform: true,
|
2016-03-24 13:18:39 +00:00
|
|
|
/* legacy android transform properties */
|
|
|
|
scaleX: true,
|
|
|
|
scaleY: true,
|
|
|
|
translateX: true,
|
|
|
|
translateY: true,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2016-08-12 01:10:16 +00:00
|
|
|
const TRANSFORM_WHITELIST = {
|
2016-06-09 17:34:41 +00:00
|
|
|
translateX: true,
|
|
|
|
translateY: true,
|
|
|
|
scale: true,
|
2016-08-02 21:23:18 +00:00
|
|
|
scaleX: true,
|
|
|
|
scaleY: true,
|
2016-06-09 17:34:41 +00:00
|
|
|
rotate: true,
|
2016-08-02 21:23:18 +00:00
|
|
|
rotateX: true,
|
|
|
|
rotateY: true,
|
|
|
|
perspective: true,
|
2016-06-09 17:34:41 +00:00
|
|
|
};
|
|
|
|
|
2016-03-24 13:18:39 +00:00
|
|
|
function validateProps(params: Object): void {
|
|
|
|
for (var key in params) {
|
|
|
|
if (!PROPS_WHITELIST.hasOwnProperty(key)) {
|
|
|
|
throw new Error(`Property '${key}' is not supported by native animated module`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-02 21:23:18 +00:00
|
|
|
function validateTransform(configs: Array<Object>): void {
|
|
|
|
configs.forEach((config) => {
|
|
|
|
if (!TRANSFORM_WHITELIST.hasOwnProperty(config.property)) {
|
|
|
|
throw new Error(`Property '${config.property}' is not supported by native animated module`);
|
2016-06-09 17:34:41 +00:00
|
|
|
}
|
2016-08-02 21:23:18 +00:00
|
|
|
});
|
2016-06-09 17:34:41 +00:00
|
|
|
}
|
|
|
|
|
2016-03-24 13:18:39 +00:00
|
|
|
function validateStyles(styles: Object): void {
|
|
|
|
var STYLES_WHITELIST = PROPS_WHITELIST.style || {};
|
|
|
|
for (var key in styles) {
|
|
|
|
if (!STYLES_WHITELIST.hasOwnProperty(key)) {
|
|
|
|
throw new Error(`Style property '${key}' is not supported by native animated module`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-23 09:37:01 +00:00
|
|
|
function validateInterpolation(config: Object): void {
|
|
|
|
var SUPPORTED_INTERPOLATION_PARAMS = {
|
|
|
|
inputRange: true,
|
|
|
|
outputRange: true,
|
2016-09-06 22:16:16 +00:00
|
|
|
extrapolate: true,
|
|
|
|
extrapolateRight: true,
|
|
|
|
extrapolateLeft: true,
|
2016-04-23 09:37:01 +00:00
|
|
|
};
|
|
|
|
for (var key in config) {
|
|
|
|
if (!SUPPORTED_INTERPOLATION_PARAMS.hasOwnProperty(key)) {
|
|
|
|
throw new Error(`Interpolation property '${key}' is not supported by native animated module`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-24 13:18:39 +00:00
|
|
|
function generateNewNodeTag(): number {
|
|
|
|
return __nativeAnimatedNodeTagCount++;
|
|
|
|
}
|
|
|
|
|
2016-04-22 07:01:55 +00:00
|
|
|
function generateNewAnimationId(): number {
|
|
|
|
return __nativeAnimationIdCount++;
|
2016-03-24 13:18:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function assertNativeAnimatedModule(): void {
|
|
|
|
invariant(NativeAnimatedModule, 'Native animated module is not available');
|
|
|
|
}
|
|
|
|
|
2016-11-02 06:55:21 +00:00
|
|
|
function isNativeAnimatedAvailable(): boolean {
|
|
|
|
return !!NativeAnimatedModule;
|
|
|
|
}
|
|
|
|
|
2016-03-24 13:18:39 +00:00
|
|
|
module.exports = {
|
|
|
|
API,
|
|
|
|
validateProps,
|
|
|
|
validateStyles,
|
2016-06-09 17:34:41 +00:00
|
|
|
validateTransform,
|
2016-04-23 09:37:01 +00:00
|
|
|
validateInterpolation,
|
2016-03-24 13:18:39 +00:00
|
|
|
generateNewNodeTag,
|
2016-04-22 07:01:55 +00:00
|
|
|
generateNewAnimationId,
|
2016-03-24 13:18:39 +00:00
|
|
|
assertNativeAnimatedModule,
|
2016-11-02 06:55:21 +00:00
|
|
|
isNativeAnimatedAvailable,
|
2016-08-12 01:10:16 +00:00
|
|
|
get nativeEventEmitter() {
|
|
|
|
if (!nativeEventEmitter) {
|
|
|
|
nativeEventEmitter = new NativeEventEmitter(NativeAnimatedModule);
|
|
|
|
}
|
|
|
|
return nativeEventEmitter;
|
|
|
|
},
|
2016-03-24 13:18:39 +00:00
|
|
|
};
|