blob: 395fed259a566c9fae7823982ff751dda16e35c9 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.launcher3.anim;
import static androidx.dynamicanimation.animation.FloatPropertyCompat.createFloatPropertyCompat;
import static com.android.launcher3.config.FeatureFlags.QUICKSTEP_SPRINGS;
import android.animation.Animator;
import android.animation.AnimatorListenerAdapter;
import android.animation.ObjectAnimator;
import android.animation.TimeInterpolator;
import android.animation.ValueAnimator;
import android.os.Handler;
import android.os.Looper;
import android.util.FloatProperty;
import android.util.Log;
import java.util.ArrayList;
import androidx.dynamicanimation.animation.DynamicAnimation.OnAnimationEndListener;
import androidx.dynamicanimation.animation.SpringAnimation;
import androidx.dynamicanimation.animation.SpringForce;
/**
* This animator allows for an object's property to be be controlled by an {@link ObjectAnimator} or
* a {@link SpringAnimation}. It extends ValueAnimator so it can be used in an AnimatorSet.
*/
public class SpringObjectAnimator<T> extends ValueAnimator {
private static final String TAG = "SpringObjectAnimator";
private static boolean DEBUG = false;
private ObjectAnimator mObjectAnimator;
private float[] mValues;
private SpringAnimation mSpring;
private SpringProperty<T> mProperty;
private ArrayList<AnimatorListener> mListeners;
private boolean mSpringEnded = true;
private boolean mAnimatorEnded = true;
private boolean mEnded = true;
public SpringObjectAnimator(T object, FloatProperty<T> property, float minimumVisibleChange,
float damping, float stiffness, float... values) {
mSpring = new SpringAnimation(object, createFloatPropertyCompat(property));
mSpring.setMinimumVisibleChange(minimumVisibleChange);
mSpring.setSpring(new SpringForce(0)
.setDampingRatio(damping)
.setStiffness(stiffness));
mSpring.setStartVelocity(0.01f);
mProperty = new SpringProperty<>(property, mSpring);
mObjectAnimator = ObjectAnimator.ofFloat(object, mProperty, values);
mValues = values;
mListeners = new ArrayList<>();
setFloatValues(values);
// We use this listener and track mListeners so that we can sync the animator and spring
// listeners.
mObjectAnimator.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationStart(Animator animation) {
mAnimatorEnded = false;
mEnded = false;
for (AnimatorListener l : mListeners) {
l.onAnimationStart(animation);
}
}
@Override
public void onAnimationEnd(Animator animation) {
mAnimatorEnded = true;
tryEnding();
}
@Override
public void onAnimationCancel(Animator animation) {
for (AnimatorListener l : mListeners) {
l.onAnimationCancel(animation);
}
mSpring.cancel();
}
});
mSpring.addUpdateListener((animation, value, velocity) -> mSpringEnded = false);
mSpring.addEndListener((animation, canceled, value, velocity) -> {
mSpringEnded = true;
tryEnding();
});
}
private void tryEnding() {
if (DEBUG) {
Log.d(TAG, "tryEnding#mAnimatorEnded=" + mAnimatorEnded + ", mSpringEnded="
+ mSpringEnded + ", mEnded=" + mEnded);
}
// If springs are disabled, ignore value of mSpringEnded
if (mAnimatorEnded && (mSpringEnded || !QUICKSTEP_SPRINGS.get()) && !mEnded) {
for (AnimatorListener l : mListeners) {
l.onAnimationEnd(this);
}
mEnded = true;
}
}
public SpringAnimation getSpring() {
return mSpring;
}
/**
* Initializes and sets up the spring to take over controlling the object.
*/
public void startSpring(float end, float velocity, OnAnimationEndListener endListener) {
// Cancel the spring so we can set new start velocity and final position. We need to remove
// the listener since the spring is not actually ending.
mSpring.removeEndListener(endListener);
mSpring.cancel();
mSpring.addEndListener(endListener);
mProperty.switchToSpring();
mSpring.setStartVelocity(velocity);
float startValue = end == 0 ? mValues[1] : mValues[0];
float endValue = end == 0 ? mValues[0] : mValues[1];
mSpring.setStartValue(startValue);
new Handler(Looper.getMainLooper()).postDelayed(() -> {
mSpring.animateToFinalPosition(endValue);
}, getStartDelay());
}
@Override
public void addListener(AnimatorListener listener) {
mListeners.add(listener);
}
public ArrayList<AnimatorListener> getObjectAnimatorListeners() {
return mObjectAnimator.getListeners();
}
@Override
public ArrayList<AnimatorListener> getListeners() {
return mListeners;
}
@Override
public void removeAllListeners() {
mListeners.clear();
}
@Override
public void removeListener(AnimatorListener listener) {
mListeners.remove(listener);
}
@Override
public void addPauseListener(AnimatorPauseListener listener) {
mObjectAnimator.addPauseListener(listener);
}
@Override
public void cancel() {
mObjectAnimator.cancel();
mSpring.cancel();
}
@Override
public void end() {
mObjectAnimator.end();
}
@Override
public long getDuration() {
return mObjectAnimator.getDuration();
}
@Override
public TimeInterpolator getInterpolator() {
return mObjectAnimator.getInterpolator();
}
@Override
public long getStartDelay() {
return mObjectAnimator.getStartDelay();
}
@Override
public long getTotalDuration() {
return mObjectAnimator.getTotalDuration();
}
@Override
public boolean isPaused() {
return mObjectAnimator.isPaused();
}
@Override
public boolean isRunning() {
return mObjectAnimator.isRunning();
}
@Override
public boolean isStarted() {
return mObjectAnimator.isStarted();
}
@Override
public void pause() {
mObjectAnimator.pause();
}
@Override
public void removePauseListener(AnimatorPauseListener listener) {
mObjectAnimator.removePauseListener(listener);
}
@Override
public void resume() {
mObjectAnimator.resume();
}
@Override
public ValueAnimator setDuration(long duration) {
return mObjectAnimator.setDuration(duration);
}
@Override
public void setInterpolator(TimeInterpolator value) {
mObjectAnimator.setInterpolator(value);
}
@Override
public void setStartDelay(long startDelay) {
mObjectAnimator.setStartDelay(startDelay);
}
@Override
public void setTarget(Object target) {
mObjectAnimator.setTarget(target);
}
@Override
public void start() {
mObjectAnimator.start();
}
@Override
public void setCurrentFraction(float fraction) {
mObjectAnimator.setCurrentFraction(fraction);
}
@Override
public void setCurrentPlayTime(long playTime) {
mObjectAnimator.setCurrentPlayTime(playTime);
}
public static class SpringProperty<T> extends FloatProperty<T> {
boolean useSpring = false;
final FloatProperty<T> mProperty;
final SpringAnimation mSpring;
public SpringProperty(FloatProperty<T> property, SpringAnimation spring) {
super(property.getName());
mProperty = property;
mSpring = spring;
}
public void switchToSpring() {
useSpring = true;
}
@Override
public Float get(T object) {
return mProperty.get(object);
}
@Override
public void setValue(T object, float progress) {
if (useSpring) {
mSpring.animateToFinalPosition(progress);
} else {
mProperty.setValue(object, progress);
}
}
}
}