I'm using the Android spring animation in my project (see here). However, these animations are getting in the way of my espresso tests.
I already tried to disable these animations using the developer options in the phone, but they seem to not be affected by these settings.
Is there any way how I can disable them just for tests?
After struggling with a flaky test due to SpringAnimations I came up with three solutions:
Solution 1: Add a function that wraps creating your SpringAnimations
This is the most invasive in terms of changing existing code, but least complex method to follow:
You can check if animations are disabled at runtime:
Then selectively return a dummy animation that immediately finishes while also setting the value to it's final state:
Solution 2: Create an IdlingResource that tells Espresso if a DynamicAnimation is running
SpringAnimation
andFlingAnimation
both extend fromDynamicAnimation
, the class which is ignoring the system Animation Scale and causing issues here.This solution isn't the prettiest as it uses reflection, but the implementation details it relies on haven't changed since
DynamicAnimation
was introduced.Based on
DataBindingIdlingResource
:For convenience a matching test rule:
This isn't the perfect solution since it will still cause your tests to wait for animations despite changing the animation scale globally
If you have infinite animations based on SpringAnimations (by setting Damping to zero), this won't work as it'll always report to Espresso that an animation is running. You could work around that by casting the DynamicAnimation to a SpringAnimation and checking if Damping was set, but I felt like that's a rare enough case to not complicate things.
Solution 3: Force all SpringAnimations to skip to their last frame
Another reflection based solution, but this one completely disables the SpringAnimations. The trade-off is that theoretically Espresso can still try to interact in the 1 frame window between a SpringAnimation being asked to end, and it actually ending.
In practice I had to rerun the test hundreds of times in a row to get this to happen, at which point the animation may not even be the source of flakiness. So the trade-off is probably worth it if the animations are dragging down how long your tests take to complete:
Call this method in your
@Before
annotated function to have it run before each test.In the
SpringAnimation
implementation,skipToEnd
sets a flag that is not checked until the next call todoAnimationFrame
, hence theanimation.doAnimationFrame(100000L)
call.