/* * Copyright (C) 2012 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.server.wm; import android.util.SparseArray; import android.view.Display; import android.view.DisplayInfo; import java.io.PrintWriter; import java.util.ArrayList; class DisplayContentList extends ArrayList { } /** * Utility class for keeping track of the WindowStates and other pertinent contents of a * particular Display. * * IMPORTANT: No method from this class should ever be used without holding * WindowManagerService.mWindowMap. */ class DisplayContent { // private final static String TAG = "DisplayContent"; /** Unique identifier of this stack. */ private final int mDisplayId; /** Z-ordered (bottom-most first) list of all Window objects. Assigned to an element * from mDisplayWindows; */ private WindowList mWindows = new WindowList(); // This protects the following display size properties, so that // getDisplaySize() doesn't need to acquire the global lock. This is // needed because the window manager sometimes needs to use ActivityThread // while it has its global state locked (for example to load animation // resources), but the ActivityThread also needs get the current display // size sometimes when it has its package lock held. // // These will only be modified with both mWindowMap and mDisplaySizeLock // held (in that order) so the window manager doesn't need to acquire this // lock when needing these values in its normal operation. final Object mDisplaySizeLock = new Object(); int mInitialDisplayWidth = 0; int mInitialDisplayHeight = 0; int mInitialDisplayDensity = 0; int mBaseDisplayWidth = 0; int mBaseDisplayHeight = 0; int mBaseDisplayDensity = 0; private final DisplayInfo mDisplayInfo = new DisplayInfo(); private final Display mDisplay; // Accessed directly by all users. boolean layoutNeeded; int pendingLayoutChanges; final boolean isDefaultDisplay; /** * List controlling the ordering of windows in different applications which must * be kept in sync with ActivityManager. */ final AppTokenList mAppTokens = new AppTokenList(); /** * AppWindowTokens in the Z order they were in at the start of an animation. Between * animations this list is maintained in the exact order of mAppTokens. If tokens * are added to mAppTokens during an animation an attempt is made to insert them at the same * logical location in this list. Note that this list is always in sync with mWindows. */ AppTokenList mAnimatingAppTokens = new AppTokenList(); /** * Window tokens that are in the process of exiting, but still * on screen for animations. */ final ArrayList mExitingTokens = new ArrayList(); /** * Application tokens that are in the process of exiting, but still * on screen for animations. */ final AppTokenList mExitingAppTokens = new AppTokenList(); /** * Sorted most recent at top, oldest at [0]. */ // ArrayList mTaskLists = new ArrayList(); SparseArray mTaskIdToTaskList = new SparseArray(); /** * @param display May not be null. */ DisplayContent(Display display) { mDisplay = display; mDisplayId = display.getDisplayId(); display.getDisplayInfo(mDisplayInfo); isDefaultDisplay = mDisplayId == Display.DEFAULT_DISPLAY; } int getDisplayId() { return mDisplayId; } WindowList getWindowList() { return mWindows; } Display getDisplay() { return mDisplay; } DisplayInfo getDisplayInfo() { return mDisplayInfo; } public void updateDisplayInfo() { mDisplay.getDisplayInfo(mDisplayInfo); } /** * Find the location to insert a new AppWindowToken into the window-ordered app token list. * @param addPos The location the token was inserted into in mAppTokens. * @param atoken The token to insert. */ void addAppToken(final int addPos, final AppWindowToken atoken) { mAppTokens.add(addPos, atoken); if (addPos == 0 || addPos == mAnimatingAppTokens.size()) { // It was inserted into the beginning or end of mAppTokens. Honor that. mAnimatingAppTokens.add(addPos, atoken); } else { // Find the item immediately above the mAppTokens insertion point and put the token // immediately below that one in mAnimatingAppTokens. final AppWindowToken aboveAnchor = mAppTokens.get(addPos + 1); mAnimatingAppTokens.add(mAnimatingAppTokens.indexOf(aboveAnchor), atoken); } TaskList task = mTaskIdToTaskList.get(atoken.groupId); if (task == null) { mTaskIdToTaskList.put(atoken.groupId, new TaskList(atoken)); } } void removeAppToken(final AppWindowToken wtoken) { mAppTokens.remove(wtoken); mAnimatingAppTokens.remove(wtoken); final int taskId = wtoken.groupId; final TaskList task = mTaskIdToTaskList.get(taskId); if (task != null) { AppTokenList appTokens = task.mAppTokens; appTokens.remove(wtoken); if (appTokens.size() == 0) { mTaskIdToTaskList.delete(taskId); } } } void refillAnimatingAppTokens() { mAnimatingAppTokens.clear(); mAnimatingAppTokens.addAll(mAppTokens); } void setAppTaskId(AppWindowToken wtoken, int newTaskId) { final int taskId = wtoken.groupId; TaskList task = mTaskIdToTaskList.get(taskId); if (task != null) { AppTokenList appTokens = task.mAppTokens; appTokens.remove(wtoken); if (appTokens.size() == 0) { mTaskIdToTaskList.delete(taskId); } } task = mTaskIdToTaskList.get(newTaskId); if (task == null) { task = new TaskList(wtoken); mTaskIdToTaskList.put(newTaskId, task); } else { task.mAppTokens.add(wtoken); } wtoken.groupId = newTaskId; } public void dump(String prefix, PrintWriter pw) { pw.print(prefix); pw.print("Display: mDisplayId="); pw.println(mDisplayId); final String subPrefix = " " + prefix; pw.print(subPrefix); pw.print("init="); pw.print(mInitialDisplayWidth); pw.print("x"); pw.print(mInitialDisplayHeight); pw.print(" "); pw.print(mInitialDisplayDensity); pw.print("dpi"); if (mInitialDisplayWidth != mBaseDisplayWidth || mInitialDisplayHeight != mBaseDisplayHeight || mInitialDisplayDensity != mBaseDisplayDensity) { pw.print(" base="); pw.print(mBaseDisplayWidth); pw.print("x"); pw.print(mBaseDisplayHeight); pw.print(" "); pw.print(mBaseDisplayDensity); pw.print("dpi"); } pw.print(" cur="); pw.print(mDisplayInfo.logicalWidth); pw.print("x"); pw.print(mDisplayInfo.logicalHeight); pw.print(" app="); pw.print(mDisplayInfo.appWidth); pw.print("x"); pw.print(mDisplayInfo.appHeight); pw.print(" rng="); pw.print(mDisplayInfo.smallestNominalAppWidth); pw.print("x"); pw.print(mDisplayInfo.smallestNominalAppHeight); pw.print("-"); pw.print(mDisplayInfo.largestNominalAppWidth); pw.print("x"); pw.println(mDisplayInfo.largestNominalAppHeight); if (mAppTokens.size() > 0) { pw.println(); pw.println(" Application tokens in Z order:"); for (int i=mAppTokens.size()-1; i>=0; i--) { pw.print(" App #"); pw.print(i); pw.print(' '); pw.print(mAppTokens.get(i)); pw.println(":"); mAppTokens.get(i).dump(pw, " "); } } if (mExitingTokens.size() > 0) { pw.println(); pw.println(" Exiting tokens:"); for (int i=mExitingTokens.size()-1; i>=0; i--) { WindowToken token = mExitingTokens.get(i); pw.print(" Exiting #"); pw.print(i); pw.print(' '); pw.print(token); pw.println(':'); token.dump(pw, " "); } } if (mExitingAppTokens.size() > 0) { pw.println(); pw.println(" Exiting application tokens:"); for (int i=mExitingAppTokens.size()-1; i>=0; i--) { WindowToken token = mExitingAppTokens.get(i); pw.print(" Exiting App #"); pw.print(i); pw.print(' '); pw.print(token); pw.println(':'); token.dump(pw, " "); } } if (mTaskIdToTaskList.size() > 0) { pw.println(); for (int i = 0; i < mTaskIdToTaskList.size(); ++i) { pw.print(" TaskList #"); pw.print(i); pw.print(" taskId="); pw.println(mTaskIdToTaskList.keyAt(i)); pw.print(" mAppTokens="); pw.println(mTaskIdToTaskList.valueAt(i).mAppTokens); pw.println(); } } pw.print(subPrefix); pw.print("layoutNeeded="); pw.println(layoutNeeded); pw.println(); } }