blob: 61aa0de87f05461e3658be26e8c02e444f1ba555 [file] [log] [blame]
Craig Mautner59c00972012-07-30 12:10:24 -07001/*
2 * Copyright (C) 2012 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
17package com.android.server.wm;
18
Craig Mautnerc00204b2013-03-05 15:02:14 -080019import android.graphics.Rect;
Craig Mautnerb47bbc32012-08-22 17:41:48 -070020import android.view.Display;
Craig Mautner59c00972012-07-30 12:10:24 -070021import android.view.DisplayInfo;
22
Craig Mautner9e4f28c2013-04-03 10:53:23 -070023import static com.android.server.am.ActivityStackSupervisor.HOME_STACK_ID;
24
Craig Mautner59c00972012-07-30 12:10:24 -070025import java.io.PrintWriter;
26import java.util.ArrayList;
27
28class DisplayContentList extends ArrayList<DisplayContent> {
29}
30
31/**
32 * Utility class for keeping track of the WindowStates and other pertinent contents of a
33 * particular Display.
34 *
35 * IMPORTANT: No method from this class should ever be used without holding
36 * WindowManagerService.mWindowMap.
37 */
38class DisplayContent {
Craig Mautner496bdbb2013-02-14 09:32:55 -080039// private final static String TAG = "DisplayContent";
Craig Mautner59c00972012-07-30 12:10:24 -070040
41 /** Unique identifier of this stack. */
42 private final int mDisplayId;
43
44 /** Z-ordered (bottom-most first) list of all Window objects. Assigned to an element
45 * from mDisplayWindows; */
46 private WindowList mWindows = new WindowList();
47
Craig Mautner59c00972012-07-30 12:10:24 -070048 // This protects the following display size properties, so that
49 // getDisplaySize() doesn't need to acquire the global lock. This is
50 // needed because the window manager sometimes needs to use ActivityThread
51 // while it has its global state locked (for example to load animation
52 // resources), but the ActivityThread also needs get the current display
53 // size sometimes when it has its package lock held.
54 //
55 // These will only be modified with both mWindowMap and mDisplaySizeLock
56 // held (in that order) so the window manager doesn't need to acquire this
57 // lock when needing these values in its normal operation.
58 final Object mDisplaySizeLock = new Object();
59 int mInitialDisplayWidth = 0;
60 int mInitialDisplayHeight = 0;
Dianne Hackborndde331c2012-08-03 14:01:57 -070061 int mInitialDisplayDensity = 0;
Craig Mautner59c00972012-07-30 12:10:24 -070062 int mBaseDisplayWidth = 0;
63 int mBaseDisplayHeight = 0;
Dianne Hackborndde331c2012-08-03 14:01:57 -070064 int mBaseDisplayDensity = 0;
Craig Mautner2d5618c2012-10-18 13:55:47 -070065 private final DisplayInfo mDisplayInfo = new DisplayInfo();
66 private final Display mDisplay;
Craig Mautner59c00972012-07-30 12:10:24 -070067
Craig Mautner39834192012-09-02 07:47:24 -070068 // Accessed directly by all users.
69 boolean layoutNeeded;
Craig Mautner76a71652012-09-03 23:23:58 -070070 int pendingLayoutChanges;
Craig Mautner69b08182012-09-05 13:07:13 -070071 final boolean isDefaultDisplay;
Craig Mautner39834192012-09-02 07:47:24 -070072
Craig Mautner2d5618c2012-10-18 13:55:47 -070073 /**
Craig Mautnerb1fd65c02013-02-05 13:34:57 -080074 * Window tokens that are in the process of exiting, but still
75 * on screen for animations.
76 */
77 final ArrayList<WindowToken> mExitingTokens = new ArrayList<WindowToken>();
78
79 /**
80 * Application tokens that are in the process of exiting, but still
81 * on screen for animations.
82 */
83 final AppTokenList mExitingAppTokens = new AppTokenList();
84
Craig Mautner00af9fe2013-03-25 09:13:41 -070085 private ArrayList<StackBox> mStackBoxes = new ArrayList<StackBox>();
Craig Mautnerc00204b2013-03-05 15:02:14 -080086
Craig Mautnerb1fd65c02013-02-05 13:34:57 -080087 /**
88 * Sorted most recent at top, oldest at [0].
89 */
Craig Mautnerd9a22882013-03-16 15:00:36 -070090 ArrayList<Task> mTmpTasks = new ArrayList<Task>();
Craig Mautnerb1fd65c02013-02-05 13:34:57 -080091
92 /**
Craig Mautner2d5618c2012-10-18 13:55:47 -070093 * @param display May not be null.
94 */
Craig Mautnerb47bbc32012-08-22 17:41:48 -070095 DisplayContent(Display display) {
96 mDisplay = display;
97 mDisplayId = display.getDisplayId();
98 display.getDisplayInfo(mDisplayInfo);
Craig Mautner69b08182012-09-05 13:07:13 -070099 isDefaultDisplay = mDisplayId == Display.DEFAULT_DISPLAY;
Craig Mautner59c00972012-07-30 12:10:24 -0700100 }
101
102 int getDisplayId() {
103 return mDisplayId;
104 }
105
106 WindowList getWindowList() {
107 return mWindows;
108 }
109
Craig Mautnerb47bbc32012-08-22 17:41:48 -0700110 Display getDisplay() {
111 return mDisplay;
112 }
113
Craig Mautner59c00972012-07-30 12:10:24 -0700114 DisplayInfo getDisplayInfo() {
115 return mDisplayInfo;
116 }
117
Craig Mautner00af9fe2013-03-25 09:13:41 -0700118 /**
119 * Retrieve the tasks on this display in stack order from the topmost TaskStack down.
120 * Note that the order of TaskStacks in the same StackBox is defined within StackBox.
121 * @return All the Tasks, in order, on this display.
122 */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800123 ArrayList<Task> getTasks() {
Craig Mautnerd9a22882013-03-16 15:00:36 -0700124 mTmpTasks.clear();
125 int numBoxes = mStackBoxes.size();
126 for (int boxNdx = 0; boxNdx < numBoxes; ++boxNdx) {
127 mTmpTasks.addAll(mStackBoxes.get(boxNdx).getTasks());
128 }
129 return mTmpTasks;
Craig Mautnerc00204b2013-03-05 15:02:14 -0800130 }
131
Craig Mautner722285e2012-09-07 13:55:58 -0700132 public void updateDisplayInfo() {
133 mDisplay.getDisplayInfo(mDisplayInfo);
134 }
135
Craig Mautner00af9fe2013-03-25 09:13:41 -0700136 /** @return The number of tokens in all of the Tasks on this display. */
Craig Mautnerf81b90872013-02-26 13:02:43 -0800137 int numTokens() {
Craig Mautnerd9a22882013-03-16 15:00:36 -0700138 getTasks();
Craig Mautnerf81b90872013-02-26 13:02:43 -0800139 int count = 0;
Craig Mautnerd9a22882013-03-16 15:00:36 -0700140 for (int taskNdx = mTmpTasks.size() - 1; taskNdx >= 0; --taskNdx) {
141 count += mTmpTasks.get(taskNdx).mAppTokens.size();
Craig Mautner05d6272ba2013-02-11 09:39:27 -0800142 }
Craig Mautnerf81b90872013-02-26 13:02:43 -0800143 return count;
Craig Mautner05d6272ba2013-02-11 09:39:27 -0800144 }
145
Craig Mautner00af9fe2013-03-25 09:13:41 -0700146 /** Refer to {@link WindowManagerService#createStack(int, int, int, float)} */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800147 TaskStack createStack(int stackId, int relativeStackId, int position, float weight) {
148 TaskStack newStack = null;
149 if (mStackBoxes.isEmpty()) {
150 StackBox newBox = new StackBox(this, new Rect(0, 0, mDisplayInfo.logicalWidth,
151 mDisplayInfo.logicalHeight));
152 mStackBoxes.add(newBox);
153 newStack = new TaskStack(stackId, newBox);
154 newBox.mStack = newStack;
155 } else {
156 int stackBoxNdx;
157 for (stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
158 final StackBox box = mStackBoxes.get(stackBoxNdx);
159 if (position == StackBox.TASK_STACK_GOES_OVER
160 || position == StackBox.TASK_STACK_GOES_UNDER) {
161 // Position indicates a new box is added at top level only.
Craig Mautner00af9fe2013-03-25 09:13:41 -0700162 if (box.contains(relativeStackId)) {
Craig Mautnerc00204b2013-03-05 15:02:14 -0800163 final int offset = position == StackBox.TASK_STACK_GOES_OVER ? 1 : 0;
164 StackBox newBox = new StackBox(this, box.mBounds);
165 newStack = new TaskStack(stackId, newBox);
166 newBox.mStack = newStack;
167 mStackBoxes.add(stackBoxNdx + offset, newBox);
168 break;
169 }
170 } else {
171 // Remaining position values indicate a box must be split.
172 newStack = box.split(stackId, relativeStackId, position, weight);
173 if (newStack != null) {
174 break;
175 }
176 }
177 }
Craig Mautner00af9fe2013-03-25 09:13:41 -0700178 if (stackBoxNdx < 0) {
179 throw new IllegalArgumentException("createStack: stackId " + relativeStackId
180 + " not found.");
Craig Mautnerc00204b2013-03-05 15:02:14 -0800181 }
182 }
183 return newStack;
184 }
185
Craig Mautner00af9fe2013-03-25 09:13:41 -0700186 /** Refer to {@link WindowManagerService#resizeStack(int, float)} */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800187 boolean resizeStack(int stackId, float weight) {
188 int stackBoxNdx;
189 for (stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
190 final StackBox box = mStackBoxes.get(stackBoxNdx);
191 if (box.resize(stackId, weight)) {
192 return true;
193 }
194 }
195 return false;
196 }
197
Craig Mautner9e4f28c2013-04-03 10:53:23 -0700198 void removeStackBox(StackBox box) {
199 final TaskStack stack = box.mStack;
200 if (stack != null && stack.mStackId == HOME_STACK_ID) {
201 // Never delete the home stack, even if it is empty.
202 return;
203 }
204 mStackBoxes.remove(box);
205 }
206
Craig Mautner00af9fe2013-03-25 09:13:41 -0700207 /**
208 * Reorder a StackBox within mStackBox. The StackBox to reorder is the one containing the
209 * specified TaskStack.
210 * @param stackId The TaskStack to reorder.
211 * @param toTop Move to the top of all StackBoxes if true, to the bottom if false. Only the
212 * topmost layer of StackBoxes, those in mStackBoxes can be reordered.
213 */
214 void moveStackBox(int stackId, boolean toTop) {
215 for (int stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
216 final StackBox box = mStackBoxes.get(stackBoxNdx);
217 if (box.contains(stackId)) {
218 mStackBoxes.remove(box);
219 mStackBoxes.add(toTop ? mStackBoxes.size() : 0, box);
220 return;
221 }
222 }
223 }
224
Craig Mautnera91f9e22012-09-14 16:22:08 -0700225 public void dump(String prefix, PrintWriter pw) {
226 pw.print(prefix); pw.print("Display: mDisplayId="); pw.println(mDisplayId);
227 final String subPrefix = " " + prefix;
228 pw.print(subPrefix); pw.print("init="); pw.print(mInitialDisplayWidth); pw.print("x");
229 pw.print(mInitialDisplayHeight); pw.print(" "); pw.print(mInitialDisplayDensity);
230 pw.print("dpi");
231 if (mInitialDisplayWidth != mBaseDisplayWidth
232 || mInitialDisplayHeight != mBaseDisplayHeight
233 || mInitialDisplayDensity != mBaseDisplayDensity) {
234 pw.print(" base=");
235 pw.print(mBaseDisplayWidth); pw.print("x"); pw.print(mBaseDisplayHeight);
236 pw.print(" "); pw.print(mBaseDisplayDensity); pw.print("dpi");
237 }
238 pw.print(" cur=");
239 pw.print(mDisplayInfo.logicalWidth);
240 pw.print("x"); pw.print(mDisplayInfo.logicalHeight);
241 pw.print(" app=");
242 pw.print(mDisplayInfo.appWidth);
243 pw.print("x"); pw.print(mDisplayInfo.appHeight);
244 pw.print(" rng="); pw.print(mDisplayInfo.smallestNominalAppWidth);
245 pw.print("x"); pw.print(mDisplayInfo.smallestNominalAppHeight);
246 pw.print("-"); pw.print(mDisplayInfo.largestNominalAppWidth);
247 pw.print("x"); pw.println(mDisplayInfo.largestNominalAppHeight);
Dianne Hackbornc652de82013-02-15 16:32:56 -0800248 pw.print(subPrefix); pw.print("layoutNeeded="); pw.println(layoutNeeded);
Craig Mautner00af9fe2013-03-25 09:13:41 -0700249 for (int boxNdx = 0; boxNdx < mStackBoxes.size(); ++boxNdx) {
250 pw.print(prefix); pw.print("StackBox #"); pw.println(boxNdx);
251 mStackBoxes.get(boxNdx).dump(prefix + " ", pw);
252 }
Craig Mautnerf81b90872013-02-26 13:02:43 -0800253 int ndx = numTokens();
254 if (ndx > 0) {
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800255 pw.println();
256 pw.println(" Application tokens in Z order:");
Craig Mautnerd9a22882013-03-16 15:00:36 -0700257 getTasks();
258 for (int taskNdx = mTmpTasks.size() - 1; taskNdx >= 0; --taskNdx) {
259 AppTokenList tokens = mTmpTasks.get(taskNdx).mAppTokens;
Craig Mautnerf81b90872013-02-26 13:02:43 -0800260 for (int tokenNdx = tokens.size() - 1; tokenNdx >= 0; --tokenNdx) {
261 final AppWindowToken wtoken = tokens.get(tokenNdx);
262 pw.print(" App #"); pw.print(ndx--);
263 pw.print(' '); pw.print(wtoken); pw.println(":");
264 wtoken.dump(pw, " ");
265 }
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800266 }
267 }
268 if (mExitingTokens.size() > 0) {
269 pw.println();
270 pw.println(" Exiting tokens:");
271 for (int i=mExitingTokens.size()-1; i>=0; i--) {
272 WindowToken token = mExitingTokens.get(i);
273 pw.print(" Exiting #"); pw.print(i);
274 pw.print(' '); pw.print(token);
275 pw.println(':');
276 token.dump(pw, " ");
277 }
278 }
279 if (mExitingAppTokens.size() > 0) {
280 pw.println();
281 pw.println(" Exiting application tokens:");
282 for (int i=mExitingAppTokens.size()-1; i>=0; i--) {
283 WindowToken token = mExitingAppTokens.get(i);
284 pw.print(" Exiting App #"); pw.print(i);
285 pw.print(' '); pw.print(token);
286 pw.println(':');
287 token.dump(pw, " ");
288 }
289 }
Craig Mautner59c00972012-07-30 12:10:24 -0700290 pw.println();
291 }
292}