Search code examples
iosuikitios10ios11uiviewpropertyanimator

UIViewPropertyAnimator different behaviour on iOS10 and iOS11 reversing an animation. Bug on `isReversed` and `fractionComplete` properties?


THE PROBLEM
Running the same code on iOS10 and iOS11 my UIViewPropertyAnimator has a different behaviour just after changing of his .isReversed property.

Everything is ok on iOS10. The animation problem happens on iOS11

CONDITIONS
It's true for any animations, not only for a particular one, and it is verifiable both by watching the animation and within the code. It happens both on simulators and real devices.

DETAILS
Once created a UIViewPropertyAnimator with his animation, during its running I just call .pauseAnimation() and change the .isReversed property to true. After that I resume the animation calling:

continueAnimation(withTimingParameters parameters: UITimingCurveProvider?, durationFactor: CGFloat)

at this point on iOS10 the animation smoothly changes his verse, on iOS11 it stops immediately and reverses itself with a bit frames lag.

If in code I check the value of .fractionComplete (called on my UIViewPropertyAnimator object it gives me back the completion of the animation in his percent value, starting from 0.0 and ending at 1.0) just after .continueAnimation(...

- On iOS 10 it remains for a few moments like if the animation is continuing and only after some fractions of time jumps to his complementary.

- On iOS 11 it jumps suddenly on his complementary


On the documentation there are non updates related to this, just a couple of new properties for the UIViewPropertyAnimator but not used because I'm targeting iOS10

Could be a bug or I'm missing something!?


Little update: just tested, same behaviour on iOS 11.0.1 and on iOS 11.1 beta1

As linked in the comment, this happens only with a non-linear curve!


Solution

  • I have been fighting this for quite a while as well, but then I noticed the scrubsLinearly property that was added to UIViewPropertyAnimator in iOS 11:

    Defaults to true. Provides the ability for an animator to pause and scrub either linearly or using the animator’s current timing.

    Note that the default of this property is true, which seems to cause a conflict with using a non-linear animation curve. That might also explain why the issue is not present when using a linear timing function.

    Setting scrubsLinearly to false, the animator seems to work as expected:

    let animator = UIViewPropertyAnimator(duration: 0.25, curve: .easeOut) {
       ...
    }
    animator.scrubsLinearly = false