react-native
Version:
A framework for building native apps using React
175 lines (151 loc) • 4.9 kB
JavaScript
/**
* Copyright (c) Meta Platforms, Inc. and affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @flow
* @format
*/
;
import type {PlatformConfig} from '../AnimatedPlatformConfig';
import {validateStyles} from '../../../src/private/animated/NativeAnimatedValidation';
import * as ReactNativeFeatureFlags from '../../../src/private/featureflags/ReactNativeFeatureFlags';
import flattenStyle from '../../StyleSheet/flattenStyle';
import Platform from '../../Utilities/Platform';
import AnimatedNode from './AnimatedNode';
import AnimatedObject from './AnimatedObject';
import AnimatedTransform from './AnimatedTransform';
import AnimatedWithChildren from './AnimatedWithChildren';
function createAnimatedStyle(
inputStyle: {[string]: mixed},
keepUnanimatedValues: boolean,
): [$ReadOnlyArray<string>, $ReadOnlyArray<AnimatedNode>, Object] {
const nodeKeys: Array<string> = [];
const nodes: Array<AnimatedNode> = [];
const style: {[string]: any} = {};
const keys = Object.keys(inputStyle);
for (let ii = 0, length = keys.length; ii < length; ii++) {
const key = keys[ii];
const value = inputStyle[key];
if (value != null && key === 'transform') {
const node = ReactNativeFeatureFlags.shouldUseAnimatedObjectForTransform()
? AnimatedObject.from(value)
: // $FlowFixMe[incompatible-call] - `value` is mixed.
new AnimatedTransform(value);
if (node == null) {
if (keepUnanimatedValues) {
style[key] = value;
}
} else {
nodeKeys.push(key);
nodes.push(node);
style[key] = node;
}
} else if (value instanceof AnimatedNode) {
const node = value;
nodeKeys.push(key);
nodes.push(node);
style[key] = value;
} else {
const node = AnimatedObject.from(value);
if (node == null) {
if (keepUnanimatedValues) {
style[key] = value;
}
} else {
nodeKeys.push(key);
nodes.push(node);
style[key] = node;
}
}
}
return [nodeKeys, nodes, style];
}
export default class AnimatedStyle extends AnimatedWithChildren {
#nodeKeys: $ReadOnlyArray<string>;
#nodes: $ReadOnlyArray<AnimatedNode>;
_inputStyle: any;
_style: {[string]: any};
constructor(inputStyle: any) {
super();
this._inputStyle = inputStyle;
const [nodeKeys, nodes, style] = createAnimatedStyle(
// NOTE: This null check should not be necessary, but the types are not
// strong nor enforced as of this writing. This check should be hoisted
// to instantiation sites.
flattenStyle(inputStyle) ?? {},
Platform.OS !== 'web',
);
this.#nodeKeys = nodeKeys;
this.#nodes = nodes;
this._style = style;
}
__getValue(): Object | Array<Object> {
const style: {[string]: any} = {};
const keys = Object.keys(this._style);
for (let ii = 0, length = keys.length; ii < length; ii++) {
const key = keys[ii];
const value = this._style[key];
if (value instanceof AnimatedNode) {
style[key] = value.__getValue();
} else {
style[key] = value;
}
}
return Platform.OS === 'web' ? [this._inputStyle, style] : style;
}
__getAnimatedValue(): Object {
const style: {[string]: any} = {};
const nodeKeys = this.#nodeKeys;
const nodes = this.#nodes;
for (let ii = 0, length = nodes.length; ii < length; ii++) {
const key = nodeKeys[ii];
const node = nodes[ii];
style[key] = node.__getAnimatedValue();
}
return style;
}
__attach(): void {
const nodes = this.#nodes;
for (let ii = 0, length = nodes.length; ii < length; ii++) {
const node = nodes[ii];
node.__addChild(this);
}
}
__detach(): void {
const nodes = this.#nodes;
for (let ii = 0, length = nodes.length; ii < length; ii++) {
const node = nodes[ii];
node.__removeChild(this);
}
super.__detach();
}
__makeNative(platformConfig: ?PlatformConfig) {
const nodes = this.#nodes;
for (let ii = 0, length = nodes.length; ii < length; ii++) {
const node = nodes[ii];
node.__makeNative(platformConfig);
}
super.__makeNative(platformConfig);
}
__getNativeConfig(): Object {
const platformConfig = this.__getPlatformConfig();
const styleConfig: {[string]: ?number} = {};
const nodeKeys = this.#nodeKeys;
const nodes = this.#nodes;
for (let ii = 0, length = nodes.length; ii < length; ii++) {
const key = nodeKeys[ii];
const node = nodes[ii];
node.__makeNative(platformConfig);
styleConfig[key] = node.__getNativeTag();
}
if (__DEV__) {
validateStyles(styleConfig);
}
return {
type: 'style',
style: styleConfig,
};
}
}