1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.tv.settings.util; 18 19 import android.animation.Animator; 20 import android.animation.AnimatorListenerAdapter; 21 import android.animation.ValueAnimator; 22 import android.animation.ValueAnimator.AnimatorUpdateListener; 23 import android.graphics.RectF; 24 import android.view.View; 25 import android.view.ViewGroup; 26 import android.view.animation.DecelerateInterpolator; 27 import android.view.animation.Interpolator; 28 29 import java.util.ArrayList; 30 import java.util.Comparator; 31 import java.util.List; 32 33 /** 34 * Class used by an activity to perform animation of multiple TransitionImageViews 35 * Usage: 36 * - on activity create: 37 * TransitionImageAnimation animation = new TransitionImageAnimation(rootView); 38 * for_each TransitionImage of source 39 * animation.addTransitionSource(source); 40 * animation.startCancelTimer(); 41 * - When the activity loads all target images 42 * for_each TransitionImage of target 43 * animation.addTransitionTarget(target); 44 * animation.startTransition(); 45 */ 46 public class TransitionImageAnimation { 47 48 public static class Listener { 49 onRemovedView(TransitionImage src, TransitionImage dst)50 public void onRemovedView(TransitionImage src, TransitionImage dst) { 51 } 52 onCancelled(TransitionImageAnimation animation)53 public void onCancelled(TransitionImageAnimation animation) { 54 } 55 onFinished(TransitionImageAnimation animation)56 public void onFinished(TransitionImageAnimation animation) { 57 } 58 } 59 60 private static final long DEFAULT_TRANSITION_TIMEOUT_MS = 2000; 61 private static final long DEFAULT_CANCEL_TRANSITION_MS = 250; 62 private static final long DEFAULT_TRANSITION_DURATION_MS = 250; 63 private static final long DEFAULT_TRANSITION_START_DELAY_MS = 160; 64 65 private Interpolator mInterpolator = new DecelerateInterpolator(); 66 private final ViewGroup mRoot; 67 private long mTransitionTimeoutMs = DEFAULT_TRANSITION_TIMEOUT_MS; 68 private long mCancelTransitionMs = DEFAULT_CANCEL_TRANSITION_MS; 69 private long mTransitionDurationMs = DEFAULT_TRANSITION_DURATION_MS; 70 private long mTransitionStartDelayMs = DEFAULT_TRANSITION_START_DELAY_MS; 71 private final List<TransitionImageView> mTransitions = new ArrayList<>(); 72 private Listener mListener; 73 private Comparator<TransitionImage> mComparator = new TransitionImageMatcher(); 74 75 private static final int STATE_INITIAL = 0; 76 private static final int STATE_WAIT_DST = 1; 77 private static final int STATE_TRANSITION = 2; 78 private static final int STATE_CANCELLED = 3; 79 private static final int STATE_FINISHED = 4; 80 private int mState; 81 82 private boolean mListeningLayout; 83 private static final RectF sTmpRect1 = new RectF(); 84 private static final RectF sTmpRect2 = new RectF(); 85 TransitionImageAnimation(ViewGroup root)86 public TransitionImageAnimation(ViewGroup root) { 87 mRoot = root; 88 mState = STATE_INITIAL; 89 } 90 91 /** 92 * Set listener for animation events 93 */ listener(Listener listener)94 public TransitionImageAnimation listener(Listener listener) { 95 mListener = listener; 96 return this; 97 } 98 getListener()99 public Listener getListener() { 100 return mListener; 101 } 102 103 /** 104 * set comparator for matching src and dst ImageTransition 105 */ comparator(Comparator<TransitionImage> comparator)106 public TransitionImageAnimation comparator(Comparator<TransitionImage> comparator) { 107 mComparator = comparator; 108 return this; 109 } 110 getComparator()111 public Comparator<TransitionImage> getComparator() { 112 return mComparator; 113 } 114 115 /** 116 * set interpolator used for animating the Transition 117 */ interpolator(Interpolator interpolator)118 public TransitionImageAnimation interpolator(Interpolator interpolator) { 119 mInterpolator = interpolator; 120 return this; 121 } 122 getInterpolator()123 public Interpolator getInterpolator() { 124 return mInterpolator; 125 } 126 127 /** 128 * set timeout in ms for {@link #startCancelTimer} 129 */ timeoutMs(long timeoutMs)130 public TransitionImageAnimation timeoutMs(long timeoutMs) { 131 mTransitionTimeoutMs = timeoutMs; 132 return this; 133 } 134 getTimeoutMs()135 public long getTimeoutMs() { 136 return mTransitionTimeoutMs; 137 } 138 139 /** 140 * set duration of fade out animation when cancel the transition 141 */ cancelDurationMs(long ms)142 public TransitionImageAnimation cancelDurationMs(long ms) { 143 mCancelTransitionMs = ms; 144 return this; 145 } 146 getCancelDurationMs()147 public long getCancelDurationMs() { 148 return mCancelTransitionMs; 149 } 150 151 /** 152 * set start delay of transition animation 153 */ transitionStartDelayMs(long delay)154 public TransitionImageAnimation transitionStartDelayMs(long delay) { 155 mTransitionStartDelayMs = delay; 156 return this; 157 } 158 getTransitionStartDelayMs()159 public long getTransitionStartDelayMs() { 160 return mTransitionStartDelayMs; 161 } 162 163 /** 164 * set duration of transition animation 165 */ transitionDurationMs(long duration)166 public TransitionImageAnimation transitionDurationMs(long duration) { 167 mTransitionDurationMs = duration; 168 return this; 169 } 170 getTransitionDurationMs()171 public long getTransitionDurationMs() { 172 return mTransitionDurationMs; 173 } 174 175 /** 176 * add source transition and create initial view in root 177 */ addTransitionSource(TransitionImage src)178 public void addTransitionSource(TransitionImage src) { 179 if (mState != STATE_INITIAL) { 180 return; 181 } 182 TransitionImageView view = new TransitionImageView(mRoot.getContext()); 183 mRoot.addView(view); 184 view.setSourceTransition(src); 185 mTransitions.add(view); 186 if (!mListeningLayout) { 187 mListeningLayout = true; 188 mRoot.addOnLayoutChangeListener(mInitializeClip); 189 } 190 } 191 192 /** 193 * kick off the timer for cancel transition 194 */ startCancelTimer()195 public void startCancelTimer() { 196 if (mState != STATE_INITIAL) { 197 return; 198 } 199 mRoot.postDelayed(mCancelTransitionRunnable, mTransitionTimeoutMs); 200 mState = STATE_WAIT_DST; 201 } 202 203 private final Runnable mCancelTransitionRunnable = new Runnable() { 204 205 @Override 206 public void run() { 207 cancelTransition(); 208 } 209 210 }; 211 setProgress(float progress)212 private void setProgress(float progress) { 213 // draw from last child (top most in z-order) 214 int lastIndex = mTransitions.size() - 1; 215 for (int i = lastIndex; i >= 0; i--) { 216 TransitionImageView view = mTransitions.get(i); 217 view.setProgress(progress); 218 sTmpRect2.left = 0; 219 sTmpRect2.top = 0; 220 sTmpRect2.right = view.getWidth(); 221 sTmpRect2.bottom = view.getHeight(); 222 WindowLocationUtil.getLocationsInWindow(view, sTmpRect2); 223 if (i == lastIndex) { 224 view.clearExcludeClipRect(); 225 sTmpRect1.set(sTmpRect2); 226 } else { 227 view.setExcludeClipRect(sTmpRect1); 228 // FIXME: this assumes 3rd image will be clipped by "1st union 2nd", 229 // applies to certain situation such as images are stacked in one row 230 sTmpRect1.union(sTmpRect2); 231 } 232 view.invalidate(); 233 } 234 } 235 236 private final View.OnLayoutChangeListener mInitializeClip = new View.OnLayoutChangeListener() { 237 @Override 238 public void onLayoutChange(View v, int left, int top, int right, int bottom, 239 int oldLeft, int oldTop, int oldRight, int oldBottom) { 240 v.removeOnLayoutChangeListener(this); 241 mListeningLayout = false; 242 // set initial clipping for all views 243 setProgress(0f); 244 } 245 }; 246 247 /** 248 * start transition 249 */ startTransition()250 public void startTransition() { 251 if (mState != STATE_WAIT_DST && mState != STATE_INITIAL) { 252 return; 253 } 254 for (int i = mTransitions.size() - 1; i >= 0; i--) { 255 TransitionImageView view = mTransitions.get(i); 256 if (view.getDestTransition() == null) { 257 cancelTransition(view); 258 mTransitions.remove(i); 259 } 260 } 261 if (mTransitions.size() == 0) { 262 mState = STATE_CANCELLED; 263 if (mListener != null) { 264 mListener.onCancelled(this); 265 } 266 return; 267 } 268 ValueAnimator v = ValueAnimator.ofFloat(0f, 1f); 269 v.setInterpolator(mInterpolator); 270 v.setDuration(mTransitionDurationMs); 271 v.setStartDelay(mTransitionStartDelayMs); 272 v.addUpdateListener(new AnimatorUpdateListener() { 273 @Override 274 public void onAnimationUpdate(ValueAnimator animation) { 275 float progress = animation.getAnimatedFraction(); 276 setProgress(progress); 277 } 278 }); 279 v.addListener(new AnimatorListenerAdapter() { 280 @Override 281 public void onAnimationEnd(Animator animation) { 282 for (int i = 0, count = mTransitions.size(); i < count; i++) { 283 final TransitionImageView view = mTransitions.get(i); 284 if (mListener != null) { 285 mListener.onRemovedView(view.getSourceTransition(), 286 view.getDestTransition()); 287 } 288 mRoot.removeView(view); 289 } 290 mTransitions.clear(); 291 mState = STATE_FINISHED; 292 if (mListener != null) { 293 mListener.onFinished(TransitionImageAnimation.this); 294 } 295 } 296 }); 297 v.start(); 298 mState = STATE_TRANSITION; 299 } 300 cancelTransition(final View iv)301 private void cancelTransition(final View iv) { 302 iv.animate().alpha(0f).setDuration(mCancelTransitionMs). 303 setListener(new AnimatorListenerAdapter() { 304 @Override 305 public void onAnimationEnd(Animator arg0) { 306 mRoot.removeView(iv); 307 } 308 }).start(); 309 } 310 311 /** 312 * Cancel the transition before it starts, no effect if it already starts 313 */ cancelTransition()314 public void cancelTransition() { 315 if (mState != STATE_WAIT_DST && mState != STATE_INITIAL) { 316 return; 317 } 318 int count = mTransitions.size(); 319 if (count > 0) { 320 for (int i = 0; i < count; i++) { 321 cancelTransition(mTransitions.get(i)); 322 } 323 mTransitions.clear(); 324 } 325 mState = STATE_CANCELLED; 326 if (mListener != null) { 327 mListener.onCancelled(this); 328 } 329 } 330 331 /** 332 * find a matching source and relate it with destination 333 */ addTransitionTarget(TransitionImage dst)334 public boolean addTransitionTarget(TransitionImage dst) { 335 if (mState != STATE_WAIT_DST && mState != STATE_INITIAL) { 336 return false; 337 } 338 for (int i = 0, count = mTransitions.size(); i < count; i++) { 339 TransitionImageView view = mTransitions.get(i); 340 if (mComparator.compare(view.getSourceTransition(), dst) == 0) { 341 view.setDestTransition(dst); 342 return true; 343 } 344 } 345 return false; 346 } 347 } 348