1 /* 2 * Copyright (C) 2018 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.server.wm; 18 19 import android.util.ArrayMap; 20 import android.util.ArraySet; 21 22 import java.io.PrintWriter; 23 import java.util.ArrayList; 24 25 /** 26 * Keeps track of all {@link ActivityRecord} that are animating and makes sure all animations are 27 * finished at the same time such that we don't run into issues with z-ordering: An activity A 28 * that has a shorter animation that is above another activity B with a longer animation in the same 29 * task, the animation layer would put the B on top of A, but from the hierarchy, A needs to be on 30 * top of B. Thus, we defer reparenting A to the original hierarchy such that it stays on top of B 31 * until B finishes animating. 32 */ 33 class AnimatingActivityRegistry { 34 35 private ArraySet<ActivityRecord> mAnimatingActivities = new ArraySet<>(); 36 private ArrayMap<ActivityRecord, Runnable> mFinishedTokens = new ArrayMap<>(); 37 38 private ArrayList<Runnable> mTmpRunnableList = new ArrayList<>(); 39 40 private boolean mEndingDeferredFinish; 41 42 /** 43 * Notifies that an {@link ActivityRecord} has started animating. 44 */ notifyStarting(ActivityRecord token)45 void notifyStarting(ActivityRecord token) { 46 mAnimatingActivities.add(token); 47 } 48 49 /** 50 * Notifies that an {@link ActivityRecord} has finished animating. 51 */ notifyFinished(ActivityRecord activity)52 void notifyFinished(ActivityRecord activity) { 53 mAnimatingActivities.remove(activity); 54 mFinishedTokens.remove(activity); 55 56 // If we were the last activity, make sure the end all deferred finishes. 57 if (mAnimatingActivities.isEmpty()) { 58 endDeferringFinished(); 59 } 60 } 61 62 /** 63 * Called when an {@link ActivityRecord} is about to finish animating. 64 * 65 * @param endDeferFinishCallback Callback to run when defer finish should be ended. 66 * @return {@code true} if finishing the animation should be deferred, {@code false} otherwise. 67 */ notifyAboutToFinish(ActivityRecord activity, Runnable endDeferFinishCallback)68 boolean notifyAboutToFinish(ActivityRecord activity, Runnable endDeferFinishCallback) { 69 final boolean removed = mAnimatingActivities.remove(activity); 70 if (!removed) { 71 return false; 72 } 73 74 if (mAnimatingActivities.isEmpty()) { 75 76 // If no animations are animating anymore, finish all others. 77 endDeferringFinished(); 78 return false; 79 } else { 80 81 // Otherwise let's put it into the pending list of to be finished animations. 82 mFinishedTokens.put(activity, endDeferFinishCallback); 83 return true; 84 } 85 } 86 endDeferringFinished()87 private void endDeferringFinished() { 88 89 // Don't start recursing. Running the finished listener invokes notifyFinished, which may 90 // invoked us again. 91 if (mEndingDeferredFinish) { 92 return; 93 } 94 try { 95 mEndingDeferredFinish = true; 96 97 // Copy it into a separate temp list to avoid modifying the collection while iterating 98 // as calling the callback may call back into notifyFinished. 99 for (int i = mFinishedTokens.size() - 1; i >= 0; i--) { 100 mTmpRunnableList.add(mFinishedTokens.valueAt(i)); 101 } 102 mFinishedTokens.clear(); 103 for (int i = mTmpRunnableList.size() - 1; i >= 0; i--) { 104 mTmpRunnableList.get(i).run(); 105 } 106 mTmpRunnableList.clear(); 107 } finally { 108 mEndingDeferredFinish = false; 109 } 110 } 111 dump(PrintWriter pw, String header, String prefix)112 void dump(PrintWriter pw, String header, String prefix) { 113 if (!mAnimatingActivities.isEmpty() || !mFinishedTokens.isEmpty()) { 114 pw.print(prefix); pw.println(header); 115 prefix = prefix + " "; 116 pw.print(prefix); pw.print("mAnimatingActivities="); pw.println(mAnimatingActivities); 117 pw.print(prefix); pw.print("mFinishedTokens="); pw.println(mFinishedTokens); 118 } 119 } 120 } 121