2020-02-25 14:37:01 -08:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2018 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;
|
|
|
|
|
|
2021-04-23 16:28:12 -07:00
|
|
|
import static com.android.launcher3.LauncherAnimUtils.VIEW_BACKGROUND_COLOR;
|
2020-02-25 14:37:01 -08:00
|
|
|
import static com.android.launcher3.anim.AnimatorPlaybackController.addAnimationHoldersRecur;
|
|
|
|
|
|
|
|
|
|
import android.animation.Animator;
|
2020-05-12 15:18:49 -07:00
|
|
|
import android.animation.Animator.AnimatorListener;
|
2020-02-25 14:37:01 -08:00
|
|
|
import android.animation.AnimatorSet;
|
2020-03-13 13:01:33 -07:00
|
|
|
import android.animation.ObjectAnimator;
|
2020-02-25 14:37:01 -08:00
|
|
|
import android.animation.TimeInterpolator;
|
2020-03-13 13:01:33 -07:00
|
|
|
import android.animation.ValueAnimator;
|
2021-04-23 16:28:12 -07:00
|
|
|
import android.graphics.drawable.ColorDrawable;
|
2020-03-13 13:01:33 -07:00
|
|
|
import android.util.FloatProperty;
|
|
|
|
|
import android.util.IntProperty;
|
|
|
|
|
import android.view.View;
|
2020-02-25 14:37:01 -08:00
|
|
|
|
|
|
|
|
import com.android.launcher3.anim.AnimatorPlaybackController.Holder;
|
|
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
|
import java.util.function.Consumer;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Utility class to keep track of a running animation.
|
|
|
|
|
*
|
|
|
|
|
* This class allows attaching end callbacks to an animation is intended to be used with
|
|
|
|
|
* {@link com.android.launcher3.anim.AnimatorPlaybackController}, since in that case
|
|
|
|
|
* AnimationListeners are not properly dispatched.
|
|
|
|
|
*
|
|
|
|
|
* TODO: Find a better name
|
|
|
|
|
*/
|
2020-03-13 13:01:33 -07:00
|
|
|
public class PendingAnimation implements PropertySetter {
|
2020-02-25 14:37:01 -08:00
|
|
|
|
2020-03-13 13:01:33 -07:00
|
|
|
private final ArrayList<Holder> mAnimHolders = new ArrayList<>();
|
|
|
|
|
private final AnimatorSet mAnim;
|
|
|
|
|
private final long mDuration;
|
|
|
|
|
|
|
|
|
|
private ValueAnimator mProgressAnimator;
|
|
|
|
|
|
|
|
|
|
public PendingAnimation(long duration) {
|
|
|
|
|
mDuration = duration;
|
2020-05-12 15:18:49 -07:00
|
|
|
mAnim = new AnimatorSet();
|
2020-03-13 13:01:33 -07:00
|
|
|
}
|
2020-02-25 14:37:01 -08:00
|
|
|
|
2021-06-24 15:23:21 -07:00
|
|
|
public long getDuration() {
|
|
|
|
|
return mDuration;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-25 14:37:01 -08:00
|
|
|
/**
|
|
|
|
|
* Utility method to sent an interpolator on an animation and add it to the list
|
|
|
|
|
*/
|
|
|
|
|
public void add(Animator anim, TimeInterpolator interpolator, SpringProperty springProperty) {
|
|
|
|
|
anim.setInterpolator(interpolator);
|
|
|
|
|
add(anim, springProperty);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void add(Animator anim) {
|
|
|
|
|
add(anim, SpringProperty.DEFAULT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void add(Animator a, SpringProperty springProperty) {
|
2020-06-09 18:35:07 -05:00
|
|
|
mAnim.play(a.setDuration(mDuration));
|
2020-06-01 17:43:54 -07:00
|
|
|
addAnimationHoldersRecur(a, mDuration, springProperty, mAnimHolders);
|
2020-02-25 14:37:01 -08:00
|
|
|
}
|
|
|
|
|
|
2020-03-13 13:01:33 -07:00
|
|
|
@Override
|
|
|
|
|
public void setViewAlpha(View view, float alpha, TimeInterpolator interpolator) {
|
|
|
|
|
if (view == null || view.getAlpha() == alpha) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
ObjectAnimator anim = ObjectAnimator.ofFloat(view, View.ALPHA, alpha);
|
|
|
|
|
anim.addListener(new AlphaUpdateListener(view));
|
2020-06-09 18:35:07 -05:00
|
|
|
anim.setInterpolator(interpolator);
|
2020-03-13 13:01:33 -07:00
|
|
|
add(anim);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-23 16:28:12 -07:00
|
|
|
@Override
|
|
|
|
|
public void setViewBackgroundColor(View view, int color, TimeInterpolator interpolator) {
|
|
|
|
|
if (view == null || (view.getBackground() instanceof ColorDrawable
|
|
|
|
|
&& ((ColorDrawable) view.getBackground()).getColor() == color)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
ObjectAnimator anim = ObjectAnimator.ofArgb(view, VIEW_BACKGROUND_COLOR, color);
|
|
|
|
|
anim.setInterpolator(interpolator);
|
|
|
|
|
add(anim);
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-13 13:01:33 -07:00
|
|
|
@Override
|
|
|
|
|
public <T> void setFloat(T target, FloatProperty<T> property, float value,
|
|
|
|
|
TimeInterpolator interpolator) {
|
|
|
|
|
if (property.get(target) == value) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
Animator anim = ObjectAnimator.ofFloat(target, property, value);
|
|
|
|
|
anim.setDuration(mDuration).setInterpolator(interpolator);
|
|
|
|
|
add(anim);
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-11 13:48:18 -07:00
|
|
|
public <T> void addFloat(T target, FloatProperty<T> property, float from, float to,
|
|
|
|
|
TimeInterpolator interpolator) {
|
|
|
|
|
Animator anim = ObjectAnimator.ofFloat(target, property, from, to);
|
2020-06-09 18:35:07 -05:00
|
|
|
anim.setInterpolator(interpolator);
|
2020-05-11 13:48:18 -07:00
|
|
|
add(anim);
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-13 13:01:33 -07:00
|
|
|
@Override
|
|
|
|
|
public <T> void setInt(T target, IntProperty<T> property, int value,
|
|
|
|
|
TimeInterpolator interpolator) {
|
|
|
|
|
if (property.get(target) == value) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
Animator anim = ObjectAnimator.ofInt(target, property, value);
|
2020-06-09 18:35:07 -05:00
|
|
|
anim.setInterpolator(interpolator);
|
2020-03-13 13:01:33 -07:00
|
|
|
add(anim);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Adds a callback to be run on every frame of the animation
|
|
|
|
|
*/
|
|
|
|
|
public void addOnFrameCallback(Runnable runnable) {
|
2021-03-19 16:04:14 +08:00
|
|
|
addOnFrameListener(anim -> runnable.run());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Adds a listener to be run on every frame of the animation
|
|
|
|
|
*/
|
|
|
|
|
public void addOnFrameListener(ValueAnimator.AnimatorUpdateListener listener) {
|
2020-03-13 13:01:33 -07:00
|
|
|
if (mProgressAnimator == null) {
|
2020-06-09 18:35:07 -05:00
|
|
|
mProgressAnimator = ValueAnimator.ofFloat(0, 1);
|
2020-03-13 13:01:33 -07:00
|
|
|
}
|
|
|
|
|
|
2021-03-19 16:04:14 +08:00
|
|
|
mProgressAnimator.addUpdateListener(listener);
|
2020-03-13 13:01:33 -07:00
|
|
|
}
|
|
|
|
|
|
2020-05-12 15:18:49 -07:00
|
|
|
/**
|
|
|
|
|
* @see AnimatorSet#addListener(AnimatorListener)
|
|
|
|
|
*/
|
|
|
|
|
public void addListener(Animator.AnimatorListener listener) {
|
|
|
|
|
mAnim.addListener(listener);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Creates and returns the underlying AnimatorSet
|
|
|
|
|
*/
|
|
|
|
|
public AnimatorSet buildAnim() {
|
|
|
|
|
// Add progress animation to the end, so that frame callback is called after all the other
|
|
|
|
|
// animation update.
|
|
|
|
|
if (mProgressAnimator != null) {
|
|
|
|
|
add(mProgressAnimator);
|
|
|
|
|
mProgressAnimator = null;
|
|
|
|
|
}
|
|
|
|
|
if (mAnimHolders.isEmpty()) {
|
2020-08-03 12:54:21 -07:00
|
|
|
// Add a placeholder animation to that the duration is respected
|
2020-06-01 17:43:54 -07:00
|
|
|
add(ValueAnimator.ofFloat(0, 1).setDuration(mDuration));
|
2020-05-12 15:18:49 -07:00
|
|
|
}
|
2020-03-13 13:01:33 -07:00
|
|
|
return mAnim;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Creates a controller for this animation
|
|
|
|
|
*/
|
|
|
|
|
public AnimatorPlaybackController createPlaybackController() {
|
2020-05-12 15:18:49 -07:00
|
|
|
return new AnimatorPlaybackController(buildAnim(), mDuration, mAnimHolders);
|
2020-03-13 13:01:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2020-10-21 15:42:05 -07:00
|
|
|
* Add a listener of receiving the success/failure callback in the end.
|
2020-03-13 13:01:33 -07:00
|
|
|
*/
|
2020-10-21 15:42:05 -07:00
|
|
|
public void addEndListener(Consumer<Boolean> listener) {
|
|
|
|
|
if (mProgressAnimator == null) {
|
|
|
|
|
mProgressAnimator = ValueAnimator.ofFloat(0, 1);
|
|
|
|
|
}
|
2021-05-13 17:14:10 +00:00
|
|
|
mProgressAnimator.addListener(AnimatorListeners.forEndCallback(listener));
|
2020-02-25 14:37:01 -08:00
|
|
|
}
|
|
|
|
|
}
|