mirror of
https://github.com/status-im/react-native.git
synced 2025-01-22 15:29:07 +00:00
1d37dd063c
Summary: Adds unit tests to the Native Animated implementation on iOS. This pretty much mirrors the tests we currently have on Android. It also fixes 2 bugs I've found when adding the tests and pass the current time in `stepAnimation` instead of using `CACurrentMediaTime` to make testing easier. - `stopListeningToAnimatedNodeValue` did not actually work at all, it should set the listener to nil. - The finished value in the animation end callback was always true, this simplifies the `RCTAnimationDriver` interface to get rid of `removeAnimation` and fixes the end callback value. **Test plan** - Run the tests - Make sure the UIExplorer example still works Closes https://github.com/facebook/react-native/pull/13068 Differential Revision: D4786701 Pulled By: javache fbshipit-source-id: a4f07e6eec1f363ca47b6f27984041793c915bfc
98 lines
3.2 KiB
Objective-C
98 lines
3.2 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 "RCTAnimationUtils.h"
|
|
|
|
#import <React/RCTLog.h>
|
|
|
|
static NSUInteger _RCTFindIndexOfNearestValue(CGFloat value, NSArray<NSNumber *> *range)
|
|
{
|
|
NSUInteger index;
|
|
NSUInteger rangeCount = range.count;
|
|
for (index = 1; index < rangeCount - 1; index++) {
|
|
NSNumber *inputValue = range[index];
|
|
if (inputValue.doubleValue >= value) {
|
|
break;
|
|
}
|
|
}
|
|
return index - 1;
|
|
}
|
|
|
|
/**
|
|
* Interpolates value by remapping it linearly fromMin->fromMax to toMin->toMax
|
|
*/
|
|
CGFloat RCTInterpolateValue(CGFloat value,
|
|
CGFloat inputMin,
|
|
CGFloat inputMax,
|
|
CGFloat outputMin,
|
|
CGFloat outputMax,
|
|
NSString *extrapolateLeft,
|
|
NSString *extrapolateRight)
|
|
{
|
|
if (value < inputMin) {
|
|
if ([extrapolateLeft isEqualToString:EXTRAPOLATE_TYPE_IDENTITY]) {
|
|
return value;
|
|
} else if ([extrapolateLeft isEqualToString:EXTRAPOLATE_TYPE_CLAMP]) {
|
|
value = inputMin;
|
|
} else if ([extrapolateLeft isEqualToString:EXTRAPOLATE_TYPE_EXTEND]) {
|
|
// noop
|
|
} else {
|
|
RCTLogError(@"Invalid extrapolation type %@ for left extrapolation", extrapolateLeft);
|
|
}
|
|
}
|
|
|
|
if (value > inputMax) {
|
|
if ([extrapolateRight isEqualToString:EXTRAPOLATE_TYPE_IDENTITY]) {
|
|
return value;
|
|
} else if ([extrapolateRight isEqualToString:EXTRAPOLATE_TYPE_CLAMP]) {
|
|
value = inputMax;
|
|
} else if ([extrapolateRight isEqualToString:EXTRAPOLATE_TYPE_EXTEND]) {
|
|
// noop
|
|
} else {
|
|
RCTLogError(@"Invalid extrapolation type %@ for right extrapolation", extrapolateRight);
|
|
}
|
|
}
|
|
|
|
return outputMin + (value - inputMin) * (outputMax - outputMin) / (inputMax - inputMin);
|
|
}
|
|
|
|
/**
|
|
* Interpolates value by mapping it from the inputRange to the outputRange.
|
|
*/
|
|
CGFloat RCTInterpolateValueInRange(CGFloat value,
|
|
NSArray<NSNumber *> *inputRange,
|
|
NSArray<NSNumber *> *outputRange,
|
|
NSString *extrapolateLeft,
|
|
NSString *extrapolateRight)
|
|
{
|
|
NSUInteger rangeIndex = _RCTFindIndexOfNearestValue(value, inputRange);
|
|
CGFloat inputMin = inputRange[rangeIndex].doubleValue;
|
|
CGFloat inputMax = inputRange[rangeIndex + 1].doubleValue;
|
|
CGFloat outputMin = outputRange[rangeIndex].doubleValue;
|
|
CGFloat outputMax = outputRange[rangeIndex + 1].doubleValue;
|
|
|
|
return RCTInterpolateValue(value,
|
|
inputMin,
|
|
inputMax,
|
|
outputMin,
|
|
outputMax,
|
|
extrapolateLeft,
|
|
extrapolateRight);
|
|
}
|
|
|
|
CGFloat RCTRadiansToDegrees(CGFloat radians)
|
|
{
|
|
return radians * 180.0 / M_PI;
|
|
}
|
|
|
|
CGFloat RCTDegreesToRadians(CGFloat degrees)
|
|
{
|
|
return degrees / 180.0 * M_PI;
|
|
}
|