blob: 8343986dc43c18e32d190cd041856f1e7587ba1a [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 Mautnerd5d5d0f2013-04-03 15:08:21 -070019import static com.android.server.am.ActivityStackSupervisor.HOME_STACK_ID;
Craig Mautnerde4ef022013-04-07 19:01:33 -070020import static com.android.server.wm.WindowManagerService.DEBUG_STACK;
21import static com.android.server.wm.WindowManagerService.TAG;
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -070022
Craig Mautnerc00204b2013-03-05 15:02:14 -080023import android.graphics.Rect;
Craig Mautnerde4ef022013-04-07 19:01:33 -070024import android.util.Slog;
Craig Mautnerb47bbc32012-08-22 17:41:48 -070025import android.view.Display;
Craig Mautner59c00972012-07-30 12:10:24 -070026import android.view.DisplayInfo;
Craig Mautnercf910b02013-04-23 11:23:27 -070027import android.view.InputChannel;
Craig Mautner59c00972012-07-30 12:10:24 -070028
29import java.io.PrintWriter;
30import java.util.ArrayList;
31
32class DisplayContentList extends ArrayList<DisplayContent> {
33}
34
35/**
36 * Utility class for keeping track of the WindowStates and other pertinent contents of a
37 * particular Display.
38 *
39 * IMPORTANT: No method from this class should ever be used without holding
40 * WindowManagerService.mWindowMap.
41 */
42class DisplayContent {
43
44 /** Unique identifier of this stack. */
45 private final int mDisplayId;
46
47 /** Z-ordered (bottom-most first) list of all Window objects. Assigned to an element
48 * from mDisplayWindows; */
49 private WindowList mWindows = new WindowList();
50
Craig Mautner59c00972012-07-30 12:10:24 -070051 // This protects the following display size properties, so that
52 // getDisplaySize() doesn't need to acquire the global lock. This is
53 // needed because the window manager sometimes needs to use ActivityThread
54 // while it has its global state locked (for example to load animation
55 // resources), but the ActivityThread also needs get the current display
56 // size sometimes when it has its package lock held.
57 //
58 // These will only be modified with both mWindowMap and mDisplaySizeLock
59 // held (in that order) so the window manager doesn't need to acquire this
60 // lock when needing these values in its normal operation.
61 final Object mDisplaySizeLock = new Object();
62 int mInitialDisplayWidth = 0;
63 int mInitialDisplayHeight = 0;
Dianne Hackborndde331c2012-08-03 14:01:57 -070064 int mInitialDisplayDensity = 0;
Craig Mautner59c00972012-07-30 12:10:24 -070065 int mBaseDisplayWidth = 0;
66 int mBaseDisplayHeight = 0;
Dianne Hackborndde331c2012-08-03 14:01:57 -070067 int mBaseDisplayDensity = 0;
Craig Mautner2d5618c2012-10-18 13:55:47 -070068 private final DisplayInfo mDisplayInfo = new DisplayInfo();
69 private final Display mDisplay;
Craig Mautner59c00972012-07-30 12:10:24 -070070
Craig Mautner39834192012-09-02 07:47:24 -070071 // Accessed directly by all users.
72 boolean layoutNeeded;
Craig Mautner76a71652012-09-03 23:23:58 -070073 int pendingLayoutChanges;
Craig Mautner69b08182012-09-05 13:07:13 -070074 final boolean isDefaultDisplay;
Craig Mautner39834192012-09-02 07:47:24 -070075
Craig Mautner2d5618c2012-10-18 13:55:47 -070076 /**
Craig Mautnerb1fd65c02013-02-05 13:34:57 -080077 * Window tokens that are in the process of exiting, but still
78 * on screen for animations.
79 */
80 final ArrayList<WindowToken> mExitingTokens = new ArrayList<WindowToken>();
81
82 /**
83 * Application tokens that are in the process of exiting, but still
84 * on screen for animations.
85 */
86 final AppTokenList mExitingAppTokens = new AppTokenList();
87
Craig Mautnercf910b02013-04-23 11:23:27 -070088 /** Array containing the home StackBox and possibly one more which would contain apps. Array
89 * is stored in display order with the current bottom stack at 0. */
Craig Mautner00af9fe2013-03-25 09:13:41 -070090 private ArrayList<StackBox> mStackBoxes = new ArrayList<StackBox>();
Craig Mautnerc00204b2013-03-05 15:02:14 -080091
Craig Mautnercf910b02013-04-23 11:23:27 -070092 /** True when the home StackBox is at the top of mStackBoxes, false otherwise. */
Craig Mautnerde4ef022013-04-07 19:01:33 -070093 private TaskStack mHomeStack = null;
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -070094
Craig Mautner4cd0c13f2013-04-16 15:55:52 -070095 /** Save allocating when retrieving tasks */
Craig Mautnerd9a22882013-03-16 15:00:36 -070096 ArrayList<Task> mTmpTasks = new ArrayList<Task>();
Craig Mautnerb1fd65c02013-02-05 13:34:57 -080097
Craig Mautner4cd0c13f2013-04-16 15:55:52 -070098 /** Sorted most recent at top, oldest at [0]. */
99 ArrayList<TaskStack> mStackHistory = new ArrayList<TaskStack>();
100
Craig Mautnercf910b02013-04-23 11:23:27 -0700101 /** Forward motion events to mTapDetector. */
102 InputChannel mTapInputChannel;
103
104 /** Detect user tapping outside of current focused stack bounds .*/
105 StackTapDetector mTapDetector;
106
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800107 /**
Craig Mautner2d5618c2012-10-18 13:55:47 -0700108 * @param display May not be null.
109 */
Craig Mautnerb47bbc32012-08-22 17:41:48 -0700110 DisplayContent(Display display) {
111 mDisplay = display;
112 mDisplayId = display.getDisplayId();
113 display.getDisplayInfo(mDisplayInfo);
Craig Mautner69b08182012-09-05 13:07:13 -0700114 isDefaultDisplay = mDisplayId == Display.DEFAULT_DISPLAY;
Craig Mautner59c00972012-07-30 12:10:24 -0700115 }
116
117 int getDisplayId() {
118 return mDisplayId;
119 }
120
121 WindowList getWindowList() {
122 return mWindows;
123 }
124
Craig Mautnerb47bbc32012-08-22 17:41:48 -0700125 Display getDisplay() {
126 return mDisplay;
127 }
128
Craig Mautner59c00972012-07-30 12:10:24 -0700129 DisplayInfo getDisplayInfo() {
130 return mDisplayInfo;
131 }
132
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700133 boolean homeOnTop() {
Craig Mautnerde4ef022013-04-07 19:01:33 -0700134 return mStackBoxes.get(0).mStack != mHomeStack;
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700135 }
136
Craig Mautner4cd0c13f2013-04-16 15:55:52 -0700137 void moveStack(TaskStack stack, boolean toTop) {
138 mStackHistory.remove(stack);
139 mStackHistory.add(toTop ? mStackHistory.size() : 0, stack);
140 }
141
Craig Mautner00af9fe2013-03-25 09:13:41 -0700142 /**
Craig Mautner4cd0c13f2013-04-16 15:55:52 -0700143 * Retrieve the tasks on this display in stack order from the bottommost TaskStack up.
Craig Mautner00af9fe2013-03-25 09:13:41 -0700144 * @return All the Tasks, in order, on this display.
145 */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800146 ArrayList<Task> getTasks() {
Craig Mautnerd9a22882013-03-16 15:00:36 -0700147 mTmpTasks.clear();
Craig Mautner4cd0c13f2013-04-16 15:55:52 -0700148 final int numStacks = mStackHistory.size();
149 for (int stackNdx = 0; stackNdx < numStacks; ++stackNdx) {
150 mTmpTasks.addAll(mStackHistory.get(stackNdx).getTasks());
Craig Mautnerd9a22882013-03-16 15:00:36 -0700151 }
Craig Mautner4cd0c13f2013-04-16 15:55:52 -0700152 if (WindowManagerService.DEBUG_LAYERS) Slog.i(TAG, "getTasks: mStackHistory=" +
153 mStackHistory);
Craig Mautnerd9a22882013-03-16 15:00:36 -0700154 return mTmpTasks;
Craig Mautnerc00204b2013-03-05 15:02:14 -0800155 }
156
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700157 TaskStack getHomeStack() {
Craig Mautnerde4ef022013-04-07 19:01:33 -0700158 return mHomeStack;
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700159 }
160
Craig Mautner722285e2012-09-07 13:55:58 -0700161 public void updateDisplayInfo() {
162 mDisplay.getDisplayInfo(mDisplayInfo);
163 }
164
Craig Mautner00af9fe2013-03-25 09:13:41 -0700165 /** @return The number of tokens in all of the Tasks on this display. */
Craig Mautnerf81b90872013-02-26 13:02:43 -0800166 int numTokens() {
Craig Mautnerd9a22882013-03-16 15:00:36 -0700167 getTasks();
Craig Mautnerf81b90872013-02-26 13:02:43 -0800168 int count = 0;
Craig Mautnerd9a22882013-03-16 15:00:36 -0700169 for (int taskNdx = mTmpTasks.size() - 1; taskNdx >= 0; --taskNdx) {
170 count += mTmpTasks.get(taskNdx).mAppTokens.size();
Craig Mautner05d6272ba2013-02-11 09:39:27 -0800171 }
Craig Mautnerf81b90872013-02-26 13:02:43 -0800172 return count;
Craig Mautner05d6272ba2013-02-11 09:39:27 -0800173 }
174
Craig Mautner00af9fe2013-03-25 09:13:41 -0700175 /** Refer to {@link WindowManagerService#createStack(int, int, int, float)} */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800176 TaskStack createStack(int stackId, int relativeStackId, int position, float weight) {
177 TaskStack newStack = null;
Craig Mautnerde4ef022013-04-07 19:01:33 -0700178 if (DEBUG_STACK) Slog.d(TAG, "createStack: stackId=" + stackId + " relativeStackId="
179 + relativeStackId + " position=" + position + " weight=" + weight);
Craig Mautnerc00204b2013-03-05 15:02:14 -0800180 if (mStackBoxes.isEmpty()) {
Craig Mautnerde4ef022013-04-07 19:01:33 -0700181 if (stackId != HOME_STACK_ID) {
182 throw new IllegalArgumentException("createStack: First stackId not "
183 + HOME_STACK_ID);
184 }
Craig Mautner967212c2013-04-13 21:10:58 -0700185 StackBox newBox = new StackBox(this, null);
Craig Mautnerc00204b2013-03-05 15:02:14 -0800186 mStackBoxes.add(newBox);
Craig Mautner967212c2013-04-13 21:10:58 -0700187 newStack = new TaskStack(stackId, this);
188 newStack.mStackBox = newBox;
Craig Mautnerc00204b2013-03-05 15:02:14 -0800189 newBox.mStack = newStack;
Craig Mautnerde4ef022013-04-07 19:01:33 -0700190 mHomeStack = newStack;
Craig Mautnerc00204b2013-03-05 15:02:14 -0800191 } else {
192 int stackBoxNdx;
193 for (stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
194 final StackBox box = mStackBoxes.get(stackBoxNdx);
195 if (position == StackBox.TASK_STACK_GOES_OVER
196 || position == StackBox.TASK_STACK_GOES_UNDER) {
197 // Position indicates a new box is added at top level only.
Craig Mautner00af9fe2013-03-25 09:13:41 -0700198 if (box.contains(relativeStackId)) {
Craig Mautner967212c2013-04-13 21:10:58 -0700199 StackBox newBox = new StackBox(this, null);
200 newStack = new TaskStack(stackId, this);
201 newStack.mStackBox = newBox;
Craig Mautnerc00204b2013-03-05 15:02:14 -0800202 newBox.mStack = newStack;
Craig Mautnerde4ef022013-04-07 19:01:33 -0700203 final int offset = position == StackBox.TASK_STACK_GOES_OVER ? 1 : 0;
204 if (DEBUG_STACK) Slog.d(TAG, "createStack: inserting stack at " +
205 (stackBoxNdx + offset));
Craig Mautnerc00204b2013-03-05 15:02:14 -0800206 mStackBoxes.add(stackBoxNdx + offset, newBox);
207 break;
208 }
209 } else {
210 // Remaining position values indicate a box must be split.
211 newStack = box.split(stackId, relativeStackId, position, weight);
212 if (newStack != null) {
213 break;
214 }
215 }
216 }
Craig Mautner00af9fe2013-03-25 09:13:41 -0700217 if (stackBoxNdx < 0) {
218 throw new IllegalArgumentException("createStack: stackId " + relativeStackId
219 + " not found.");
Craig Mautnerc00204b2013-03-05 15:02:14 -0800220 }
221 }
Craig Mautner967212c2013-04-13 21:10:58 -0700222 if (newStack != null) {
223 layoutNeeded = true;
224 }
Craig Mautnerc00204b2013-03-05 15:02:14 -0800225 return newStack;
226 }
227
Craig Mautner00af9fe2013-03-25 09:13:41 -0700228 /** Refer to {@link WindowManagerService#resizeStack(int, float)} */
Craig Mautnerc00204b2013-03-05 15:02:14 -0800229 boolean resizeStack(int stackId, float weight) {
Craig Mautner967212c2013-04-13 21:10:58 -0700230 for (int stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
Craig Mautnerc00204b2013-03-05 15:02:14 -0800231 final StackBox box = mStackBoxes.get(stackBoxNdx);
232 if (box.resize(stackId, weight)) {
233 return true;
234 }
235 }
236 return false;
237 }
238
Craig Mautner4cd0c13f2013-04-16 15:55:52 -0700239 void addStackBox(StackBox box, boolean toTop) {
240 if (mStackBoxes.size() >= 2) {
241 throw new RuntimeException("addStackBox: Too many toplevel StackBoxes!");
242 }
243 mStackBoxes.add(toTop ? mStackBoxes.size() : 0, box);
244 }
245
Craig Mautner9e4f28c2013-04-03 10:53:23 -0700246 void removeStackBox(StackBox box) {
Craig Mautnerde4ef022013-04-07 19:01:33 -0700247 if (DEBUG_STACK) Slog.d(TAG, "removeStackBox: box=" + box);
Craig Mautner9e4f28c2013-04-03 10:53:23 -0700248 final TaskStack stack = box.mStack;
249 if (stack != null && stack.mStackId == HOME_STACK_ID) {
250 // Never delete the home stack, even if it is empty.
Craig Mautnerde4ef022013-04-07 19:01:33 -0700251 if (DEBUG_STACK) Slog.d(TAG, "removeStackBox: Not deleting home stack.");
Craig Mautner9e4f28c2013-04-03 10:53:23 -0700252 return;
253 }
254 mStackBoxes.remove(box);
255 }
256
Craig Mautner00af9fe2013-03-25 09:13:41 -0700257 /**
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700258 * Move the home StackBox to the top or bottom of mStackBoxes. That is the only place
259 * it is allowed to be. This is a nop if the home StackBox is already in the correct position.
260 * @param toTop Move home to the top of mStackBoxes if true, to the bottom if false.
261 * @return true if a change was made, false otherwise.
Craig Mautner00af9fe2013-03-25 09:13:41 -0700262 */
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700263 boolean moveHomeStackBox(boolean toTop) {
Craig Mautnerde4ef022013-04-07 19:01:33 -0700264 if (DEBUG_STACK) Slog.d(TAG, "moveHomeStackBox: toTop=" + toTop);
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700265 switch (mStackBoxes.size()) {
266 case 0: throw new RuntimeException("moveHomeStackBox: No home StackBox!");
267 case 1: return false; // Only the home StackBox exists.
Craig Mautnerde4ef022013-04-07 19:01:33 -0700268 case 2:
269 if (homeOnTop() ^ toTop) {
270 mStackBoxes.add(mStackBoxes.remove(0));
Craig Mautnerd5d5d0f2013-04-03 15:08:21 -0700271 return true;
272 }
273 return false;
274 default: throw new RuntimeException("moveHomeStackBox: Too many toplevel StackBoxes!");
Craig Mautner00af9fe2013-03-25 09:13:41 -0700275 }
276 }
277
Craig Mautner967212c2013-04-13 21:10:58 -0700278 /**
279 * Propagate the new bounds to all child stack boxes, applying weights as we move down.
280 * @param contentRect The bounds to apply at the top level.
281 */
282 void setStackBoxSize(Rect contentRect) {
283 for (int stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
284 mStackBoxes.get(stackBoxNdx).setStackBoxSizes(contentRect);
285 }
286 }
287
288 Rect getStackBounds(int stackId) {
289 for (int stackBoxNdx = mStackBoxes.size() - 1; stackBoxNdx >= 0; --stackBoxNdx) {
290 Rect bounds = mStackBoxes.get(stackBoxNdx).getStackBounds(stackId);
291 if (bounds != null) {
292 return bounds;
293 }
294 }
295 return null;
296 }
297
Craig Mautnercf910b02013-04-23 11:23:27 -0700298 int stackIdFromPoint(int x, int y) {
299 StackBox topBox = mStackBoxes.get(mStackBoxes.size() - 1);
300 return topBox.stackIdFromPoint(x, y);
301 }
302
Craig Mautnera91f9e22012-09-14 16:22:08 -0700303 public void dump(String prefix, PrintWriter pw) {
304 pw.print(prefix); pw.print("Display: mDisplayId="); pw.println(mDisplayId);
305 final String subPrefix = " " + prefix;
306 pw.print(subPrefix); pw.print("init="); pw.print(mInitialDisplayWidth); pw.print("x");
307 pw.print(mInitialDisplayHeight); pw.print(" "); pw.print(mInitialDisplayDensity);
308 pw.print("dpi");
309 if (mInitialDisplayWidth != mBaseDisplayWidth
310 || mInitialDisplayHeight != mBaseDisplayHeight
311 || mInitialDisplayDensity != mBaseDisplayDensity) {
312 pw.print(" base=");
313 pw.print(mBaseDisplayWidth); pw.print("x"); pw.print(mBaseDisplayHeight);
314 pw.print(" "); pw.print(mBaseDisplayDensity); pw.print("dpi");
315 }
316 pw.print(" cur=");
317 pw.print(mDisplayInfo.logicalWidth);
318 pw.print("x"); pw.print(mDisplayInfo.logicalHeight);
319 pw.print(" app=");
320 pw.print(mDisplayInfo.appWidth);
321 pw.print("x"); pw.print(mDisplayInfo.appHeight);
322 pw.print(" rng="); pw.print(mDisplayInfo.smallestNominalAppWidth);
323 pw.print("x"); pw.print(mDisplayInfo.smallestNominalAppHeight);
324 pw.print("-"); pw.print(mDisplayInfo.largestNominalAppWidth);
325 pw.print("x"); pw.println(mDisplayInfo.largestNominalAppHeight);
Dianne Hackbornc652de82013-02-15 16:32:56 -0800326 pw.print(subPrefix); pw.print("layoutNeeded="); pw.println(layoutNeeded);
Craig Mautnerde4ef022013-04-07 19:01:33 -0700327 for (int boxNdx = 0; boxNdx < mStackBoxes.size(); ++boxNdx) {
328 pw.print(prefix); pw.print("StackBox #"); pw.println(boxNdx);
329 mStackBoxes.get(boxNdx).dump(prefix + " ", pw);
330 }
331 int ndx = numTokens();
332 if (ndx > 0) {
333 pw.println();
334 pw.println(" Application tokens in Z order:");
335 getTasks();
336 for (int taskNdx = mTmpTasks.size() - 1; taskNdx >= 0; --taskNdx) {
337 AppTokenList tokens = mTmpTasks.get(taskNdx).mAppTokens;
338 for (int tokenNdx = tokens.size() - 1; tokenNdx >= 0; --tokenNdx) {
339 final AppWindowToken wtoken = tokens.get(tokenNdx);
340 pw.print(" App #"); pw.print(ndx--);
341 pw.print(' '); pw.print(wtoken); pw.println(":");
342 wtoken.dump(pw, " ");
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800343 }
344 }
Craig Mautnerde4ef022013-04-07 19:01:33 -0700345 }
346 if (mExitingTokens.size() > 0) {
347 pw.println();
348 pw.println(" Exiting tokens:");
349 for (int i=mExitingTokens.size()-1; i>=0; i--) {
350 WindowToken token = mExitingTokens.get(i);
351 pw.print(" Exiting #"); pw.print(i);
352 pw.print(' '); pw.print(token);
353 pw.println(':');
354 token.dump(pw, " ");
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800355 }
Craig Mautnerde4ef022013-04-07 19:01:33 -0700356 }
357 if (mExitingAppTokens.size() > 0) {
358 pw.println();
359 pw.println(" Exiting application tokens:");
360 for (int i=mExitingAppTokens.size()-1; i>=0; i--) {
361 WindowToken token = mExitingAppTokens.get(i);
362 pw.print(" Exiting App #"); pw.print(i);
363 pw.print(' '); pw.print(token);
364 pw.println(':');
365 token.dump(pw, " ");
Craig Mautnerb1fd65c02013-02-05 13:34:57 -0800366 }
Craig Mautnerde4ef022013-04-07 19:01:33 -0700367 }
Craig Mautner59c00972012-07-30 12:10:24 -0700368 pw.println();
369 }
370}