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 package android.app; 17 18 import android.animation.Animator; 19 import android.animation.AnimatorListenerAdapter; 20 import android.animation.ObjectAnimator; 21 import android.app.SharedElementCallback.OnSharedElementsReadyListener; 22 import android.graphics.Color; 23 import android.graphics.drawable.ColorDrawable; 24 import android.graphics.drawable.Drawable; 25 import android.os.Bundle; 26 import android.os.ResultReceiver; 27 import android.text.TextUtils; 28 import android.transition.Transition; 29 import android.transition.TransitionListenerAdapter; 30 import android.transition.TransitionManager; 31 import android.util.ArrayMap; 32 import android.view.View; 33 import android.view.ViewGroup; 34 import android.view.ViewGroupOverlay; 35 import android.view.ViewTreeObserver; 36 import android.view.Window; 37 import android.view.accessibility.AccessibilityEvent; 38 39 import com.android.internal.view.OneShotPreDrawListener; 40 41 import java.util.ArrayList; 42 43 /** 44 * This ActivityTransitionCoordinator is created by the Activity to manage 45 * the enter scene and shared element transfer into the Scene, either during 46 * launch of an Activity or returning from a launched Activity. 47 */ 48 class EnterTransitionCoordinator extends ActivityTransitionCoordinator { 49 private static final String TAG = "EnterTransitionCoordinator"; 50 51 private static final int MIN_ANIMATION_FRAMES = 2; 52 53 private boolean mSharedElementTransitionStarted; 54 private Activity mActivity; 55 private boolean mIsTaskRoot; 56 private boolean mHasStopped; 57 private boolean mIsCanceled; 58 private ObjectAnimator mBackgroundAnimator; 59 private boolean mIsExitTransitionComplete; 60 private boolean mIsReadyForTransition; 61 private Bundle mSharedElementsBundle; 62 private boolean mWasOpaque; 63 private boolean mAreViewsReady; 64 private boolean mIsViewsTransitionStarted; 65 private Transition mEnterViewsTransition; 66 private OneShotPreDrawListener mViewsReadyListener; 67 private final boolean mIsCrossTask; 68 private Drawable mReplacedBackground; 69 private ArrayList<String> mPendingExitNames; 70 private Runnable mOnTransitionComplete; 71 EnterTransitionCoordinator(Activity activity, ResultReceiver resultReceiver, ArrayList<String> sharedElementNames, boolean isReturning, boolean isCrossTask)72 EnterTransitionCoordinator(Activity activity, ResultReceiver resultReceiver, 73 ArrayList<String> sharedElementNames, boolean isReturning, boolean isCrossTask) { 74 super(activity.getWindow(), sharedElementNames, 75 getListener(activity, isReturning && !isCrossTask), isReturning); 76 mActivity = activity; 77 mIsCrossTask = isCrossTask; 78 setResultReceiver(resultReceiver); 79 prepareEnter(); 80 Bundle resultReceiverBundle = new Bundle(); 81 resultReceiverBundle.putParcelable(KEY_REMOTE_RECEIVER, this); 82 mResultReceiver.send(MSG_SET_REMOTE_RECEIVER, resultReceiverBundle); 83 final View decorView = getDecor(); 84 if (decorView != null) { 85 final ViewTreeObserver viewTreeObserver = decorView.getViewTreeObserver(); 86 viewTreeObserver.addOnPreDrawListener( 87 new ViewTreeObserver.OnPreDrawListener() { 88 @Override 89 public boolean onPreDraw() { 90 if (mIsReadyForTransition) { 91 if (viewTreeObserver.isAlive()) { 92 viewTreeObserver.removeOnPreDrawListener(this); 93 } else { 94 decorView.getViewTreeObserver().removeOnPreDrawListener(this); 95 } 96 } 97 return false; 98 } 99 }); 100 } 101 } 102 isCrossTask()103 boolean isCrossTask() { 104 return mIsCrossTask; 105 } 106 viewInstancesReady(ArrayList<String> accepted, ArrayList<String> localNames, ArrayList<View> localViews)107 public void viewInstancesReady(ArrayList<String> accepted, ArrayList<String> localNames, 108 ArrayList<View> localViews) { 109 boolean remap = false; 110 for (int i = 0; i < localViews.size(); i++) { 111 View view = localViews.get(i); 112 if (!TextUtils.equals(view.getTransitionName(), localNames.get(i)) 113 || !view.isAttachedToWindow()) { 114 remap = true; 115 break; 116 } 117 } 118 if (remap) { 119 triggerViewsReady(mapNamedElements(accepted, localNames)); 120 } else { 121 triggerViewsReady(mapSharedElements(accepted, localViews)); 122 } 123 } 124 namedViewsReady(ArrayList<String> accepted, ArrayList<String> localNames)125 public void namedViewsReady(ArrayList<String> accepted, ArrayList<String> localNames) { 126 triggerViewsReady(mapNamedElements(accepted, localNames)); 127 } 128 getEnterViewsTransition()129 public Transition getEnterViewsTransition() { 130 return mEnterViewsTransition; 131 } 132 133 @Override viewsReady(ArrayMap<String, View> sharedElements)134 protected void viewsReady(ArrayMap<String, View> sharedElements) { 135 super.viewsReady(sharedElements); 136 mIsReadyForTransition = true; 137 hideViews(mSharedElements); 138 Transition viewsTransition = getViewsTransition(); 139 if (viewsTransition != null && mTransitioningViews != null) { 140 removeExcludedViews(viewsTransition, mTransitioningViews); 141 stripOffscreenViews(); 142 hideViews(mTransitioningViews); 143 } 144 if (mIsReturning) { 145 sendSharedElementDestination(); 146 } else { 147 moveSharedElementsToOverlay(); 148 } 149 if (mSharedElementsBundle != null) { 150 onTakeSharedElements(); 151 } 152 } 153 triggerViewsReady(final ArrayMap<String, View> sharedElements)154 private void triggerViewsReady(final ArrayMap<String, View> sharedElements) { 155 if (mAreViewsReady) { 156 return; 157 } 158 mAreViewsReady = true; 159 final ViewGroup decor = getDecor(); 160 // Ensure the views have been laid out before capturing the views -- we need the epicenter. 161 if (decor == null || (decor.isAttachedToWindow() && 162 (sharedElements.isEmpty() || !sharedElements.valueAt(0).isLayoutRequested()))) { 163 viewsReady(sharedElements); 164 } else { 165 mViewsReadyListener = OneShotPreDrawListener.add(decor, () -> { 166 mViewsReadyListener = null; 167 viewsReady(sharedElements); 168 }); 169 decor.invalidate(); 170 } 171 } 172 mapNamedElements(ArrayList<String> accepted, ArrayList<String> localNames)173 private ArrayMap<String, View> mapNamedElements(ArrayList<String> accepted, 174 ArrayList<String> localNames) { 175 ArrayMap<String, View> sharedElements = new ArrayMap<String, View>(); 176 ViewGroup decorView = getDecor(); 177 if (decorView != null) { 178 decorView.findNamedViews(sharedElements); 179 } 180 if (accepted != null) { 181 for (int i = 0; i < localNames.size(); i++) { 182 String localName = localNames.get(i); 183 String acceptedName = accepted.get(i); 184 if (localName != null && !localName.equals(acceptedName)) { 185 View view = sharedElements.get(localName); 186 if (view != null) { 187 sharedElements.put(acceptedName, view); 188 } 189 } 190 } 191 } 192 return sharedElements; 193 } 194 sendSharedElementDestination()195 private void sendSharedElementDestination() { 196 boolean allReady; 197 final View decorView = getDecor(); 198 if (allowOverlappingTransitions() && getEnterViewsTransition() != null) { 199 allReady = false; 200 } else if (decorView == null) { 201 allReady = true; 202 } else { 203 allReady = !decorView.isLayoutRequested(); 204 if (allReady) { 205 for (int i = 0; i < mSharedElements.size(); i++) { 206 if (mSharedElements.get(i).isLayoutRequested()) { 207 allReady = false; 208 break; 209 } 210 } 211 } 212 } 213 if (allReady) { 214 Bundle state = captureSharedElementState(); 215 moveSharedElementsToOverlay(); 216 mResultReceiver.send(MSG_SHARED_ELEMENT_DESTINATION, state); 217 } else if (decorView != null) { 218 OneShotPreDrawListener.add(decorView, () -> { 219 if (mResultReceiver != null) { 220 Bundle state = captureSharedElementState(); 221 moveSharedElementsToOverlay(); 222 mResultReceiver.send(MSG_SHARED_ELEMENT_DESTINATION, state); 223 } 224 }); 225 } 226 if (allowOverlappingTransitions()) { 227 startEnterTransitionOnly(); 228 } 229 } 230 getListener(Activity activity, boolean isReturning)231 private static SharedElementCallback getListener(Activity activity, boolean isReturning) { 232 return isReturning ? activity.mExitTransitionListener : activity.mEnterTransitionListener; 233 } 234 235 @Override onReceiveResult(int resultCode, Bundle resultData)236 protected void onReceiveResult(int resultCode, Bundle resultData) { 237 switch (resultCode) { 238 case MSG_TAKE_SHARED_ELEMENTS: 239 if (!mIsCanceled) { 240 mSharedElementsBundle = resultData; 241 onTakeSharedElements(); 242 } 243 break; 244 case MSG_EXIT_TRANSITION_COMPLETE: 245 if (!mIsCanceled) { 246 mIsExitTransitionComplete = true; 247 if (mSharedElementTransitionStarted) { 248 onRemoteExitTransitionComplete(); 249 } 250 } 251 break; 252 case MSG_CANCEL: 253 cancel(); 254 break; 255 case MSG_ALLOW_RETURN_TRANSITION: 256 if (!mIsCanceled && !mIsTaskRoot) { 257 mPendingExitNames = mAllSharedElementNames; 258 } 259 break; 260 } 261 } 262 isWaitingForRemoteExit()263 public boolean isWaitingForRemoteExit() { 264 return mIsReturning && mResultReceiver != null; 265 } 266 getPendingExitSharedElementNames()267 public ArrayList<String> getPendingExitSharedElementNames() { 268 return mPendingExitNames; 269 } 270 271 /** 272 * This is called onResume. If an Activity is resuming and the transitions 273 * haven't started yet, force the views to appear. This is likely to be 274 * caused by the top Activity finishing before the transitions started. 275 * In that case, we can finish any transition that was started, but we 276 * should cancel any pending transition and just bring those Views visible. 277 */ forceViewsToAppear()278 public void forceViewsToAppear() { 279 if (!mIsReturning) { 280 return; 281 } 282 if (!mIsReadyForTransition) { 283 mIsReadyForTransition = true; 284 final ViewGroup decor = getDecor(); 285 if (decor != null && mViewsReadyListener != null) { 286 mViewsReadyListener.removeListener(); 287 mViewsReadyListener = null; 288 } 289 showViews(mTransitioningViews, true); 290 setTransitioningViewsVisiblity(View.VISIBLE, true); 291 mSharedElements.clear(); 292 mAllSharedElementNames.clear(); 293 mTransitioningViews.clear(); 294 mIsReadyForTransition = true; 295 viewsTransitionComplete(); 296 sharedElementTransitionComplete(); 297 } else { 298 if (!mSharedElementTransitionStarted) { 299 moveSharedElementsFromOverlay(); 300 mSharedElementTransitionStarted = true; 301 showViews(mSharedElements, true); 302 mSharedElements.clear(); 303 sharedElementTransitionComplete(); 304 } 305 if (!mIsViewsTransitionStarted) { 306 mIsViewsTransitionStarted = true; 307 showViews(mTransitioningViews, true); 308 setTransitioningViewsVisiblity(View.VISIBLE, true); 309 mTransitioningViews.clear(); 310 viewsTransitionComplete(); 311 } 312 cancelPendingTransitions(); 313 } 314 mAreViewsReady = true; 315 if (mResultReceiver != null) { 316 mResultReceiver.send(MSG_CANCEL, null); 317 mResultReceiver = null; 318 } 319 } 320 cancel()321 private void cancel() { 322 if (!mIsCanceled) { 323 mIsCanceled = true; 324 if (getViewsTransition() == null || mIsViewsTransitionStarted) { 325 showViews(mSharedElements, true); 326 } else if (mTransitioningViews != null) { 327 mTransitioningViews.addAll(mSharedElements); 328 } 329 moveSharedElementsFromOverlay(); 330 mSharedElementNames.clear(); 331 mSharedElements.clear(); 332 mAllSharedElementNames.clear(); 333 startSharedElementTransition(null); 334 onRemoteExitTransitionComplete(); 335 } 336 } 337 isReturning()338 public boolean isReturning() { 339 return mIsReturning; 340 } 341 prepareEnter()342 protected void prepareEnter() { 343 ViewGroup decorView = getDecor(); 344 if (mActivity == null || decorView == null) { 345 return; 346 } 347 348 mIsTaskRoot = mActivity.isTaskRoot(); 349 350 if (!isCrossTask()) { 351 mActivity.overridePendingTransition(0, 0); 352 } 353 if (!mIsReturning) { 354 mWasOpaque = mActivity.convertToTranslucent(null, null); 355 Drawable background = decorView.getBackground(); 356 if (background == null) { 357 background = new ColorDrawable(Color.TRANSPARENT); 358 mReplacedBackground = background; 359 } else { 360 getWindow().setBackgroundDrawable(null); 361 background = background.mutate(); 362 background.setAlpha(0); 363 } 364 getWindow().setBackgroundDrawable(background); 365 } else { 366 mActivity = null; // all done with it now. 367 } 368 } 369 370 @Override getViewsTransition()371 protected Transition getViewsTransition() { 372 Window window = getWindow(); 373 if (window == null) { 374 return null; 375 } 376 if (mIsReturning) { 377 return window.getReenterTransition(); 378 } else { 379 return window.getEnterTransition(); 380 } 381 } 382 getSharedElementTransition()383 protected Transition getSharedElementTransition() { 384 Window window = getWindow(); 385 if (window == null) { 386 return null; 387 } 388 if (mIsReturning) { 389 return window.getSharedElementReenterTransition(); 390 } else { 391 return window.getSharedElementEnterTransition(); 392 } 393 } 394 startSharedElementTransition(Bundle sharedElementState)395 private void startSharedElementTransition(Bundle sharedElementState) { 396 ViewGroup decorView = getDecor(); 397 if (decorView == null) { 398 return; 399 } 400 // Remove rejected shared elements 401 ArrayList<String> rejectedNames = new ArrayList<String>(mAllSharedElementNames); 402 rejectedNames.removeAll(mSharedElementNames); 403 ArrayList<View> rejectedSnapshots = createSnapshots(sharedElementState, rejectedNames); 404 if (mListener != null) { 405 mListener.onRejectSharedElements(rejectedSnapshots); 406 } 407 removeNullViews(rejectedSnapshots); 408 startRejectedAnimations(rejectedSnapshots); 409 410 // Now start shared element transition 411 ArrayList<View> sharedElementSnapshots = createSnapshots(sharedElementState, 412 mSharedElementNames); 413 showViews(mSharedElements, true); 414 scheduleSetSharedElementEnd(sharedElementSnapshots); 415 ArrayList<SharedElementOriginalState> originalImageViewState = 416 setSharedElementState(sharedElementState, sharedElementSnapshots); 417 requestLayoutForSharedElements(); 418 419 boolean startEnterTransition = allowOverlappingTransitions() && !mIsReturning; 420 boolean startSharedElementTransition = true; 421 setGhostVisibility(View.INVISIBLE); 422 scheduleGhostVisibilityChange(View.INVISIBLE); 423 pauseInput(); 424 Transition transition = beginTransition(decorView, startEnterTransition, 425 startSharedElementTransition); 426 scheduleGhostVisibilityChange(View.VISIBLE); 427 setGhostVisibility(View.VISIBLE); 428 429 if (startEnterTransition) { 430 startEnterTransition(transition); 431 } 432 433 setOriginalSharedElementState(mSharedElements, originalImageViewState); 434 435 if (mResultReceiver != null) { 436 // We can't trust that the view will disappear on the same frame that the shared 437 // element appears here. Assure that we get at least 2 frames for double-buffering. 438 decorView.postOnAnimation(new Runnable() { 439 int mAnimations; 440 441 @Override 442 public void run() { 443 if (mAnimations++ < MIN_ANIMATION_FRAMES) { 444 View decorView = getDecor(); 445 if (decorView != null) { 446 decorView.postOnAnimation(this); 447 } 448 } else if (mResultReceiver != null) { 449 mResultReceiver.send(MSG_HIDE_SHARED_ELEMENTS, null); 450 mResultReceiver = null; // all done sending messages. 451 } 452 } 453 }); 454 } 455 } 456 removeNullViews(ArrayList<View> views)457 private static void removeNullViews(ArrayList<View> views) { 458 if (views != null) { 459 for (int i = views.size() - 1; i >= 0; i--) { 460 if (views.get(i) == null) { 461 views.remove(i); 462 } 463 } 464 } 465 } 466 onTakeSharedElements()467 private void onTakeSharedElements() { 468 if (!mIsReadyForTransition || mSharedElementsBundle == null) { 469 return; 470 } 471 final Bundle sharedElementState = mSharedElementsBundle; 472 mSharedElementsBundle = null; 473 OnSharedElementsReadyListener listener = new OnSharedElementsReadyListener() { 474 @Override 475 public void onSharedElementsReady() { 476 final View decorView = getDecor(); 477 if (decorView != null) { 478 OneShotPreDrawListener.add(decorView, false, () -> { 479 startTransition(() -> { 480 startSharedElementTransition(sharedElementState); 481 }); 482 }); 483 decorView.invalidate(); 484 } 485 } 486 }; 487 if (mListener == null) { 488 listener.onSharedElementsReady(); 489 } else { 490 mListener.onSharedElementsArrived(mSharedElementNames, mSharedElements, listener); 491 } 492 } 493 requestLayoutForSharedElements()494 private void requestLayoutForSharedElements() { 495 int numSharedElements = mSharedElements.size(); 496 for (int i = 0; i < numSharedElements; i++) { 497 mSharedElements.get(i).requestLayout(); 498 } 499 } 500 beginTransition(ViewGroup decorView, boolean startEnterTransition, boolean startSharedElementTransition)501 private Transition beginTransition(ViewGroup decorView, boolean startEnterTransition, 502 boolean startSharedElementTransition) { 503 Transition sharedElementTransition = null; 504 if (startSharedElementTransition) { 505 if (!mSharedElementNames.isEmpty()) { 506 sharedElementTransition = configureTransition(getSharedElementTransition(), false); 507 } 508 if (sharedElementTransition == null) { 509 sharedElementTransitionStarted(); 510 sharedElementTransitionComplete(); 511 } else { 512 sharedElementTransition.addListener(new TransitionListenerAdapter() { 513 @Override 514 public void onTransitionStart(Transition transition) { 515 sharedElementTransitionStarted(); 516 } 517 518 @Override 519 public void onTransitionEnd(Transition transition) { 520 transition.removeListener(this); 521 sharedElementTransitionComplete(); 522 } 523 }); 524 } 525 } 526 Transition viewsTransition = null; 527 if (startEnterTransition) { 528 mIsViewsTransitionStarted = true; 529 if (mTransitioningViews != null && !mTransitioningViews.isEmpty()) { 530 viewsTransition = configureTransition(getViewsTransition(), true); 531 } 532 if (viewsTransition == null) { 533 viewsTransitionComplete(); 534 } else { 535 final ArrayList<View> transitioningViews = mTransitioningViews; 536 viewsTransition.addListener(new ContinueTransitionListener() { 537 @Override 538 public void onTransitionStart(Transition transition) { 539 mEnterViewsTransition = transition; 540 if (transitioningViews != null) { 541 showViews(transitioningViews, false); 542 } 543 super.onTransitionStart(transition); 544 } 545 546 @Override 547 public void onTransitionEnd(Transition transition) { 548 mEnterViewsTransition = null; 549 transition.removeListener(this); 550 viewsTransitionComplete(); 551 super.onTransitionEnd(transition); 552 } 553 }); 554 } 555 } 556 557 Transition transition = mergeTransitions(sharedElementTransition, viewsTransition); 558 if (transition != null) { 559 transition.addListener(new ContinueTransitionListener()); 560 if (startEnterTransition) { 561 setTransitioningViewsVisiblity(View.INVISIBLE, false); 562 } 563 TransitionManager.beginDelayedTransition(decorView, transition); 564 if (startEnterTransition) { 565 setTransitioningViewsVisiblity(View.VISIBLE, false); 566 } 567 decorView.invalidate(); 568 } else { 569 transitionStarted(); 570 } 571 return transition; 572 } 573 runAfterTransitionsComplete(Runnable onTransitionComplete)574 public void runAfterTransitionsComplete(Runnable onTransitionComplete) { 575 if (!isTransitionRunning()) { 576 onTransitionsComplete(); 577 } else { 578 mOnTransitionComplete = onTransitionComplete; 579 } 580 } 581 582 @Override onTransitionsComplete()583 protected void onTransitionsComplete() { 584 moveSharedElementsFromOverlay(); 585 final ViewGroup decorView = getDecor(); 586 if (decorView != null) { 587 decorView.sendAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED); 588 589 Window window = getWindow(); 590 if (window != null && mReplacedBackground == decorView.getBackground()) { 591 window.setBackgroundDrawable(null); 592 } 593 } 594 if (mOnTransitionComplete != null) { 595 mOnTransitionComplete.run(); 596 mOnTransitionComplete = null; 597 } 598 } 599 sharedElementTransitionStarted()600 private void sharedElementTransitionStarted() { 601 mSharedElementTransitionStarted = true; 602 if (mIsExitTransitionComplete) { 603 send(MSG_EXIT_TRANSITION_COMPLETE, null); 604 } 605 } 606 startEnterTransition(Transition transition)607 private void startEnterTransition(Transition transition) { 608 ViewGroup decorView = getDecor(); 609 if (!mIsReturning && decorView != null) { 610 Drawable background = decorView.getBackground(); 611 if (background != null) { 612 background = background.mutate(); 613 getWindow().setBackgroundDrawable(background); 614 mBackgroundAnimator = ObjectAnimator.ofInt(background, "alpha", 255); 615 mBackgroundAnimator.setDuration(getFadeDuration()); 616 mBackgroundAnimator.addListener(new AnimatorListenerAdapter() { 617 @Override 618 public void onAnimationEnd(Animator animation) { 619 makeOpaque(); 620 backgroundAnimatorComplete(); 621 } 622 }); 623 mBackgroundAnimator.start(); 624 } else if (transition != null) { 625 transition.addListener(new TransitionListenerAdapter() { 626 @Override 627 public void onTransitionEnd(Transition transition) { 628 transition.removeListener(this); 629 makeOpaque(); 630 } 631 }); 632 backgroundAnimatorComplete(); 633 } else { 634 makeOpaque(); 635 backgroundAnimatorComplete(); 636 } 637 } else { 638 backgroundAnimatorComplete(); 639 } 640 } 641 stop()642 public void stop() { 643 // Restore the background to its previous state since the 644 // Activity is stopping. 645 if (mBackgroundAnimator != null) { 646 mBackgroundAnimator.end(); 647 mBackgroundAnimator = null; 648 } else if (mWasOpaque) { 649 ViewGroup decorView = getDecor(); 650 if (decorView != null) { 651 Drawable drawable = decorView.getBackground(); 652 if (drawable != null) { 653 drawable.setAlpha(255); 654 } 655 } 656 } 657 makeOpaque(); 658 mIsCanceled = true; 659 mResultReceiver = null; 660 mActivity = null; 661 moveSharedElementsFromOverlay(); 662 if (mTransitioningViews != null) { 663 showViews(mTransitioningViews, true); 664 setTransitioningViewsVisiblity(View.VISIBLE, true); 665 } 666 showViews(mSharedElements, true); 667 clearState(); 668 } 669 670 /** 671 * Cancels the enter transition. 672 * @return True if the enter transition is still pending capturing the target state. If so, 673 * any transition started on the decor will do nothing. 674 */ cancelEnter()675 public boolean cancelEnter() { 676 setGhostVisibility(View.INVISIBLE); 677 mHasStopped = true; 678 mIsCanceled = true; 679 clearState(); 680 return super.cancelPendingTransitions(); 681 } 682 683 @Override clearState()684 protected void clearState() { 685 mSharedElementsBundle = null; 686 mEnterViewsTransition = null; 687 mResultReceiver = null; 688 if (mBackgroundAnimator != null) { 689 mBackgroundAnimator.cancel(); 690 mBackgroundAnimator = null; 691 } 692 if (mOnTransitionComplete != null) { 693 mOnTransitionComplete.run(); 694 mOnTransitionComplete = null; 695 } 696 super.clearState(); 697 } 698 makeOpaque()699 private void makeOpaque() { 700 if (!mHasStopped && mActivity != null) { 701 if (mWasOpaque) { 702 mActivity.convertFromTranslucent(); 703 } 704 mActivity = null; 705 } 706 } 707 allowOverlappingTransitions()708 private boolean allowOverlappingTransitions() { 709 final Window window = getWindow(); 710 if (window == null) { 711 return false; 712 } 713 return mIsReturning ? window.getAllowReturnTransitionOverlap() 714 : window.getAllowEnterTransitionOverlap(); 715 } 716 startRejectedAnimations(final ArrayList<View> rejectedSnapshots)717 private void startRejectedAnimations(final ArrayList<View> rejectedSnapshots) { 718 if (rejectedSnapshots == null || rejectedSnapshots.isEmpty()) { 719 return; 720 } 721 final ViewGroup decorView = getDecor(); 722 if (decorView != null) { 723 ViewGroupOverlay overlay = decorView.getOverlay(); 724 ObjectAnimator animator = null; 725 int numRejected = rejectedSnapshots.size(); 726 for (int i = 0; i < numRejected; i++) { 727 View snapshot = rejectedSnapshots.get(i); 728 overlay.add(snapshot); 729 animator = ObjectAnimator.ofFloat(snapshot, View.ALPHA, 1, 0); 730 animator.start(); 731 } 732 animator.addListener(new AnimatorListenerAdapter() { 733 @Override 734 public void onAnimationEnd(Animator animation) { 735 ViewGroupOverlay overlay = decorView.getOverlay(); 736 int numRejected = rejectedSnapshots.size(); 737 for (int i = 0; i < numRejected; i++) { 738 overlay.remove(rejectedSnapshots.get(i)); 739 } 740 } 741 }); 742 } 743 } 744 onRemoteExitTransitionComplete()745 protected void onRemoteExitTransitionComplete() { 746 if (!allowOverlappingTransitions()) { 747 startEnterTransitionOnly(); 748 } 749 } 750 startEnterTransitionOnly()751 private void startEnterTransitionOnly() { 752 startTransition(new Runnable() { 753 @Override 754 public void run() { 755 boolean startEnterTransition = true; 756 boolean startSharedElementTransition = false; 757 ViewGroup decorView = getDecor(); 758 if (decorView != null) { 759 Transition transition = beginTransition(decorView, startEnterTransition, 760 startSharedElementTransition); 761 startEnterTransition(transition); 762 } 763 } 764 }); 765 } 766 } 767