Skip to content

Commit

Permalink
Migrate DecayAnimation to Kotlin (#45719)
Browse files Browse the repository at this point in the history
Summary:
Pull Request resolved: #45719

# Changelog:
[Internal] -

As in the title.

Reviewed By: tdn120

Differential Revision: D60284323

fbshipit-source-id: 6cf47a9f9528067b397b0c52b763b416bf62ed9f
  • Loading branch information
rshest authored and facebook-github-bot committed Jul 28, 2024
1 parent e86a5ee commit 9e77602
Show file tree
Hide file tree
Showing 3 changed files with 75 additions and 83 deletions.

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
/*
* 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.
*/

package com.facebook.react.animated

import com.facebook.react.bridge.ReadableMap
import kotlin.math.abs
import kotlin.math.exp

/**
* Implementation of [AnimationDriver] providing support for decay animations. The implementation is
* copied from the JS version in `AnimatedImplementation.js`.
*/
internal class DecayAnimation(config: ReadableMap) : AnimationDriver() {
private var velocity: Double = 0.0
private var deceleration: Double = 0.0
private var startFrameTimeMillis: Long = -1
private var fromValue: Double = 0.0
private var lastValue: Double = 0.0
private var iterations: Int = 1
private var currentLoop: Int = 1

init {
resetConfig(config)
}

public override fun resetConfig(config: ReadableMap): Unit {
velocity = config.getDouble("velocity")
deceleration = config.getDouble("deceleration")
startFrameTimeMillis = -1
fromValue = 0.0
lastValue = 0.0
iterations = if (config.hasKey("iterations")) config.getInt("iterations") else 1
currentLoop = 1
mHasFinished = iterations == 0
}

public override fun runAnimationStep(frameTimeNanos: Long) {
val frameTimeMillis = frameTimeNanos / 1000000
if (startFrameTimeMillis == -1L) {
// since this is the first animation step, consider the start to be on the previous frame
startFrameTimeMillis = frameTimeMillis - 16
if (fromValue == lastValue) { // first iteration, assign mFromValue based on mAnimatedValue
fromValue = mAnimatedValue.nodeValue
} else { // not the first iteration, reset mAnimatedValue based on mFromValue
mAnimatedValue.nodeValue = fromValue
}
lastValue = mAnimatedValue.nodeValue
}
val value =
(fromValue +
velocity / (1 - deceleration) *
(1 - exp(-(1 - deceleration) * (frameTimeMillis - startFrameTimeMillis))))
if (abs(lastValue - value) < 0.1) {
if (iterations == -1 || currentLoop < iterations) { // looping animation, return to start
// set mStartFrameTimeMillis to -1 to reset instance variables on the next runAnimationStep
startFrameTimeMillis = -1
currentLoop++
} else { // animation has completed
mHasFinished = true
return
}
}
lastValue = value
mAnimatedValue.nodeValue = value
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -1120,7 +1120,7 @@ class NativeAnimatedNodeTraversalTest {
}

// at this point we expect tracking value to be at 75
assertThat((nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).value)
assertThat((nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).getValue())
.isEqualTo(75.0)

// we update "toValue" again to 100 and expect the animation to restart from the current
Expand Down Expand Up @@ -1228,11 +1228,11 @@ class NativeAnimatedNodeTraversalTest {
// passes the final point (that is 1) while going backwards
var isBoucingBack: Boolean = false
var previousValue: Double =
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).value
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).getValue()
for (i in 500 downTo 0) {
nativeAnimatedNodesManager.runUpdates(nextFrameTime())
val currentValue: Double =
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).value
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).getValue()
if (previousValue >= 1.0 && currentValue < 1.0) {
isBoucingBack = true
break
Expand All @@ -1253,7 +1253,7 @@ class NativeAnimatedNodeTraversalTest {
for (i in 0 until 8 * 60) {
nativeAnimatedNodesManager.runUpdates(nextFrameTime())
val currentValue: Double =
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).value
(nativeAnimatedNodesManager.getNodeById(3) as ValueAnimatedNode).getValue()
if (!hasTurnedForward) {
if (currentValue <= previousValue) {
bounceBackInitialFrames++
Expand Down

0 comments on commit 9e77602

Please sign in to comment.