blob: 48737a54cadc7f9514abde647aba8f507d42a251 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2006 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.am;
18
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -080019import static android.app.ActivityManager.RESIZE_MODE_FORCED;
Winson Chung74666102017-02-22 17:49:24 -080020import static android.app.ActivityManager.RESIZE_MODE_SYSTEM;
Jorim Jaggi0a932142016-02-01 17:42:25 -080021import static android.app.ActivityManager.StackId.INVALID_STACK_ID;
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -070022import static android.app.WindowConfiguration.ACTIVITY_TYPE_STANDARD;
23import static android.app.WindowConfiguration.ACTIVITY_TYPE_UNDEFINED;
Wale Ogunwale04a05ac2017-09-17 21:35:02 -070024import static android.app.WindowConfiguration.WINDOWING_MODE_FREEFORM;
25import static android.app.WindowConfiguration.WINDOWING_MODE_FULLSCREEN;
26import static android.app.WindowConfiguration.WINDOWING_MODE_PINNED;
27import static android.app.WindowConfiguration.WINDOWING_MODE_SPLIT_SCREEN_PRIMARY;
Wale Ogunwale44f036f2017-09-29 05:09:09 -070028import static android.app.WindowConfiguration.WINDOWING_MODE_SPLIT_SCREEN_SECONDARY;
Jorim Jaggi0a932142016-02-01 17:42:25 -080029import static android.content.Intent.FLAG_ACTIVITY_NEW_DOCUMENT;
Wale Ogunwale66e16852017-10-19 13:35:52 -070030import static android.content.Intent.FLAG_ACTIVITY_NEW_TASK;
Jorim Jaggi0a932142016-02-01 17:42:25 -080031import static android.content.Intent.FLAG_ACTIVITY_RETAIN_IN_RECENTS;
Wale Ogunwale66e16852017-10-19 13:35:52 -070032import static android.content.Intent.FLAG_ACTIVITY_TASK_ON_HOME;
Wale Ogunwale3eadad72016-10-13 09:16:59 -070033import static android.content.pm.ActivityInfo.FLAG_RELINQUISH_TASK_IDENTITY;
Jorim Jaggi0a932142016-02-01 17:42:25 -080034import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_ALWAYS;
35import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_DEFAULT;
36import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_IF_WHITELISTED;
37import static android.content.pm.ActivityInfo.LOCK_TASK_LAUNCH_MODE_NEVER;
skuhne@google.com322347b2016-12-02 12:54:03 -080038import static android.content.pm.ActivityInfo.RESIZE_MODE_FORCE_RESIZABLE_LANDSCAPE_ONLY;
39import static android.content.pm.ActivityInfo.RESIZE_MODE_FORCE_RESIZABLE_PORTRAIT_ONLY;
40import static android.content.pm.ActivityInfo.RESIZE_MODE_FORCE_RESIZABLE_PRESERVE_ORIENTATION;
Wale Ogunwaled829d362016-02-10 19:24:49 -080041import static android.content.pm.ActivityInfo.RESIZE_MODE_FORCE_RESIZEABLE;
Wale Ogunwale625ed0c2016-10-18 08:50:31 -070042import static android.content.pm.ActivityInfo.RESIZE_MODE_RESIZEABLE;
Winson Chungd3395382016-12-13 11:49:09 -080043import static android.content.pm.ActivityInfo.RESIZE_MODE_RESIZEABLE_AND_PIPABLE_DEPRECATED;
Wale Ogunwale625ed0c2016-10-18 08:50:31 -070044import static android.content.pm.ActivityInfo.RESIZE_MODE_RESIZEABLE_VIA_SDK_VERSION;
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -080045import static android.os.Trace.TRACE_TAG_ACTIVITY_MANAGER;
Suprabh Shukla7745c142016-03-07 18:21:10 -080046import static android.provider.Settings.Secure.USER_SETUP_COMPLETE;
Andrii Kulian036e3ad2017-04-19 10:55:10 -070047import static android.view.Display.DEFAULT_DISPLAY;
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -070048
Jorim Jaggi0a932142016-02-01 17:42:25 -080049import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_ADD_REMOVE;
50import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_LOCKTASK;
51import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_RECENTS;
52import static com.android.server.am.ActivityManagerDebugConfig.DEBUG_TASKS;
53import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_ADD_REMOVE;
54import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_LOCKTASK;
55import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_RECENTS;
56import static com.android.server.am.ActivityManagerDebugConfig.POSTFIX_TASKS;
57import static com.android.server.am.ActivityManagerDebugConfig.TAG_AM;
58import static com.android.server.am.ActivityManagerDebugConfig.TAG_WITH_CLASS_NAME;
Wale Ogunwale3b232392016-05-13 15:37:13 -070059import static com.android.server.am.ActivityRecord.STARTING_WINDOW_SHOWN;
Wale Ogunwalec5cc3012017-01-13 13:26:16 -080060import static com.android.server.am.ActivityStack.REMOVE_TASK_MODE_MOVING;
Wale Ogunwale56d8d162017-05-30 11:12:20 -070061import static com.android.server.am.ActivityStack.REMOVE_TASK_MODE_MOVING_TO_TOP;
Wale Ogunwaleab5de372017-10-18 06:46:31 -070062import static com.android.server.am.ActivityStackSupervisor.ON_TOP;
Winson Chung6954fc92017-03-24 16:22:12 -070063import static com.android.server.am.ActivityStackSupervisor.PAUSE_IMMEDIATELY;
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -080064import static com.android.server.am.ActivityStackSupervisor.PRESERVE_WINDOWS;
Steven Timotius4346f0a2017-09-12 11:07:21 -070065import static com.android.server.am.proto.TaskRecordProto.ACTIVITIES;
66import static com.android.server.am.proto.TaskRecordProto.BOUNDS;
67import static com.android.server.am.proto.TaskRecordProto.CONFIGURATION_CONTAINER;
68import static com.android.server.am.proto.TaskRecordProto.FULLSCREEN;
69import static com.android.server.am.proto.TaskRecordProto.ID;
70import static com.android.server.am.proto.TaskRecordProto.LAST_NON_FULLSCREEN_BOUNDS;
71import static com.android.server.am.proto.TaskRecordProto.MIN_HEIGHT;
72import static com.android.server.am.proto.TaskRecordProto.MIN_WIDTH;
73import static com.android.server.am.proto.TaskRecordProto.ORIG_ACTIVITY;
74import static com.android.server.am.proto.TaskRecordProto.REAL_ACTIVITY;
75import static com.android.server.am.proto.TaskRecordProto.RESIZE_MODE;
Steven Timotius4346f0a2017-09-12 11:07:21 -070076import static com.android.server.am.proto.TaskRecordProto.STACK_ID;
77import static com.android.server.am.proto.TaskRecordProto.ACTIVITY_TYPE;
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -070078
Winson Chung74666102017-02-22 17:49:24 -080079import static java.lang.Integer.MAX_VALUE;
80
Jorim Jaggie7d2b852017-08-28 17:55:15 +020081import android.annotation.IntDef;
82import android.annotation.Nullable;
83import android.app.Activity;
84import android.app.ActivityManager;
Jorim Jaggie7d2b852017-08-28 17:55:15 +020085import android.app.ActivityManager.TaskDescription;
86import android.app.ActivityManager.TaskSnapshot;
87import android.app.ActivityOptions;
88import android.app.AppGlobals;
89import android.app.IActivityManager;
90import android.content.ComponentName;
91import android.content.Intent;
92import android.content.pm.ActivityInfo;
93import android.content.pm.ApplicationInfo;
94import android.content.pm.IPackageManager;
95import android.content.pm.PackageManager;
96import android.content.res.Configuration;
97import android.graphics.Rect;
98import android.os.Debug;
99import android.os.RemoteException;
Winson Chungfb44d212017-10-04 11:39:10 -0700100import android.os.SystemClock;
Jorim Jaggie7d2b852017-08-28 17:55:15 +0200101import android.os.Trace;
102import android.os.UserHandle;
103import android.provider.Settings;
104import android.service.voice.IVoiceInteractionSession;
105import android.util.DisplayMetrics;
106import android.util.Slog;
Steven Timotius4346f0a2017-09-12 11:07:21 -0700107import android.util.proto.ProtoOutputStream;
Jorim Jaggie7d2b852017-08-28 17:55:15 +0200108
109import com.android.internal.annotations.VisibleForTesting;
110import com.android.internal.app.IVoiceInteractor;
111import com.android.internal.util.XmlUtils;
Winson Chung61c9e5a2017-10-11 10:39:32 -0700112import com.android.server.am.ActivityStack.ActivityState;
Jorim Jaggie7d2b852017-08-28 17:55:15 +0200113import com.android.server.wm.AppWindowContainerController;
114import com.android.server.wm.ConfigurationContainer;
115import com.android.server.wm.StackWindowController;
116import com.android.server.wm.TaskWindowContainerController;
117import com.android.server.wm.TaskWindowContainerListener;
118import com.android.server.wm.WindowManagerService;
119
120import org.xmlpull.v1.XmlPullParser;
121import org.xmlpull.v1.XmlPullParserException;
122import org.xmlpull.v1.XmlSerializer;
123
124import java.io.IOException;
125import java.io.PrintWriter;
126import java.lang.annotation.Retention;
127import java.lang.annotation.RetentionPolicy;
128import java.util.ArrayList;
129import java.util.Objects;
130
Benjamin Franza83859f2017-07-03 16:34:14 +0100131class TaskRecord extends ConfigurationContainer implements TaskWindowContainerListener {
Wale Ogunwalee23149f2015-03-06 15:39:44 -0800132 private static final String TAG = TAG_WITH_CLASS_NAME ? "TaskRecord" : TAG_AM;
Wale Ogunwale0fc365c2015-05-25 19:35:42 -0700133 private static final String TAG_ADD_REMOVE = TAG + POSTFIX_ADD_REMOVE;
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700134 private static final String TAG_RECENTS = TAG + POSTFIX_RECENTS;
Craig Mautnere0570202015-05-13 13:06:11 -0700135 private static final String TAG_LOCKTASK = TAG + POSTFIX_LOCKTASK;
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700136 private static final String TAG_TASKS = TAG + POSTFIX_TASKS;
Wale Ogunwalee23149f2015-03-06 15:39:44 -0800137
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700138 private static final String ATTR_TASKID = "task_id";
Craig Mautner21d24a22014-04-23 11:45:37 -0700139 private static final String TAG_INTENT = "intent";
140 private static final String TAG_AFFINITYINTENT = "affinity_intent";
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700141 private static final String ATTR_REALACTIVITY = "real_activity";
142 private static final String ATTR_REALACTIVITY_SUSPENDED = "real_activity_suspended";
Craig Mautner21d24a22014-04-23 11:45:37 -0700143 private static final String ATTR_ORIGACTIVITY = "orig_activity";
Stefan Kuhnee88d1e52015-05-18 10:33:45 -0700144 private static final String TAG_ACTIVITY = "activity";
Craig Mautner21d24a22014-04-23 11:45:37 -0700145 private static final String ATTR_AFFINITY = "affinity";
Dianne Hackborn79228822014-09-16 11:11:23 -0700146 private static final String ATTR_ROOT_AFFINITY = "root_affinity";
Craig Mautner21d24a22014-04-23 11:45:37 -0700147 private static final String ATTR_ROOTHASRESET = "root_has_reset";
Dianne Hackborn13420f22014-07-18 15:43:56 -0700148 private static final String ATTR_AUTOREMOVERECENTS = "auto_remove_recents";
Craig Mautner21d24a22014-04-23 11:45:37 -0700149 private static final String ATTR_ASKEDCOMPATMODE = "asked_compat_mode";
150 private static final String ATTR_USERID = "user_id";
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800151 private static final String ATTR_USER_SETUP_COMPLETE = "user_setup_complete";
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700152 private static final String ATTR_EFFECTIVE_UID = "effective_uid";
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -0700153 @Deprecated
Craig Mautner21d24a22014-04-23 11:45:37 -0700154 private static final String ATTR_TASKTYPE = "task_type";
Craig Mautner21d24a22014-04-23 11:45:37 -0700155 private static final String ATTR_LASTDESCRIPTION = "last_description";
156 private static final String ATTR_LASTTIMEMOVED = "last_time_moved";
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700157 private static final String ATTR_NEVERRELINQUISH = "never_relinquish_identity";
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700158 private static final String ATTR_TASK_AFFILIATION = "task_affiliation";
Craig Mautnera228ae92014-07-09 05:44:55 -0700159 private static final String ATTR_PREV_AFFILIATION = "prev_affiliation";
160 private static final String ATTR_NEXT_AFFILIATION = "next_affiliation";
Winson Chungec396d62014-08-06 17:08:00 -0700161 private static final String ATTR_TASK_AFFILIATION_COLOR = "task_affiliation_color";
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700162 private static final String ATTR_CALLING_UID = "calling_uid";
163 private static final String ATTR_CALLING_PACKAGE = "calling_package";
Winson Chungd3395382016-12-13 11:49:09 -0800164 private static final String ATTR_SUPPORTS_PICTURE_IN_PICTURE = "supports_picture_in_picture";
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800165 private static final String ATTR_RESIZE_MODE = "resize_mode";
Wale Ogunwale706ed792015-08-02 10:29:44 -0700166 private static final String ATTR_NON_FULLSCREEN_BOUNDS = "non_fullscreen_bounds";
Andrii Kulianf66a83d2016-05-17 12:17:44 -0700167 private static final String ATTR_MIN_WIDTH = "min_width";
168 private static final String ATTR_MIN_HEIGHT = "min_height";
Wale Ogunwale625ed0c2016-10-18 08:50:31 -0700169 private static final String ATTR_PERSIST_TASK_VERSION = "persist_task_version";
Andrii Kulian18d75122016-03-27 20:20:28 -0700170
Wale Ogunwale625ed0c2016-10-18 08:50:31 -0700171 // Current version of the task record we persist. Used to check if we need to run any upgrade
172 // code.
173 private static final int PERSIST_TASK_VERSION = 1;
Craig Mautner21d24a22014-04-23 11:45:37 -0700174
Wale Ogunwale18795a22014-12-03 11:38:33 -0800175 static final int INVALID_TASK_ID = -1;
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700176 private static final int INVALID_MIN_SIZE = -1;
Wale Ogunwale18795a22014-12-03 11:38:33 -0800177
Winson Chung74666102017-02-22 17:49:24 -0800178 /**
179 * The modes to control how the stack is moved to the front when calling
180 * {@link TaskRecord#reparent}.
181 */
182 @Retention(RetentionPolicy.SOURCE)
183 @IntDef({
184 REPARENT_MOVE_STACK_TO_FRONT,
185 REPARENT_KEEP_STACK_AT_FRONT,
186 REPARENT_LEAVE_STACK_IN_PLACE
187 })
Wale Ogunwale66e16852017-10-19 13:35:52 -0700188 @interface ReparentMoveStackMode {}
Winson Chung74666102017-02-22 17:49:24 -0800189 // Moves the stack to the front if it was not at the front
Wale Ogunwale66e16852017-10-19 13:35:52 -0700190 static final int REPARENT_MOVE_STACK_TO_FRONT = 0;
Winson Chung74666102017-02-22 17:49:24 -0800191 // Only moves the stack to the front if it was focused or front most already
Wale Ogunwale66e16852017-10-19 13:35:52 -0700192 static final int REPARENT_KEEP_STACK_AT_FRONT = 1;
Winson Chung74666102017-02-22 17:49:24 -0800193 // Do not move the stack as a part of reparenting
Wale Ogunwale66e16852017-10-19 13:35:52 -0700194 static final int REPARENT_LEAVE_STACK_IN_PLACE = 2;
Winson Chung74666102017-02-22 17:49:24 -0800195
Garfield Tan9b1efea2017-12-05 16:43:46 -0800196 /**
197 * The factory used to create {@link TaskRecord}. This allows OEM subclass {@link TaskRecord}.
198 */
199 private static TaskRecordFactory sTaskRecordFactory;
200
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800201 final int taskId; // Unique identifier for this task.
Dianne Hackborn79228822014-09-16 11:11:23 -0700202 String affinity; // The affinity name for this task, or null; may change identity.
203 String rootAffinity; // Initial base affinity, or null; does not change from initial root.
Dianne Hackborn91097de2014-04-04 18:02:06 -0700204 final IVoiceInteractionSession voiceSession; // Voice interaction session driving task
205 final IVoiceInteractor voiceInteractor; // Associated interactor to provide to app
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800206 Intent intent; // The original intent that started the task.
207 Intent affinityIntent; // Intent of affinity-moved activity that started this task.
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700208 int effectiveUid; // The current effective uid of the identity of this task.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800209 ComponentName origActivity; // The non-alias activity component of the intent.
210 ComponentName realActivity; // The actual activity component that started the task.
Andrei Stingaceanu4ccec532016-01-13 12:10:21 +0000211 boolean realActivitySuspended; // True if the actual activity component that started the
212 // task is suspended.
Dianne Hackborn852975d2014-08-22 17:42:43 -0700213 boolean inRecents; // Actually in the recents list?
Winson Chungfb44d212017-10-04 11:39:10 -0700214 long lastActiveTime; // Last time this task was active in the current device session,
215 // including sleep. This time is initialized to the elapsed time when
216 // restored from disk.
Dianne Hackborn852975d2014-08-22 17:42:43 -0700217 boolean isAvailable; // Is the activity available to be launched?
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800218 boolean rootWasReset; // True if the intent at the root of the task had
219 // the FLAG_ACTIVITY_RESET_TASK_IF_NEEDED flag.
Dianne Hackborn13420f22014-07-18 15:43:56 -0700220 boolean autoRemoveRecents; // If true, we should automatically remove the task from
221 // recents when activity finishes
Dianne Hackborn36cd41f2011-05-25 21:00:46 -0700222 boolean askedCompatMode;// Have asked the user about compat mode for this task.
Dianne Hackbornd38aed82014-06-10 21:36:35 -0700223 boolean hasBeenVisible; // Set if any activities in the task have been visible to the user.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800224
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700225 String stringName; // caching of toString() result.
Dianne Hackborn9da2d402012-03-15 13:43:08 -0700226 int userId; // user for which this task was created
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800227 boolean mUserSetupComplete; // The user set-up is complete as of the last time the task activity
228 // was changed.
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800229
230 int numFullscreen; // Number of fullscreen activities.
231
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800232 int mResizeMode; // The resize mode of this task and its activities.
233 // Based on the {@link ActivityInfo#resizeMode} of the root activity.
Wale Ogunwale069bbd32017-02-03 07:58:14 -0800234 private boolean mSupportsPictureInPicture; // Whether or not this task and its activities
235 // support PiP. Based on the {@link ActivityInfo#FLAG_SUPPORTS_PICTURE_IN_PICTURE} flag
236 // of the root activity.
Craig Mautner15df08a2015-04-01 12:17:18 -0700237 /** Can't be put in lockTask mode. */
238 final static int LOCK_TASK_AUTH_DONT_LOCK = 0;
Benjamin Franz469dd582015-06-09 14:24:36 +0100239 /** Can enter app pinning with user approval. Can never start over existing lockTask task. */
Craig Mautner15df08a2015-04-01 12:17:18 -0700240 final static int LOCK_TASK_AUTH_PINNABLE = 1;
241 /** Starts in LOCK_TASK_MODE_LOCKED automatically. Can start over existing lockTask task. */
242 final static int LOCK_TASK_AUTH_LAUNCHABLE = 2;
Benjamin Franz469dd582015-06-09 14:24:36 +0100243 /** Can enter lockTask without user approval. Can start over existing lockTask task. */
Craig Mautner15df08a2015-04-01 12:17:18 -0700244 final static int LOCK_TASK_AUTH_WHITELISTED = 3;
Benjamin Franz469dd582015-06-09 14:24:36 +0100245 /** Priv-app that starts in LOCK_TASK_MODE_LOCKED automatically. Can start over existing
246 * lockTask task. */
247 final static int LOCK_TASK_AUTH_LAUNCHABLE_PRIV = 4;
Craig Mautner15df08a2015-04-01 12:17:18 -0700248 int mLockTaskAuth = LOCK_TASK_AUTH_PINNABLE;
249
250 int mLockTaskUid = -1; // The uid of the application that called startLockTask().
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800251
Winson Chung03a9bae2014-05-02 09:56:12 -0700252 // This represents the last resolved activity values for this task
253 // NOTE: This value needs to be persisted with each task
Craig Mautner648f69b2014-09-18 14:16:26 -0700254 TaskDescription lastTaskDescription = new TaskDescription();
Winson Chung03a9bae2014-05-02 09:56:12 -0700255
Craig Mautnerd2328952013-03-05 12:46:26 -0800256 /** List of all activities in the task arranged in history order */
Craig Mautner21d24a22014-04-23 11:45:37 -0700257 final ArrayList<ActivityRecord> mActivities;
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800258
Andrii Kulian02b7a832016-10-06 23:11:56 -0700259 /** Current stack. Setter must always be used to update the value. */
260 private ActivityStack mStack;
Craig Mautnerd2328952013-03-05 12:46:26 -0800261
Dianne Hackborn68a06332017-11-15 17:54:18 -0800262 /** The process that had previously hosted the root activity of this task.
263 * Used to know that we should try harder to keep this process around, in case the
264 * user wants to return to it. */
265 private ProcessRecord mRootProcess;
266
Craig Mautner21d24a22014-04-23 11:45:37 -0700267 /** Takes on same value as first root activity */
268 boolean isPersistable = false;
Craig Mautnerffcfcaa2014-06-05 09:54:38 -0700269 int maxRecents;
Craig Mautner21d24a22014-04-23 11:45:37 -0700270
271 /** Only used for persistable tasks, otherwise 0. The last time this task was moved. Used for
272 * determining the order when restoring. Sign indicates whether last task movement was to front
273 * (positive) or back (negative). Absolute value indicates time. */
274 long mLastTimeMoved = System.currentTimeMillis();
275
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700276 /** If original intent did not allow relinquishing task identity, save that information */
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700277 private boolean mNeverRelinquishIdentity = true;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700278
Craig Mautner362449a2014-06-20 14:04:39 -0700279 // Used in the unique case where we are clearing the task in order to reuse it. In that case we
280 // do not want to delete the stack when the task goes empty.
Filip Gruszczynskibe9dabd2016-01-19 12:23:10 -0800281 private boolean mReuseTask = false;
Craig Mautner362449a2014-06-20 14:04:39 -0700282
Craig Mautnerc0ffce52014-07-01 12:38:52 -0700283 CharSequence lastDescription; // Last description captured for this item.
284
Craig Mautnera228ae92014-07-09 05:44:55 -0700285 int mAffiliatedTaskId; // taskId of parent affiliation or self if no parent.
Winson Chungec396d62014-08-06 17:08:00 -0700286 int mAffiliatedTaskColor; // color of the parent task affiliation.
Craig Mautnera228ae92014-07-09 05:44:55 -0700287 TaskRecord mPrevAffiliate; // previous task in affiliated chain.
Wale Ogunwale18795a22014-12-03 11:38:33 -0800288 int mPrevAffiliateTaskId = INVALID_TASK_ID; // previous id for persistence.
Craig Mautnera228ae92014-07-09 05:44:55 -0700289 TaskRecord mNextAffiliate; // next task in affiliated chain.
Wale Ogunwale18795a22014-12-03 11:38:33 -0800290 int mNextAffiliateTaskId = INVALID_TASK_ID; // next id for persistence.
Craig Mautnera228ae92014-07-09 05:44:55 -0700291
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700292 // For relaunching the task from recents as though it was launched by the original launcher.
293 int mCallingUid;
294 String mCallingPackage;
295
Craig Mautner21d24a22014-04-23 11:45:37 -0700296 final ActivityManagerService mService;
297
Jorim Jaggi82c9dc92016-02-05 15:10:33 -0800298 private final Rect mTmpStableBounds = new Rect();
299 private final Rect mTmpNonDecorBounds = new Rect();
Wale Ogunwale9a08f822016-02-17 19:03:58 -0800300 private final Rect mTmpRect = new Rect();
Jorim Jaggi0a932142016-02-01 17:42:25 -0800301
Wale Ogunwale706ed792015-08-02 10:29:44 -0700302 // Last non-fullscreen bounds the task was launched in or resized to.
303 // The information is persisted and used to determine the appropriate stack to launch the
304 // task into on restore.
305 Rect mLastNonFullscreenBounds = null;
Andrii Kulian2e751b82016-03-16 16:59:32 -0700306 // Minimal width and height of this task when it's resizeable. -1 means it should use the
307 // default minimal width/height.
Andrii Kulianf66a83d2016-05-17 12:17:44 -0700308 int mMinWidth;
309 int mMinHeight;
Wale Ogunwalee4a0c572015-06-30 08:40:31 -0700310
Chong Zhangfdcc4d42015-10-14 16:50:12 -0700311 // Ranking (from top) of this task among all visible tasks. (-1 means it's not visible)
312 // This number will be assigned when we evaluate OOM scores for all visible tasks.
313 int mLayerRank = -1;
314
Andrii Kulian1779e612016-10-12 21:58:25 -0700315 /** Helper object used for updating override configuration. */
316 private Configuration mTmpConfig = new Configuration();
Wale Ogunwalee4a0c572015-06-30 08:40:31 -0700317
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800318 private TaskWindowContainerController mWindowContainerController;
319
Garfield Tan9b1efea2017-12-05 16:43:46 -0800320 /**
321 * Don't use constructor directly. Use {@link #create(ActivityManagerService, int, ActivityInfo,
322 * Intent, TaskDescription)} instead.
323 */
Craig Mautner21d24a22014-04-23 11:45:37 -0700324 TaskRecord(ActivityManagerService service, int _taskId, ActivityInfo info, Intent _intent,
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -0700325 IVoiceInteractionSession _voiceSession, IVoiceInteractor _voiceInteractor) {
Craig Mautner21d24a22014-04-23 11:45:37 -0700326 mService = service;
Suprabh Shukla23593142015-11-03 17:31:15 -0800327 userId = UserHandle.getUserId(info.applicationInfo.uid);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800328 taskId = _taskId;
Winson Chungfb44d212017-10-04 11:39:10 -0700329 lastActiveTime = SystemClock.elapsedRealtime();
Craig Mautnera228ae92014-07-09 05:44:55 -0700330 mAffiliatedTaskId = _taskId;
Dianne Hackborn91097de2014-04-04 18:02:06 -0700331 voiceSession = _voiceSession;
332 voiceInteractor = _voiceInteractor;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700333 isAvailable = true;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800334 mActivities = new ArrayList<>();
Craig Mautner15df08a2015-04-01 12:17:18 -0700335 mCallingUid = info.applicationInfo.uid;
336 mCallingPackage = info.packageName;
Martijn Coenend4a69702014-06-30 11:12:17 -0700337 setIntent(_intent, info);
Andrii Kulian2e751b82016-03-16 16:59:32 -0700338 setMinDimensions(info);
Winson730bf062016-03-31 18:04:56 -0700339 touchActiveTime();
Yorke Leebd54c2a2016-10-25 13:49:23 -0700340 mService.mTaskChangeNotificationController.notifyTaskCreated(_taskId, realActivity);
Craig Mautner21d24a22014-04-23 11:45:37 -0700341 }
342
Garfield Tan9b1efea2017-12-05 16:43:46 -0800343 /**
344 * Don't use constructor directly. Use {@link #create(ActivityManagerService, int, ActivityInfo,
345 * Intent, IVoiceInteractionSession, IVoiceInteractor)} instead.
346 */
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700347 TaskRecord(ActivityManagerService service, int _taskId, ActivityInfo info, Intent _intent,
Jorim Jaggie7d2b852017-08-28 17:55:15 +0200348 TaskDescription _taskDescription) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700349 mService = service;
Suprabh Shukla23593142015-11-03 17:31:15 -0800350 userId = UserHandle.getUserId(info.applicationInfo.uid);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700351 taskId = _taskId;
Winson Chungfb44d212017-10-04 11:39:10 -0700352 lastActiveTime = SystemClock.elapsedRealtime();
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700353 mAffiliatedTaskId = _taskId;
354 voiceSession = null;
355 voiceInteractor = null;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700356 isAvailable = true;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800357 mActivities = new ArrayList<>();
Craig Mautner15df08a2015-04-01 12:17:18 -0700358 mCallingUid = info.applicationInfo.uid;
359 mCallingPackage = info.packageName;
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700360 setIntent(_intent, info);
Andrii Kulian2e751b82016-03-16 16:59:32 -0700361 setMinDimensions(info);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700362
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700363 isPersistable = true;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700364 // Clamp to [1, max].
365 maxRecents = Math.min(Math.max(info.maxRecents, 1),
366 ActivityManager.getMaxAppRecentsLimitStatic());
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700367
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700368 lastTaskDescription = _taskDescription;
Winson730bf062016-03-31 18:04:56 -0700369 touchActiveTime();
Yorke Leebd54c2a2016-10-25 13:49:23 -0700370 mService.mTaskChangeNotificationController.notifyTaskCreated(_taskId, realActivity);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -0700371 }
372
Garfield Tan9b1efea2017-12-05 16:43:46 -0800373 /**
374 * Don't use constructor directly. This is only used by XML parser.
375 */
376 TaskRecord(ActivityManagerService service, int _taskId, Intent _intent,
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800377 Intent _affinityIntent, String _affinity, String _rootAffinity,
378 ComponentName _realActivity, ComponentName _origActivity, boolean _rootWasReset,
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -0700379 boolean _autoRemoveRecents, boolean _askedCompatMode, int _userId,
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800380 int _effectiveUid, String _lastDescription, ArrayList<ActivityRecord> activities,
Winson Chungfb44d212017-10-04 11:39:10 -0700381 long lastTimeMoved, boolean neverRelinquishIdentity,
382 TaskDescription _lastTaskDescription, int taskAffiliation, int prevTaskId,
383 int nextTaskId, int taskAffiliationColor, int callingUid, String callingPackage,
Charles He2bf28322017-10-12 22:24:49 +0100384 int resizeMode, boolean supportsPictureInPicture, boolean _realActivitySuspended,
385 boolean userSetupComplete, int minWidth, int minHeight) {
Craig Mautner21d24a22014-04-23 11:45:37 -0700386 mService = service;
387 taskId = _taskId;
388 intent = _intent;
389 affinityIntent = _affinityIntent;
390 affinity = _affinity;
Wale Ogunwale9d3de4c2015-02-01 16:49:44 -0800391 rootAffinity = _rootAffinity;
Craig Mautner21d24a22014-04-23 11:45:37 -0700392 voiceSession = null;
393 voiceInteractor = null;
394 realActivity = _realActivity;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800395 realActivitySuspended = _realActivitySuspended;
Craig Mautner21d24a22014-04-23 11:45:37 -0700396 origActivity = _origActivity;
397 rootWasReset = _rootWasReset;
Dianne Hackborn852975d2014-08-22 17:42:43 -0700398 isAvailable = true;
Dianne Hackborn13420f22014-07-18 15:43:56 -0700399 autoRemoveRecents = _autoRemoveRecents;
Craig Mautner21d24a22014-04-23 11:45:37 -0700400 askedCompatMode = _askedCompatMode;
Craig Mautner21d24a22014-04-23 11:45:37 -0700401 userId = _userId;
Wale Ogunwalef80170f2016-02-04 15:12:29 -0800402 mUserSetupComplete = userSetupComplete;
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700403 effectiveUid = _effectiveUid;
Winson Chungfb44d212017-10-04 11:39:10 -0700404 lastActiveTime = SystemClock.elapsedRealtime();
Craig Mautner21d24a22014-04-23 11:45:37 -0700405 lastDescription = _lastDescription;
406 mActivities = activities;
407 mLastTimeMoved = lastTimeMoved;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700408 mNeverRelinquishIdentity = neverRelinquishIdentity;
Winson Chung2cb86c72014-06-25 12:03:30 -0700409 lastTaskDescription = _lastTaskDescription;
Craig Mautnera228ae92014-07-09 05:44:55 -0700410 mAffiliatedTaskId = taskAffiliation;
Winson Chungec396d62014-08-06 17:08:00 -0700411 mAffiliatedTaskColor = taskAffiliationColor;
Craig Mautnera228ae92014-07-09 05:44:55 -0700412 mPrevAffiliateTaskId = prevTaskId;
413 mNextAffiliateTaskId = nextTaskId;
Craig Mautnerdc00cbe2014-07-20 17:48:47 -0700414 mCallingUid = callingUid;
415 mCallingPackage = callingPackage;
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800416 mResizeMode = resizeMode;
Winson Chungd3395382016-12-13 11:49:09 -0800417 mSupportsPictureInPicture = supportsPictureInPicture;
Andrii Kulianf66a83d2016-05-17 12:17:44 -0700418 mMinWidth = minWidth;
419 mMinHeight = minHeight;
Yorke Leebd54c2a2016-10-25 13:49:23 -0700420 mService.mTaskChangeNotificationController.notifyTaskCreated(_taskId, realActivity);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800421 }
422
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800423 TaskWindowContainerController getWindowContainerController() {
424 return mWindowContainerController;
425 }
426
427 void createWindowContainer(boolean onTop, boolean showForAllUsers) {
428 if (mWindowContainerController != null) {
429 throw new IllegalArgumentException("Window container=" + mWindowContainerController
430 + " already created for task=" + this);
431 }
432
433 final Rect bounds = updateOverrideConfigurationFromLaunchBounds();
Bryce Lee04ab3462017-04-10 15:06:33 -0700434 setWindowContainerController(new TaskWindowContainerController(taskId, this,
Wale Ogunwale034a8ec2017-09-02 17:14:40 -0700435 getStack().getWindowContainerController(), userId, bounds,
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -0700436 mResizeMode, mSupportsPictureInPicture, onTop,
437 showForAllUsers, lastTaskDescription));
Bryce Lee04ab3462017-04-10 15:06:33 -0700438 }
439
440 /**
441 * Should only be invoked from {@link #createWindowContainer(boolean, boolean)}.
442 */
443 @VisibleForTesting
444 protected void setWindowContainerController(TaskWindowContainerController controller) {
445 if (mWindowContainerController != null) {
446 throw new IllegalArgumentException("Window container=" + mWindowContainerController
447 + " already created for task=" + this);
448 }
449
450 mWindowContainerController = controller;
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800451 }
452
453 void removeWindowContainer() {
Charles Heff9b4dff2017-09-22 10:18:37 +0100454 mService.mLockTaskController.clearLockedTask(this);
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800455 mWindowContainerController.removeContainer();
Wale Ogunwale3382ab12017-07-27 08:55:03 -0700456 if (!getWindowConfiguration().persistTaskBounds()) {
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800457 // Reset current bounds for task whose bounds shouldn't be persisted so it uses
458 // default configuration the next time it launches.
459 updateOverrideConfiguration(null);
460 }
461 mService.mTaskChangeNotificationController.notifyTaskRemoved(taskId);
462 mWindowContainerController = null;
463 }
464
Jorim Jaggifb9d78a2017-01-05 18:57:12 +0100465 @Override
466 public void onSnapshotChanged(TaskSnapshot snapshot) {
467 mService.mTaskChangeNotificationController.notifyTaskSnapshotChanged(taskId, snapshot);
468 }
469
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800470 void setResizeMode(int resizeMode) {
471 if (mResizeMode == resizeMode) {
472 return;
473 }
474 mResizeMode = resizeMode;
475 mWindowContainerController.setResizeable(resizeMode);
476 mService.mStackSupervisor.ensureActivitiesVisibleLocked(null, 0, !PRESERVE_WINDOWS);
477 mService.mStackSupervisor.resumeFocusedStackTopActivityLocked();
478 }
479
480 void setTaskDockedResizing(boolean resizing) {
481 mWindowContainerController.setTaskDockedResizing(resizing);
482 }
483
Wale Ogunwale1666e312016-12-16 11:27:18 -0800484 // TODO: Consolidate this with the resize() method below.
485 @Override
486 public void requestResize(Rect bounds, int resizeMode) {
487 mService.resizeTask(taskId, bounds, resizeMode);
488 }
489
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800490 boolean resize(Rect bounds, int resizeMode, boolean preserveWindow, boolean deferResume) {
Bryce Leef3c6a472017-11-14 14:53:06 -0800491 mService.mWindowManager.deferSurfaceLayout();
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800492
Bryce Leef3c6a472017-11-14 14:53:06 -0800493 try {
494 if (!isResizeable()) {
495 Slog.w(TAG, "resizeTask: task " + this + " not resizeable.");
496 return true;
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800497 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800498
Bryce Leef3c6a472017-11-14 14:53:06 -0800499 // If this is a forced resize, let it go through even if the bounds is not changing,
500 // as we might need a relayout due to surface size change (to/from fullscreen).
501 final boolean forced = (resizeMode & RESIZE_MODE_FORCED) != 0;
502 if (equivalentOverrideBounds(bounds) && !forced) {
503 // Nothing to do here...
504 return true;
505 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800506
Bryce Leef3c6a472017-11-14 14:53:06 -0800507 if (mWindowContainerController == null) {
508 // Task doesn't exist in window manager yet (e.g. was restored from recents).
509 // All we can do for now is update the bounds so it can be used when the task is
510 // added to window manager.
511 updateOverrideConfiguration(bounds);
512 if (!inFreeformWindowingMode()) {
513 // re-restore the task so it can have the proper stack association.
514 mService.mStackSupervisor.restoreRecentTaskLocked(this, null, !ON_TOP);
515 }
516 return true;
517 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800518
Bryce Leef3c6a472017-11-14 14:53:06 -0800519 if (!canResizeToBounds(bounds)) {
520 throw new IllegalArgumentException("resizeTask: Can not resize task=" + this
521 + " to bounds=" + bounds + " resizeMode=" + mResizeMode);
522 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800523
Bryce Leef3c6a472017-11-14 14:53:06 -0800524 // Do not move the task to another stack here.
525 // This method assumes that the task is already placed in the right stack.
526 // we do not mess with that decision and we only do the resize!
527
528 Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "am.resizeTask_" + taskId);
529
530 final boolean updatedConfig = updateOverrideConfiguration(bounds);
531 // This variable holds information whether the configuration didn't change in a significant
532
533 // way and the activity was kept the way it was. If it's false, it means the activity
534 // had
535 // to be relaunched due to configuration change.
536 boolean kept = true;
537 if (updatedConfig) {
538 final ActivityRecord r = topRunningActivityLocked();
539 if (r != null && !deferResume) {
540 kept = r.ensureActivityConfigurationLocked(0 /* globalChanges */,
541 preserveWindow);
542 mService.mStackSupervisor.ensureActivitiesVisibleLocked(r, 0,
543 !PRESERVE_WINDOWS);
544 if (!kept) {
545 mService.mStackSupervisor.resumeFocusedStackTopActivityLocked();
546 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800547 }
548 }
Bryce Leef3c6a472017-11-14 14:53:06 -0800549 mWindowContainerController.resize(kept, forced);
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800550
Bryce Leef3c6a472017-11-14 14:53:06 -0800551 Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
552 return kept;
553 } finally {
554 mService.mWindowManager.continueSurfaceLayout();
555 }
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800556 }
557
558 // TODO: Investigate combining with the resize() method above.
559 void resizeWindowContainer() {
Bryce Leef3c6a472017-11-14 14:53:06 -0800560 mWindowContainerController.resize(false /* relayout */, false /* forced */);
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800561 }
562
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800563 void getWindowContainerBounds(Rect bounds) {
564 mWindowContainerController.getBounds(bounds);
565 }
566
Winson Chung74666102017-02-22 17:49:24 -0800567 /**
568 * Convenience method to reparent a task to the top or bottom position of the stack.
569 */
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700570 boolean reparent(ActivityStack preferredStack, boolean toTop,
571 @ReparentMoveStackMode int moveStackMode, boolean animate, boolean deferResume,
572 String reason) {
573 return reparent(preferredStack, toTop ? MAX_VALUE : 0, moveStackMode, animate, deferResume,
574 true /* schedulePictureInPictureModeChange */, reason);
Winson Chung5af42fc2017-03-24 17:11:33 -0700575 }
576
577 /**
578 * Convenience method to reparent a task to the top or bottom position of the stack, with
579 * an option to skip scheduling the picture-in-picture mode change.
580 */
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700581 boolean reparent(ActivityStack preferredStack, boolean toTop,
582 @ReparentMoveStackMode int moveStackMode, boolean animate, boolean deferResume,
583 boolean schedulePictureInPictureModeChange, String reason) {
584 return reparent(preferredStack, toTop ? MAX_VALUE : 0, moveStackMode, animate,
Winson Chung5af42fc2017-03-24 17:11:33 -0700585 deferResume, schedulePictureInPictureModeChange, reason);
586 }
587
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700588 /** Convenience method to reparent a task to a specific position of the stack. */
589 boolean reparent(ActivityStack preferredStack, int position,
590 @ReparentMoveStackMode int moveStackMode, boolean animate, boolean deferResume,
591 String reason) {
592 return reparent(preferredStack, position, moveStackMode, animate, deferResume,
Winson Chung5af42fc2017-03-24 17:11:33 -0700593 true /* schedulePictureInPictureModeChange */, reason);
Winson Chung74666102017-02-22 17:49:24 -0800594 }
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800595
Winson Chung74666102017-02-22 17:49:24 -0800596 /**
597 * Reparents the task into a preferred stack, creating it if necessary.
598 *
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700599 * @param preferredStack the target stack to move this task
Winson Chung74666102017-02-22 17:49:24 -0800600 * @param position the position to place this task in the new stack
601 * @param animate whether or not we should wait for the new window created as a part of the
Winson Chung5af42fc2017-03-24 17:11:33 -0700602 * reparenting to be drawn and animated in
Winson Chung74666102017-02-22 17:49:24 -0800603 * @param moveStackMode whether or not to move the stack to the front always, only if it was
Winson Chung5af42fc2017-03-24 17:11:33 -0700604 * previously focused & in front, or never
Winson Chung74666102017-02-22 17:49:24 -0800605 * @param deferResume whether or not to update the visibility of other tasks and stacks that may
Winson Chung5af42fc2017-03-24 17:11:33 -0700606 * have changed as a result of this reparenting
607 * @param schedulePictureInPictureModeChange specifies whether or not to schedule the PiP mode
608 * change. Callers may set this to false if they are explicitly scheduling PiP mode
609 * changes themselves, like during the PiP animation
Winson Chung74666102017-02-22 17:49:24 -0800610 * @param reason the caller of this reparenting
Winson Chung5af42fc2017-03-24 17:11:33 -0700611 * @return whether the task was reparented
Winson Chung74666102017-02-22 17:49:24 -0800612 */
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700613 // TODO: Inspect all call sites and change to just changing windowing mode of the stack vs.
Wale Ogunwale44f036f2017-09-29 05:09:09 -0700614 // re-parenting the task. Can only be done when we are no longer using static stack Ids.
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700615 boolean reparent(ActivityStack preferredStack, int position,
616 @ReparentMoveStackMode int moveStackMode, boolean animate, boolean deferResume,
617 boolean schedulePictureInPictureModeChange, String reason) {
Winson Chung74666102017-02-22 17:49:24 -0800618 final ActivityStackSupervisor supervisor = mService.mStackSupervisor;
619 final WindowManagerService windowManager = mService.mWindowManager;
620 final ActivityStack sourceStack = getStack();
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700621 final ActivityStack toStack = supervisor.getReparentTargetStack(this, preferredStack,
Winson Chung74666102017-02-22 17:49:24 -0800622 position == MAX_VALUE);
623 if (toStack == sourceStack) {
624 return false;
625 }
626
Wale Ogunwale44f036f2017-09-29 05:09:09 -0700627 final int toStackWindowingMode = toStack.getWindowingMode();
Winson Chung74666102017-02-22 17:49:24 -0800628 final ActivityRecord topActivity = getTopActivity();
629
Wale Ogunwale30e441d2017-11-09 08:28:45 -0800630 final boolean mightReplaceWindow = topActivity != null
631 && replaceWindowsOnTaskMove(getWindowingMode(), toStackWindowingMode);
Winson Chung74666102017-02-22 17:49:24 -0800632 if (mightReplaceWindow) {
633 // We are about to relaunch the activity because its configuration changed due to
634 // being maximized, i.e. size change. The activity will first remove the old window
635 // and then add a new one. This call will tell window manager about this, so it can
636 // preserve the old window until the new one is drawn. This prevents having a gap
637 // between the removal and addition, in which no window is visible. We also want the
638 // entrance of the new window to be properly animated.
639 // Note here we always set the replacing window first, as the flags might be needed
640 // during the relaunch. If we end up not doing any relaunch, we clear the flags later.
641 windowManager.setWillReplaceWindow(topActivity.appToken, animate);
642 }
643
644 windowManager.deferSurfaceLayout();
645 boolean kept = true;
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800646 try {
Winson Chung74666102017-02-22 17:49:24 -0800647 final ActivityRecord r = topRunningActivityLocked();
Winson Chung95f8f0e2017-03-24 09:20:17 -0700648 final boolean wasFocused = r != null && supervisor.isFocusedStack(sourceStack)
Winson Chung74666102017-02-22 17:49:24 -0800649 && (topRunningActivityLocked() == r);
Winson Chung95f8f0e2017-03-24 09:20:17 -0700650 final boolean wasResumed = r != null && sourceStack.mResumedActivity == r;
651 final boolean wasPaused = r != null && sourceStack.mPausingActivity == r;
Winson Chung74666102017-02-22 17:49:24 -0800652
653 // In some cases the focused stack isn't the front stack. E.g. pinned stack.
654 // Whenever we are moving the top activity from the front stack we want to make sure to
655 // move the stack to the front.
Wale Ogunwalea0f5b5e2017-10-11 09:37:23 -0700656 final boolean wasFront = r != null && sourceStack.isTopStackOnDisplay()
Winson Chung74666102017-02-22 17:49:24 -0800657 && (sourceStack.topRunningActivityLocked() == r);
658
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800659 // Adjust the position for the new parent stack as needed.
Winson Chung74666102017-02-22 17:49:24 -0800660 position = toStack.getAdjustedPositionForTask(this, position, null /* starting */);
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800661
662 // Must reparent first in window manager to avoid a situation where AM can delete the
663 // we are coming from in WM before we reparent because it became empty.
Wale Ogunwale2719cc12017-04-14 09:45:27 -0700664 mWindowContainerController.reparent(toStack.getWindowContainerController(), position,
665 moveStackMode == REPARENT_MOVE_STACK_TO_FRONT);
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800666
Wale Ogunwale56d8d162017-05-30 11:12:20 -0700667 final boolean moveStackToFront = moveStackMode == REPARENT_MOVE_STACK_TO_FRONT
668 || (moveStackMode == REPARENT_KEEP_STACK_AT_FRONT && (wasFocused || wasFront));
Winson Chung74666102017-02-22 17:49:24 -0800669 // Move the task
Wale Ogunwale56d8d162017-05-30 11:12:20 -0700670 sourceStack.removeTask(this, reason, moveStackToFront
671 ? REMOVE_TASK_MODE_MOVING_TO_TOP : REMOVE_TASK_MODE_MOVING);
Winson Chung5af42fc2017-03-24 17:11:33 -0700672 toStack.addTask(this, position, false /* schedulePictureInPictureModeChange */, reason);
Winson Chung74666102017-02-22 17:49:24 -0800673
Winson Chung5af42fc2017-03-24 17:11:33 -0700674 if (schedulePictureInPictureModeChange) {
675 // Notify of picture-in-picture mode changes
676 supervisor.scheduleUpdatePictureInPictureModeIfNeeded(this, sourceStack);
677 }
Winson Chung74666102017-02-22 17:49:24 -0800678
679 // TODO: Ensure that this is actually necessary here
680 // Notify the voice session if required
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800681 if (voiceSession != null) {
682 try {
683 voiceSession.taskStarted(intent, taskId);
684 } catch (RemoteException e) {
685 }
686 }
Winson Chung74666102017-02-22 17:49:24 -0800687
688 // If the task had focus before (or we're requested to move focus), move focus to the
689 // new stack by moving the stack to the front.
Winson Chung95f8f0e2017-03-24 09:20:17 -0700690 if (r != null) {
691 toStack.moveToFrontAndResumeStateIfNeeded(r, moveStackToFront, wasResumed,
692 wasPaused, reason);
693 }
Winson Chung74666102017-02-22 17:49:24 -0800694 if (!animate) {
695 toStack.mNoAnimActivities.add(topActivity);
696 }
697
698 // We might trigger a configuration change. Save the current task bounds for freezing.
699 // TODO: Should this call be moved inside the resize method in WM?
700 toStack.prepareFreezingTaskBounds();
701
702 // Make sure the task has the appropriate bounds/size for the stack it is in.
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700703 final boolean toStackSplitScreenPrimary =
704 toStackWindowingMode == WINDOWING_MODE_SPLIT_SCREEN_PRIMARY;
Bryce Leef3c6a472017-11-14 14:53:06 -0800705 final Rect configBounds = getOverrideBounds();
Wale Ogunwale44f036f2017-09-29 05:09:09 -0700706 if ((toStackWindowingMode == WINDOWING_MODE_FULLSCREEN
707 || toStackWindowingMode == WINDOWING_MODE_SPLIT_SCREEN_SECONDARY)
Bryce Leef3c6a472017-11-14 14:53:06 -0800708 && !Objects.equals(configBounds, toStack.getOverrideBounds())) {
709 kept = resize(toStack.getOverrideBounds(), RESIZE_MODE_SYSTEM, !mightReplaceWindow,
Winson Chung74666102017-02-22 17:49:24 -0800710 deferResume);
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700711 } else if (toStackWindowingMode == WINDOWING_MODE_FREEFORM) {
Winson Chung74666102017-02-22 17:49:24 -0800712 Rect bounds = getLaunchBounds();
713 if (bounds == null) {
Bryce Leedacefc42017-10-10 12:56:02 -0700714 mService.mStackSupervisor.getLaunchingBoundsController().layoutTask(this, null);
Bryce Leef3c6a472017-11-14 14:53:06 -0800715 bounds = configBounds;
Winson Chung74666102017-02-22 17:49:24 -0800716 }
717 kept = resize(bounds, RESIZE_MODE_FORCED, !mightReplaceWindow, deferResume);
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700718 } else if (toStackSplitScreenPrimary || toStackWindowingMode == WINDOWING_MODE_PINNED) {
719 if (toStackSplitScreenPrimary && moveStackMode == REPARENT_KEEP_STACK_AT_FRONT) {
Matthew Ng330757d2017-02-28 14:19:17 -0800720 // Move recents to front so it is not behind home stack when going into docked
721 // mode
722 mService.mStackSupervisor.moveRecentsStackToFront(reason);
723 }
Bryce Leef3c6a472017-11-14 14:53:06 -0800724 kept = resize(toStack.getOverrideBounds(), RESIZE_MODE_SYSTEM, !mightReplaceWindow,
Winson Chung74666102017-02-22 17:49:24 -0800725 deferResume);
726 }
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800727 } finally {
Winson Chung74666102017-02-22 17:49:24 -0800728 windowManager.continueSurfaceLayout();
Wale Ogunwalec5cc3012017-01-13 13:26:16 -0800729 }
Winson Chung74666102017-02-22 17:49:24 -0800730
731 if (mightReplaceWindow) {
732 // If we didn't actual do a relaunch (indicated by kept==true meaning we kept the old
733 // window), we need to clear the replace window settings. Otherwise, we schedule a
734 // timeout to remove the old window if the replacing window is not coming in time.
735 windowManager.scheduleClearWillReplaceWindows(topActivity.appToken, !kept);
736 }
737
738 if (!deferResume) {
739 // The task might have already been running and its visibility needs to be synchronized
740 // with the visibility of the stack / windows.
741 supervisor.ensureActivitiesVisibleLocked(null, 0, !mightReplaceWindow);
742 supervisor.resumeFocusedStackTopActivityLocked();
743 }
744
Andrii Kulian036e3ad2017-04-19 10:55:10 -0700745 // TODO: Handle incorrect request to move before the actual move, not after.
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700746 supervisor.handleNonResizableTaskIfNeeded(this, preferredStack.getWindowingMode(),
Wale Ogunwalea0f5b5e2017-10-11 09:37:23 -0700747 DEFAULT_DISPLAY, toStack);
Winson Chung74666102017-02-22 17:49:24 -0800748
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700749 boolean successful = (preferredStack == toStack);
750 if (successful && toStack.getWindowingMode() == WINDOWING_MODE_SPLIT_SCREEN_PRIMARY) {
Matthew Ng330757d2017-02-28 14:19:17 -0800751 // If task moved to docked stack - show recents if needed.
752 mService.mWindowManager.showRecentApps(false /* fromHome */);
753 }
754 return successful;
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800755 }
756
Wale Ogunwale44f036f2017-09-29 05:09:09 -0700757 /**
Wale Ogunwale30e441d2017-11-09 08:28:45 -0800758 * @return True if the windows of tasks being moved to the target stack from the source stack
759 * should be replaced, meaning that window manager will keep the old window around until the new
760 * is ready.
Wale Ogunwale44f036f2017-09-29 05:09:09 -0700761 */
762 private static boolean replaceWindowsOnTaskMove(
763 int sourceWindowingMode, int targetWindowingMode) {
764 return sourceWindowingMode == WINDOWING_MODE_FREEFORM
765 || targetWindowingMode == WINDOWING_MODE_FREEFORM;
766 }
767
Wale Ogunwalee1fe7fa22016-12-15 18:27:00 -0800768 void cancelWindowTransition() {
769 mWindowContainerController.cancelWindowTransition();
770 }
771
772 void cancelThumbnailTransition() {
773 mWindowContainerController.cancelThumbnailTransition();
774 }
775
Jorim Jaggi7361bab2017-01-16 17:17:58 +0100776 /**
777 * DO NOT HOLD THE ACTIVITY MANAGER LOCK WHEN CALLING THIS METHOD!
778 */
Jorim Jaggi35e3f532017-03-17 17:06:50 +0100779 TaskSnapshot getSnapshot(boolean reducedResolution) {
Jorim Jaggi7361bab2017-01-16 17:17:58 +0100780
781 // TODO: Move this to {@link TaskWindowContainerController} once recent tasks are more
782 // synchronized between AM and WM.
Jorim Jaggi35e3f532017-03-17 17:06:50 +0100783 return mService.mWindowManager.getTaskSnapshot(taskId, userId, reducedResolution);
Jorim Jaggi02886a82016-12-06 09:10:06 -0800784 }
785
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800786 void touchActiveTime() {
Winson Chungfb44d212017-10-04 11:39:10 -0700787 lastActiveTime = SystemClock.elapsedRealtime();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800788 }
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700789
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800790 long getInactiveDuration() {
Winson Chungfb44d212017-10-04 11:39:10 -0700791 return SystemClock.elapsedRealtime() - lastActiveTime;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800792 }
Craig Mautner9db9a0b2013-04-29 17:05:56 -0700793
Winson Chungfee26772014-08-05 12:21:52 -0700794 /** Sets the original intent, and the calling uid and package. */
795 void setIntent(ActivityRecord r) {
Winson Chungfee26772014-08-05 12:21:52 -0700796 mCallingUid = r.launchedFromUid;
797 mCallingPackage = r.launchedFromPackage;
Craig Mautner15df08a2015-04-01 12:17:18 -0700798 setIntent(r.intent, r.info);
Charles He2bf28322017-10-12 22:24:49 +0100799 setLockTaskAuth(r);
Winson Chungfee26772014-08-05 12:21:52 -0700800 }
801
802 /** Sets the original intent, _without_ updating the calling uid or package. */
803 private void setIntent(Intent _intent, ActivityInfo info) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700804 if (intent == null) {
805 mNeverRelinquishIdentity =
Wale Ogunwale3eadad72016-10-13 09:16:59 -0700806 (info.flags & FLAG_RELINQUISH_TASK_IDENTITY) == 0;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -0700807 } else if (mNeverRelinquishIdentity) {
808 return;
809 }
810
811 affinity = info.taskAffinity;
Dianne Hackborn79228822014-09-16 11:11:23 -0700812 if (intent == null) {
813 // If this task already has an intent associated with it, don't set the root
814 // affinity -- we don't want it changing after initially set, but the initially
815 // set value may be null.
816 rootAffinity = affinity;
817 }
Dianne Hackborn885fbe52014-08-23 15:23:58 -0700818 effectiveUid = info.applicationInfo.uid;
Dianne Hackborn1d442e02009-04-20 18:14:05 -0700819 stringName = null;
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800820
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800821 if (info.targetActivity == null) {
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800822 if (_intent != null) {
823 // If this Intent has a selector, we want to clear it for the
824 // recent task since it is not relevant if the user later wants
825 // to re-launch the app.
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700826 if (_intent.getSelector() != null || _intent.getSourceBounds() != null) {
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800827 _intent = new Intent(_intent);
828 _intent.setSelector(null);
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700829 _intent.setSourceBounds(null);
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800830 }
831 }
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700832 if (DEBUG_TASKS) Slog.v(TAG_TASKS, "Setting Intent of " + this + " to " + _intent);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800833 intent = _intent;
834 realActivity = _intent != null ? _intent.getComponent() : null;
835 origActivity = null;
836 } else {
837 ComponentName targetComponent = new ComponentName(
838 info.packageName, info.targetActivity);
839 if (_intent != null) {
840 Intent targetIntent = new Intent(_intent);
841 targetIntent.setComponent(targetComponent);
Dianne Hackbornf5b86712011-12-05 17:42:41 -0800842 targetIntent.setSelector(null);
Dianne Hackbornd367ca82012-05-07 15:49:39 -0700843 targetIntent.setSourceBounds(null);
Wale Ogunwaleee006da2015-03-30 14:49:25 -0700844 if (DEBUG_TASKS) Slog.v(TAG_TASKS,
Dianne Hackborn7f96b792012-05-29 18:46:45 -0700845 "Setting Intent of " + this + " to target " + targetIntent);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800846 intent = targetIntent;
847 realActivity = targetComponent;
848 origActivity = _intent.getComponent();
849 } else {
850 intent = null;
851 realActivity = targetComponent;
852 origActivity = new ComponentName(info.packageName, info.name);
853 }
854 }
Amith Yamasani742a6712011-05-04 14:49:28 -0700855
Craig Mautner47b20ba2014-09-17 17:23:44 -0700856 final int intentFlags = intent == null ? 0 : intent.getFlags();
857 if ((intentFlags & Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED) != 0) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800858 // Once we are set to an Intent with this flag, we count this
859 // task as having a true root activity.
860 rootWasReset = true;
861 }
Dianne Hackborn09233282014-04-30 11:33:59 -0700862 userId = UserHandle.getUserId(info.applicationInfo.uid);
Winson Chung36f3f032016-09-08 23:29:43 +0000863 mUserSetupComplete = Settings.Secure.getIntForUser(mService.mContext.getContentResolver(),
864 USER_SETUP_COMPLETE, 0, userId) != 0;
Craig Mautner41db4a72014-05-07 17:20:56 -0700865 if ((info.flags & ActivityInfo.FLAG_AUTO_REMOVE_FROM_RECENTS) != 0) {
Dianne Hackborn13420f22014-07-18 15:43:56 -0700866 // If the activity itself has requested auto-remove, then just always do it.
867 autoRemoveRecents = true;
Wale Ogunwale843bfb92015-03-27 11:06:48 -0700868 } else if ((intentFlags & (FLAG_ACTIVITY_NEW_DOCUMENT | FLAG_ACTIVITY_RETAIN_IN_RECENTS))
869 == FLAG_ACTIVITY_NEW_DOCUMENT) {
Dianne Hackborn13420f22014-07-18 15:43:56 -0700870 // If the caller has not asked for the document to be retained, then we may
871 // want to turn on auto-remove, depending on whether the target has set its
872 // own document launch mode.
873 if (info.documentLaunchMode != ActivityInfo.DOCUMENT_LAUNCH_NONE) {
874 autoRemoveRecents = false;
875 } else {
876 autoRemoveRecents = true;
877 }
878 } else {
879 autoRemoveRecents = false;
Craig Mautner41db4a72014-05-07 17:20:56 -0700880 }
Wale Ogunwaleb1faf602016-01-27 09:12:31 -0800881 mResizeMode = info.resizeMode;
Winson Chungd3395382016-12-13 11:49:09 -0800882 mSupportsPictureInPicture = info.supportsPictureInPicture();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800883 }
Craig Mautner5d9c7be2013-02-15 14:02:56 -0800884
Andrii Kulian2e751b82016-03-16 16:59:32 -0700885 /** Sets the original minimal width and height. */
886 private void setMinDimensions(ActivityInfo info) {
887 if (info != null && info.windowLayout != null) {
Andrii Kulianf66a83d2016-05-17 12:17:44 -0700888 mMinWidth = info.windowLayout.minWidth;
889 mMinHeight = info.windowLayout.minHeight;
Andrii Kulian2e751b82016-03-16 16:59:32 -0700890 } else {
Andrii Kulianf66a83d2016-05-17 12:17:44 -0700891 mMinWidth = INVALID_MIN_SIZE;
892 mMinHeight = INVALID_MIN_SIZE;
Andrii Kulian2e751b82016-03-16 16:59:32 -0700893 }
894 }
895
Wale Ogunwale715a1dc2016-02-29 14:27:32 -0800896 /**
Andrii Kulian206b9fa2016-06-02 13:18:01 -0700897 * Return true if the input activity has the same intent filter as the intent this task
Wale Ogunwale715a1dc2016-02-29 14:27:32 -0800898 * record is based on (normally the root activity intent).
899 */
Andrii Kulian206b9fa2016-06-02 13:18:01 -0700900 boolean isSameIntentFilter(ActivityRecord r) {
Wale Ogunwale715a1dc2016-02-29 14:27:32 -0800901 final Intent intent = new Intent(r.intent);
902 // Correct the activity intent for aliasing. The task record intent will always be based on
903 // the real activity that will be launched not the alias, so we need to use an intent with
904 // the component name pointing to the real activity not the alias in the activity record.
905 intent.setComponent(r.realActivity);
906 return this.intent.filterEquals(intent);
907 }
908
Wale Ogunwale66e16852017-10-19 13:35:52 -0700909 boolean returnsToHomeStack() {
910 final int returnHomeFlags = FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_TASK_ON_HOME;
911 return (intent.getFlags() & returnHomeFlags) == returnHomeFlags;
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -0700912 }
913
Craig Mautnera228ae92014-07-09 05:44:55 -0700914 void setPrevAffiliate(TaskRecord prevAffiliate) {
915 mPrevAffiliate = prevAffiliate;
Wale Ogunwale18795a22014-12-03 11:38:33 -0800916 mPrevAffiliateTaskId = prevAffiliate == null ? INVALID_TASK_ID : prevAffiliate.taskId;
Craig Mautnera228ae92014-07-09 05:44:55 -0700917 }
918
919 void setNextAffiliate(TaskRecord nextAffiliate) {
920 mNextAffiliate = nextAffiliate;
Wale Ogunwale18795a22014-12-03 11:38:33 -0800921 mNextAffiliateTaskId = nextAffiliate == null ? INVALID_TASK_ID : nextAffiliate.taskId;
Craig Mautnera228ae92014-07-09 05:44:55 -0700922 }
923
Wale Ogunwale04a05ac2017-09-17 21:35:02 -0700924 <T extends ActivityStack> T getStack() {
925 return (T) mStack;
Andrii Kulian02b7a832016-10-06 23:11:56 -0700926 }
927
Andrii Kulianfb1bf692017-01-17 11:17:34 -0800928 /**
929 * Must be used for setting parent stack because it performs configuration updates.
930 * Must be called after adding task as a child to the stack.
931 */
Andrii Kulian02b7a832016-10-06 23:11:56 -0700932 void setStack(ActivityStack stack) {
Andrii Kulianfb1bf692017-01-17 11:17:34 -0800933 if (stack != null && !stack.isInStackLocked(this)) {
934 throw new IllegalStateException("Task must be added as a Stack child first.");
935 }
Andrii Kulian02b7a832016-10-06 23:11:56 -0700936 mStack = stack;
Andrii Kulian1779e612016-10-12 21:58:25 -0700937 onParentChanged();
Andrii Kulian02b7a832016-10-06 23:11:56 -0700938 }
939
940 /**
941 * @return Id of current stack, {@link INVALID_STACK_ID} if no stack is set.
942 */
943 int getStackId() {
944 return mStack != null ? mStack.mStackId : INVALID_STACK_ID;
945 }
946
Andrii Kulian1779e612016-10-12 21:58:25 -0700947 @Override
948 protected int getChildCount() {
Wale Ogunwale55ddf8f2017-03-20 08:56:38 -0700949 return mActivities.size();
Andrii Kulian1779e612016-10-12 21:58:25 -0700950 }
951
952 @Override
953 protected ConfigurationContainer getChildAt(int index) {
Wale Ogunwale55ddf8f2017-03-20 08:56:38 -0700954 return mActivities.get(index);
Andrii Kulian1779e612016-10-12 21:58:25 -0700955 }
956
957 @Override
958 protected ConfigurationContainer getParent() {
959 return mStack;
960 }
961
Andrii Kulianfb1bf692017-01-17 11:17:34 -0800962 @Override
Wale Ogunwale98d62312017-07-12 09:24:56 -0700963 protected void onParentChanged() {
Andrii Kulianfb1bf692017-01-17 11:17:34 -0800964 super.onParentChanged();
965 mService.mStackSupervisor.updateUIDsPresentOnDisplay();
966 }
967
Craig Mautnera228ae92014-07-09 05:44:55 -0700968 // Close up recents linked list.
Wale Ogunwale55ddf8f2017-03-20 08:56:38 -0700969 private void closeRecentsChain() {
Craig Mautnera228ae92014-07-09 05:44:55 -0700970 if (mPrevAffiliate != null) {
971 mPrevAffiliate.setNextAffiliate(mNextAffiliate);
972 }
973 if (mNextAffiliate != null) {
974 mNextAffiliate.setPrevAffiliate(mPrevAffiliate);
975 }
976 setPrevAffiliate(null);
977 setNextAffiliate(null);
978 }
979
Winson Chung740c3ac2014-11-12 16:14:38 -0800980 void removedFromRecents() {
Dianne Hackborn852975d2014-08-22 17:42:43 -0700981 closeRecentsChain();
982 if (inRecents) {
983 inRecents = false;
Winson Chung740c3ac2014-11-12 16:14:38 -0800984 mService.notifyTaskPersisterLocked(this, false);
Dianne Hackborn852975d2014-08-22 17:42:43 -0700985 }
Jorim Jaggif9084ec2017-01-16 13:16:59 +0100986
Dianne Hackborn68a06332017-11-15 17:54:18 -0800987 clearRootProcess();
988
Jorim Jaggif9084ec2017-01-16 13:16:59 +0100989 // TODO: Use window container controller once tasks are better synced between AM and WM
990 mService.mWindowManager.notifyTaskRemovedFromRecents(taskId, userId);
Dianne Hackborn852975d2014-08-22 17:42:43 -0700991 }
992
Craig Mautnera228ae92014-07-09 05:44:55 -0700993 void setTaskToAffiliateWith(TaskRecord taskToAffiliateWith) {
994 closeRecentsChain();
995 mAffiliatedTaskId = taskToAffiliateWith.mAffiliatedTaskId;
Winson Chungec396d62014-08-06 17:08:00 -0700996 mAffiliatedTaskColor = taskToAffiliateWith.mAffiliatedTaskColor;
Craig Mautnera228ae92014-07-09 05:44:55 -0700997 // Find the end
998 while (taskToAffiliateWith.mNextAffiliate != null) {
999 final TaskRecord nextRecents = taskToAffiliateWith.mNextAffiliate;
1000 if (nextRecents.mAffiliatedTaskId != mAffiliatedTaskId) {
1001 Slog.e(TAG, "setTaskToAffiliateWith: nextRecents=" + nextRecents + " affilTaskId="
1002 + nextRecents.mAffiliatedTaskId + " should be " + mAffiliatedTaskId);
1003 if (nextRecents.mPrevAffiliate == taskToAffiliateWith) {
1004 nextRecents.setPrevAffiliate(null);
1005 }
1006 taskToAffiliateWith.setNextAffiliate(null);
1007 break;
1008 }
1009 taskToAffiliateWith = nextRecents;
1010 }
1011 taskToAffiliateWith.setNextAffiliate(this);
1012 setPrevAffiliate(taskToAffiliateWith);
1013 setNextAffiliate(null);
1014 }
1015
Winson Chung1147c402014-05-14 11:05:00 -07001016 /** Returns the intent for the root activity for this task */
1017 Intent getBaseIntent() {
1018 return intent != null ? intent : affinityIntent;
1019 }
1020
Winson Chung3b3f4642014-04-22 10:08:18 -07001021 /** Returns the first non-finishing activity from the root. */
1022 ActivityRecord getRootActivity() {
1023 for (int i = 0; i < mActivities.size(); i++) {
1024 final ActivityRecord r = mActivities.get(i);
1025 if (r.finishing) {
1026 continue;
1027 }
1028 return r;
1029 }
1030 return null;
1031 }
1032
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001033 ActivityRecord getTopActivity() {
Bryce Lee9f6affd2017-09-01 09:18:35 -07001034 return getTopActivity(true /* includeOverlays */);
1035 }
1036
1037 ActivityRecord getTopActivity(boolean includeOverlays) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001038 for (int i = mActivities.size() - 1; i >= 0; --i) {
1039 final ActivityRecord r = mActivities.get(i);
Bryce Lee9f6affd2017-09-01 09:18:35 -07001040 if (r.finishing || (!includeOverlays && r.mTaskOverlay)) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001041 continue;
1042 }
1043 return r;
1044 }
1045 return null;
1046 }
1047
Filip Gruszczynski3e85ba22015-10-05 22:48:30 -07001048 ActivityRecord topRunningActivityLocked() {
Andrii Kulian02b7a832016-10-06 23:11:56 -07001049 if (mStack != null) {
Wale Ogunwale7d701172015-03-11 15:36:30 -07001050 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
1051 ActivityRecord r = mActivities.get(activityNdx);
Chong Zhang87761972016-08-22 13:53:24 -07001052 if (!r.finishing && r.okToShowLocked()) {
Wale Ogunwale7d701172015-03-11 15:36:30 -07001053 return r;
1054 }
Craig Mautner6b74cb52013-09-27 17:02:21 -07001055 }
1056 }
1057 return null;
1058 }
1059
Jorim Jaggi172e99f2017-10-20 14:33:18 +02001060 boolean isVisible() {
1061 for (int i = mActivities.size() - 1; i >= 0; --i) {
1062 final ActivityRecord r = mActivities.get(i);
1063 if (r.visible) {
1064 return true;
1065 }
1066 }
1067 return false;
1068 }
1069
Jorim Jaggiea039a82017-08-02 14:37:49 +02001070 void getAllRunningVisibleActivitiesLocked(ArrayList<ActivityRecord> outActivities) {
1071 if (mStack != null) {
1072 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
1073 ActivityRecord r = mActivities.get(activityNdx);
Jorim Jaggi02f1d2f2017-08-04 14:29:16 +02001074 if (!r.finishing && r.okToShowLocked() && r.visibleIgnoringKeyguard) {
Jorim Jaggiea039a82017-08-02 14:37:49 +02001075 outActivities.add(r);
1076 }
1077 }
1078 }
1079 }
1080
Wale Ogunwale3b232392016-05-13 15:37:13 -07001081 ActivityRecord topRunningActivityWithStartingWindowLocked() {
Andrii Kulian02b7a832016-10-06 23:11:56 -07001082 if (mStack != null) {
Wale Ogunwale3b232392016-05-13 15:37:13 -07001083 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
1084 ActivityRecord r = mActivities.get(activityNdx);
1085 if (r.mStartingWindowState != STARTING_WINDOW_SHOWN
Chong Zhang87761972016-08-22 13:53:24 -07001086 || r.finishing || !r.okToShowLocked()) {
Wale Ogunwale3b232392016-05-13 15:37:13 -07001087 continue;
1088 }
1089 return r;
1090 }
1091 }
1092 return null;
1093 }
1094
Winson Chung61c9e5a2017-10-11 10:39:32 -07001095 /**
1096 * Return the number of running activities, and the number of non-finishing/initializing
1097 * activities in the provided {@param reportOut} respectively.
1098 */
1099 void getNumRunningActivities(TaskActivitiesReport reportOut) {
1100 reportOut.reset();
1101 for (int i = mActivities.size() - 1; i >= 0; --i) {
1102 final ActivityRecord r = mActivities.get(i);
1103 if (r.finishing) {
1104 continue;
1105 }
1106
1107 reportOut.base = r;
1108
1109 // Increment the total number of non-finishing activities
1110 reportOut.numActivities++;
1111
1112 if (reportOut.top == null || (reportOut.top.state == ActivityState.INITIALIZING)) {
1113 reportOut.top = r;
1114 // Reset the number of running activities until we hit the first non-initializing
1115 // activity
1116 reportOut.numRunning = 0;
1117 }
1118 if (r.app != null && r.app.thread != null) {
1119 // Increment the number of actually running activities
1120 reportOut.numRunning++;
1121 }
1122 }
1123 }
1124
Chong Zhang87761972016-08-22 13:53:24 -07001125 boolean okToShowLocked() {
1126 // NOTE: If {@link TaskRecord#topRunningActivityLocked} return is not null then it is
1127 // okay to show the activity when locked.
1128 return mService.mStackSupervisor.isCurrentProfileLocked(userId)
1129 || topRunningActivityLocked() != null;
1130 }
1131
Craig Mautner3b475fe2013-12-16 15:58:31 -08001132 /** Call after activity movement or finish to make sure that frontOfTask is set correctly */
Bryce Leed71317c2017-02-07 14:27:22 -08001133 final void setFrontOfTask() {
1134 boolean foundFront = false;
Craig Mautner3b475fe2013-12-16 15:58:31 -08001135 final int numActivities = mActivities.size();
Craig Mautner704e40b2013-12-18 16:43:51 -08001136 for (int activityNdx = 0; activityNdx < numActivities; ++activityNdx) {
Craig Mautner3b475fe2013-12-16 15:58:31 -08001137 final ActivityRecord r = mActivities.get(activityNdx);
1138 if (foundFront || r.finishing) {
1139 r.frontOfTask = false;
1140 } else {
1141 r.frontOfTask = true;
1142 // Set frontOfTask false for every following activity.
1143 foundFront = true;
1144 }
1145 }
Craig Mautner9587ee02014-06-23 15:00:10 +00001146 if (!foundFront && numActivities > 0) {
1147 // All activities of this task are finishing. As we ought to have a frontOfTask
1148 // activity, make the bottom activity front.
1149 mActivities.get(0).frontOfTask = true;
1150 }
Craig Mautner3b475fe2013-12-16 15:58:31 -08001151 }
1152
Craig Mautnerde4ef022013-04-07 19:01:33 -07001153 /**
Craig Mautner3b475fe2013-12-16 15:58:31 -08001154 * Reorder the history stack so that the passed activity is brought to the front.
Craig Mautnerde4ef022013-04-07 19:01:33 -07001155 */
1156 final void moveActivityToFrontLocked(ActivityRecord newTop) {
Wale Ogunwale0fc365c2015-05-25 19:35:42 -07001157 if (DEBUG_ADD_REMOVE) Slog.i(TAG_ADD_REMOVE,
1158 "Removing and adding activity " + newTop
1159 + " to stack at top callers=" + Debug.getCallers(4));
Craig Mautnerde4ef022013-04-07 19:01:33 -07001160
Craig Mautnerde4ef022013-04-07 19:01:33 -07001161 mActivities.remove(newTop);
1162 mActivities.add(newTop);
Bryce Leed58d7b32017-09-08 15:55:22 -07001163
1164 // Make sure window manager is aware of the position change.
1165 mWindowContainerController.positionChildAtTop(newTop.mWindowContainerController);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001166 updateEffectiveIntent();
Craig Mautner3b475fe2013-12-16 15:58:31 -08001167
Bryce Leed71317c2017-02-07 14:27:22 -08001168 setFrontOfTask();
Craig Mautnerde4ef022013-04-07 19:01:33 -07001169 }
1170
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001171 void addActivityAtBottom(ActivityRecord r) {
Craig Mautner77878772013-03-04 19:46:24 -08001172 addActivityAtIndex(0, r);
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001173 }
1174
1175 void addActivityToTop(ActivityRecord r) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001176 addActivityAtIndex(mActivities.size(), r);
1177 }
1178
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -07001179 @Override
Wale Ogunwaleeea34ee92017-08-31 20:07:45 -07001180 /*@WindowConfiguration.ActivityType*/
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -07001181 public int getActivityType() {
1182 final int applicationType = super.getActivityType();
1183 if (applicationType != ACTIVITY_TYPE_UNDEFINED || mActivities.isEmpty()) {
1184 return applicationType;
1185 }
1186 return mActivities.get(0).getActivityType();
1187 }
1188
Winson Chung30480042017-01-26 10:55:34 -08001189 /**
1190 * Adds an activity {@param r} at the given {@param index}. The activity {@param r} must either
1191 * be in the current task or unparented to any task.
1192 */
Craig Mautner1602ec22013-05-12 10:24:27 -07001193 void addActivityAtIndex(int index, ActivityRecord r) {
Bryce Leeaf691c02017-03-20 14:20:22 -07001194 TaskRecord task = r.getTask();
1195 if (task != null && task != this) {
Wale Ogunwalea0cd15e2017-02-01 15:33:08 -08001196 throw new IllegalArgumentException("Can not add r=" + " to task=" + this
Bryce Leeaf691c02017-03-20 14:20:22 -07001197 + " current parent=" + task);
Wale Ogunwalea0cd15e2017-02-01 15:33:08 -08001198 }
Bryce Leeaf691c02017-03-20 14:20:22 -07001199
1200 r.setTask(this);
Wale Ogunwalea0cd15e2017-02-01 15:33:08 -08001201
Craig Mautner6170f732013-04-02 13:05:23 -07001202 // Remove r first, and if it wasn't already in the list and it's fullscreen, count it.
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001203 if (!mActivities.remove(r) && r.fullscreen) {
1204 // Was not previously in list.
1205 numFullscreen++;
1206 }
Craig Mautner2c1faed2013-07-23 12:56:02 -07001207 // Only set this based on the first activity
1208 if (mActivities.isEmpty()) {
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -07001209 if (r.getActivityType() == ACTIVITY_TYPE_UNDEFINED) {
1210 // Normally non-standard activity type for the activity record will be set when the
1211 // object is created, however we delay setting the standard application type until
1212 // this point so that the task can set the type for additional activities added in
1213 // the else condition below.
1214 r.setActivityType(ACTIVITY_TYPE_STANDARD);
1215 }
1216 setActivityType(r.getActivityType());
Craig Mautner21d24a22014-04-23 11:45:37 -07001217 isPersistable = r.isPersistable();
Craig Mautnerdc00cbe2014-07-20 17:48:47 -07001218 mCallingUid = r.launchedFromUid;
1219 mCallingPackage = r.launchedFromPackage;
Dianne Hackborn852975d2014-08-22 17:42:43 -07001220 // Clamp to [1, max].
1221 maxRecents = Math.min(Math.max(r.info.maxRecents, 1),
1222 ActivityManager.getMaxAppRecentsLimitStatic());
Craig Mautner2c1faed2013-07-23 12:56:02 -07001223 } else {
1224 // Otherwise make all added activities match this one.
Wale Ogunwale6fbde9f2017-08-24 07:24:12 -07001225 r.setActivityType(getActivityType());
Craig Mautner78733002013-06-10 13:54:49 -07001226 }
Wale Ogunwale3b232392016-05-13 15:37:13 -07001227
1228 final int size = mActivities.size();
1229
1230 if (index == size && size > 0) {
1231 final ActivityRecord top = mActivities.get(size - 1);
1232 if (top.mTaskOverlay) {
1233 // Place below the task overlay activity since the overlay activity should always
1234 // be on top.
1235 index--;
1236 }
1237 }
1238
Wale Ogunwalea0cd15e2017-02-01 15:33:08 -08001239 index = Math.min(size, index);
Craig Mautner77878772013-03-04 19:46:24 -08001240 mActivities.add(index, r);
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001241 updateEffectiveIntent();
Craig Mautner21d24a22014-04-23 11:45:37 -07001242 if (r.isPersistable()) {
1243 mService.notifyTaskPersisterLocked(this, false);
1244 }
Wale Ogunwale26c0dfe2016-12-14 14:42:30 -08001245
1246 // Sync. with window manager
1247 updateOverrideConfigurationFromLaunchBounds();
Wale Ogunwalea0cd15e2017-02-01 15:33:08 -08001248 final AppWindowContainerController appController = r.getWindowContainerController();
1249 if (appController != null) {
1250 // Only attempt to move in WM if the child has a controller. It is possible we haven't
1251 // created controller for the activity we are starting yet.
1252 mWindowContainerController.positionChildAt(appController, index);
1253 }
David Stevens82ea6cb2017-03-03 16:18:50 -08001254
1255 // Make sure the list of display UID whitelists is updated
1256 // now that this record is in a new task.
1257 mService.mStackSupervisor.updateUIDsPresentOnDisplay();
Craig Mautner77878772013-03-04 19:46:24 -08001258 }
1259
Bryce Leeaa5e8c32017-03-01 16:01:06 -08001260 /**
Bryce Leeaf691c02017-03-20 14:20:22 -07001261 * Removes the specified activity from this task.
1262 * @param r The {@link ActivityRecord} to remove.
1263 * @return true if this was the last activity in the task.
Bryce Leeaa5e8c32017-03-01 16:01:06 -08001264 */
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001265 boolean removeActivity(ActivityRecord r) {
Bryce Leeaf691c02017-03-20 14:20:22 -07001266 return removeActivity(r, false /*reparenting*/);
1267 }
1268
1269 boolean removeActivity(ActivityRecord r, boolean reparenting) {
1270 if (r.getTask() != this) {
Bryce Leeaa5e8c32017-03-01 16:01:06 -08001271 throw new IllegalArgumentException(
1272 "Activity=" + r + " does not belong to task=" + this);
1273 }
1274
Bryce Leeaf691c02017-03-20 14:20:22 -07001275 r.setTask(null /*task*/, reparenting);
Bryce Leeaa5e8c32017-03-01 16:01:06 -08001276
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001277 if (mActivities.remove(r) && r.fullscreen) {
1278 // Was previously in list.
1279 numFullscreen--;
1280 }
Craig Mautner21d24a22014-04-23 11:45:37 -07001281 if (r.isPersistable()) {
1282 mService.notifyTaskPersisterLocked(this, false);
1283 }
Wale Ogunwale89182d52016-03-11 10:38:36 -08001284
Wale Ogunwale44f036f2017-09-29 05:09:09 -07001285 if (inPinnedWindowingMode()) {
Wale Ogunwale89182d52016-03-11 10:38:36 -08001286 // We normally notify listeners of task stack changes on pause, however pinned stack
1287 // activities are normally in the paused state so no notification will be sent there
1288 // before the activity is removed. We send it here so instead.
Yorke Leebd54c2a2016-10-25 13:49:23 -07001289 mService.mTaskChangeNotificationController.notifyTaskStackChanged();
Wale Ogunwale89182d52016-03-11 10:38:36 -08001290 }
1291
Craig Mautner41326202014-06-20 14:38:21 -07001292 if (mActivities.isEmpty()) {
Craig Mautner5afcd4d2014-06-21 18:11:33 -07001293 return !mReuseTask;
Craig Mautner41326202014-06-20 14:38:21 -07001294 }
1295 updateEffectiveIntent();
1296 return false;
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001297 }
1298
Winson Chung6954fc92017-03-24 16:22:12 -07001299 /**
1300 * @return whether or not there are ONLY task overlay activities in the stack.
1301 * If {@param excludeFinishing} is set, then ignore finishing activities in the check.
1302 * If there are no task overlay activities, this call returns false.
1303 */
1304 boolean onlyHasTaskOverlayActivities(boolean excludeFinishing) {
1305 int count = 0;
1306 for (int i = mActivities.size() - 1; i >= 0; i--) {
1307 final ActivityRecord r = mActivities.get(i);
1308 if (excludeFinishing && r.finishing) {
1309 continue;
1310 }
1311 if (!r.mTaskOverlay) {
1312 return false;
1313 }
1314 count++;
1315 }
1316 return count > 0;
1317 }
1318
Craig Mautner41db4a72014-05-07 17:20:56 -07001319 boolean autoRemoveFromRecents() {
Dianne Hackbornd38aed82014-06-10 21:36:35 -07001320 // We will automatically remove the task either if it has explicitly asked for
1321 // this, or it is empty and has never contained an activity that got shown to
1322 // the user.
Dianne Hackborn13420f22014-07-18 15:43:56 -07001323 return autoRemoveRecents || (mActivities.isEmpty() && !hasBeenVisible);
Craig Mautner41db4a72014-05-07 17:20:56 -07001324 }
1325
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001326 /**
1327 * Completely remove all activities associated with an existing
1328 * task starting at a specified index.
1329 */
Winson Chung0ec2a352017-10-26 11:38:30 -07001330 final void performClearTaskAtIndexLocked(int activityNdx, boolean pauseImmediately,
1331 String reason) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001332 int numActivities = mActivities.size();
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001333 for ( ; activityNdx < numActivities; ++activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001334 final ActivityRecord r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001335 if (r.finishing) {
1336 continue;
1337 }
Andrii Kulian02b7a832016-10-06 23:11:56 -07001338 if (mStack == null) {
Craig Mautner21d24a22014-04-23 11:45:37 -07001339 // Task was restored from persistent storage.
1340 r.takeFromHistory();
1341 mActivities.remove(activityNdx);
1342 --activityNdx;
1343 --numActivities;
Winson Chung6954fc92017-03-24 16:22:12 -07001344 } else if (mStack.finishActivityLocked(r, Activity.RESULT_CANCELED, null,
Winson Chung0ec2a352017-10-26 11:38:30 -07001345 reason, false, pauseImmediately)) {
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001346 --activityNdx;
1347 --numActivities;
1348 }
1349 }
1350 }
1351
1352 /**
1353 * Completely remove all activities associated with an existing task.
1354 */
Benjamin Franza83859f2017-07-03 16:34:14 +01001355 void performClearTaskLocked() {
Craig Mautner362449a2014-06-20 14:04:39 -07001356 mReuseTask = true;
Winson Chung0ec2a352017-10-26 11:38:30 -07001357 performClearTaskAtIndexLocked(0, !PAUSE_IMMEDIATELY, "clear-task-all");
Craig Mautner362449a2014-06-20 14:04:39 -07001358 mReuseTask = false;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001359 }
1360
Filip Gruszczynskibe9dabd2016-01-19 12:23:10 -08001361 ActivityRecord performClearTaskForReuseLocked(ActivityRecord newR, int launchFlags) {
1362 mReuseTask = true;
1363 final ActivityRecord result = performClearTaskLocked(newR, launchFlags);
1364 mReuseTask = false;
1365 return result;
1366 }
1367
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001368 /**
1369 * Perform clear operation as requested by
1370 * {@link Intent#FLAG_ACTIVITY_CLEAR_TOP}: search from the top of the
1371 * stack to the given task, then look for
1372 * an instance of that activity in the stack and, if found, finish all
1373 * activities on top of it and return the instance.
1374 *
1375 * @param newR Description of the new activity being started.
1376 * @return Returns the old activity that should be continued to be used,
1377 * or null if none was found.
1378 */
1379 final ActivityRecord performClearTaskLocked(ActivityRecord newR, int launchFlags) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001380 int numActivities = mActivities.size();
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001381 for (int activityNdx = numActivities - 1; activityNdx >= 0; --activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001382 ActivityRecord r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001383 if (r.finishing) {
1384 continue;
1385 }
1386 if (r.realActivity.equals(newR.realActivity)) {
1387 // Here it is! Now finish everything in front...
Craig Mautner1602ec22013-05-12 10:24:27 -07001388 final ActivityRecord ret = r;
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001389
1390 for (++activityNdx; activityNdx < numActivities; ++activityNdx) {
Craig Mautner1602ec22013-05-12 10:24:27 -07001391 r = mActivities.get(activityNdx);
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001392 if (r.finishing) {
1393 continue;
1394 }
1395 ActivityOptions opts = r.takeOptionsLocked();
1396 if (opts != null) {
1397 ret.updateOptionsLocked(opts);
1398 }
Andrii Kulian02b7a832016-10-06 23:11:56 -07001399 if (mStack != null && mStack.finishActivityLocked(
Todd Kennedy539db512014-12-15 09:57:55 -08001400 r, Activity.RESULT_CANCELED, null, "clear-task-stack", false)) {
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001401 --activityNdx;
1402 --numActivities;
1403 }
1404 }
1405
1406 // Finally, if this is a normal launch mode (that is, not
1407 // expecting onNewIntent()), then we will finish the current
1408 // instance of the activity so a new fresh one can be started.
1409 if (ret.launchMode == ActivityInfo.LAUNCH_MULTIPLE
Daichi Hirono15a02992016-04-27 18:47:01 +09001410 && (launchFlags & Intent.FLAG_ACTIVITY_SINGLE_TOP) == 0
1411 && !ActivityStarter.isDocumentLaunchesIntoExisting(launchFlags)) {
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001412 if (!ret.finishing) {
Andrii Kulian02b7a832016-10-06 23:11:56 -07001413 if (mStack != null) {
1414 mStack.finishActivityLocked(
Wale Ogunwale7d701172015-03-11 15:36:30 -07001415 ret, Activity.RESULT_CANCELED, null, "clear-task-top", false);
1416 }
Craig Mautnerb0f7dc72013-04-01 16:34:45 -07001417 return null;
1418 }
1419 }
1420
1421 return ret;
1422 }
1423 }
1424
1425 return null;
1426 }
1427
Winson Chung0ec2a352017-10-26 11:38:30 -07001428 void removeTaskActivitiesLocked(boolean pauseImmediately, String reason) {
Craig Mautnerc0ffce52014-07-01 12:38:52 -07001429 // Just remove the entire task.
Winson Chung0ec2a352017-10-26 11:38:30 -07001430 performClearTaskAtIndexLocked(0, pauseImmediately, reason);
Craig Mautner9db9a0b2013-04-29 17:05:56 -07001431 }
1432
Craig Mautner432f64e2015-05-20 14:59:57 -07001433 String lockTaskAuthToString() {
1434 switch (mLockTaskAuth) {
1435 case LOCK_TASK_AUTH_DONT_LOCK: return "LOCK_TASK_AUTH_DONT_LOCK";
1436 case LOCK_TASK_AUTH_PINNABLE: return "LOCK_TASK_AUTH_PINNABLE";
1437 case LOCK_TASK_AUTH_LAUNCHABLE: return "LOCK_TASK_AUTH_LAUNCHABLE";
1438 case LOCK_TASK_AUTH_WHITELISTED: return "LOCK_TASK_AUTH_WHITELISTED";
Benjamin Franz469dd582015-06-09 14:24:36 +01001439 case LOCK_TASK_AUTH_LAUNCHABLE_PRIV: return "LOCK_TASK_AUTH_LAUNCHABLE_PRIV";
Craig Mautner432f64e2015-05-20 14:59:57 -07001440 default: return "unknown=" + mLockTaskAuth;
1441 }
1442 }
1443
Craig Mautner15df08a2015-04-01 12:17:18 -07001444 void setLockTaskAuth() {
Charles He2bf28322017-10-12 22:24:49 +01001445 setLockTaskAuth(getRootActivity());
1446 }
1447
1448 private void setLockTaskAuth(@Nullable ActivityRecord r) {
1449 if (r == null) {
1450 mLockTaskAuth = LOCK_TASK_AUTH_PINNABLE;
1451 return;
1452 }
1453
Charles He520b2832017-09-02 15:27:16 +01001454 final String pkg = (realActivity != null) ? realActivity.getPackageName() : null;
Charles He2bf28322017-10-12 22:24:49 +01001455 switch (r.lockTaskLaunchMode) {
Craig Mautner15df08a2015-04-01 12:17:18 -07001456 case LOCK_TASK_LAUNCH_MODE_DEFAULT:
Charles He520b2832017-09-02 15:27:16 +01001457 mLockTaskAuth = mService.mLockTaskController.isPackageWhitelisted(userId, pkg)
1458 ? LOCK_TASK_AUTH_WHITELISTED : LOCK_TASK_AUTH_PINNABLE;
Craig Mautner15df08a2015-04-01 12:17:18 -07001459 break;
1460
1461 case LOCK_TASK_LAUNCH_MODE_NEVER:
Benjamin Franz469dd582015-06-09 14:24:36 +01001462 mLockTaskAuth = LOCK_TASK_AUTH_DONT_LOCK;
Craig Mautner15df08a2015-04-01 12:17:18 -07001463 break;
1464
1465 case LOCK_TASK_LAUNCH_MODE_ALWAYS:
Benjamin Franz469dd582015-06-09 14:24:36 +01001466 mLockTaskAuth = LOCK_TASK_AUTH_LAUNCHABLE_PRIV;
Craig Mautner15df08a2015-04-01 12:17:18 -07001467 break;
1468
1469 case LOCK_TASK_LAUNCH_MODE_IF_WHITELISTED:
Charles He520b2832017-09-02 15:27:16 +01001470 mLockTaskAuth = mService.mLockTaskController.isPackageWhitelisted(userId, pkg)
1471 ? LOCK_TASK_AUTH_LAUNCHABLE : LOCK_TASK_AUTH_PINNABLE;
Craig Mautner15df08a2015-04-01 12:17:18 -07001472 break;
1473 }
Craig Mautner432f64e2015-05-20 14:59:57 -07001474 if (DEBUG_LOCKTASK) Slog.d(TAG_LOCKTASK, "setLockTaskAuth: task=" + this +
1475 " mLockTaskAuth=" + lockTaskAuthToString());
Craig Mautner15df08a2015-04-01 12:17:18 -07001476 }
1477
Winson Chungd3395382016-12-13 11:49:09 -08001478 private boolean isResizeable(boolean checkSupportsPip) {
1479 return (mService.mForceResizableActivities || ActivityInfo.isResizeableMode(mResizeMode)
Wale Ogunwale7e1f5f52017-10-18 15:19:59 -07001480 || (checkSupportsPip && mSupportsPictureInPicture));
Winson Chungd3395382016-12-13 11:49:09 -08001481 }
1482
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001483 boolean isResizeable() {
Winson Chungd3395382016-12-13 11:49:09 -08001484 return isResizeable(true /* checkSupportsPip */);
1485 }
1486
Wale Ogunwale04a05ac2017-09-17 21:35:02 -07001487 @Override
1488 public boolean supportsSplitScreenWindowingMode() {
Winson Chungd3395382016-12-13 11:49:09 -08001489 // A task can not be docked even if it is considered resizeable because it only supports
1490 // picture-in-picture mode but has a non-resizeable resizeMode
Wale Ogunwale04a05ac2017-09-17 21:35:02 -07001491 return super.supportsSplitScreenWindowingMode()
1492 && mService.mSupportsSplitScreenMultiWindow
Bryce Leec857a5b2017-08-16 10:04:52 -07001493 && (mService.mForceResizableActivities
1494 || (isResizeable(false /* checkSupportsPip */)
1495 && !ActivityInfo.isPreserveOrientationMode(mResizeMode)));
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001496 }
1497
skuhne@google.com322347b2016-12-02 12:54:03 -08001498 /**
Andrii Kulian036e3ad2017-04-19 10:55:10 -07001499 * Check whether this task can be launched on the specified display.
1500 * @param displayId Target display id.
1501 * @return {@code true} if either it is the default display or this activity is resizeable and
1502 * can be put a secondary screen.
1503 */
1504 boolean canBeLaunchedOnDisplay(int displayId) {
1505 return mService.mStackSupervisor.canPlaceEntityOnDisplay(displayId,
Andrii Kulian02689a72017-07-06 14:28:59 -07001506 isResizeable(false /* checkSupportsPip */), -1 /* don't check PID */,
1507 -1 /* don't check UID */, null /* activityInfo */);
Andrii Kulian036e3ad2017-04-19 10:55:10 -07001508 }
1509
1510 /**
skuhne@google.com322347b2016-12-02 12:54:03 -08001511 * Check that a given bounds matches the application requested orientation.
1512 *
1513 * @param bounds The bounds to be tested.
1514 * @return True if the requested bounds are okay for a resizing request.
1515 */
Wale Ogunwale069bbd32017-02-03 07:58:14 -08001516 private boolean canResizeToBounds(Rect bounds) {
Wale Ogunwale44f036f2017-09-29 05:09:09 -07001517 if (bounds == null || !inFreeformWindowingMode()) {
skuhne@google.com322347b2016-12-02 12:54:03 -08001518 // Note: If not on the freeform workspace, we ignore the bounds.
1519 return true;
1520 }
1521 final boolean landscape = bounds.width() > bounds.height();
Bryce Leef3c6a472017-11-14 14:53:06 -08001522 final Rect configBounds = getOverrideBounds();
skuhne@google.com322347b2016-12-02 12:54:03 -08001523 if (mResizeMode == RESIZE_MODE_FORCE_RESIZABLE_PRESERVE_ORIENTATION) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001524 return configBounds.isEmpty()
1525 || landscape == (configBounds.width() > configBounds.height());
skuhne@google.com322347b2016-12-02 12:54:03 -08001526 }
1527 return (mResizeMode != RESIZE_MODE_FORCE_RESIZABLE_PORTRAIT_ONLY || !landscape)
1528 && (mResizeMode != RESIZE_MODE_FORCE_RESIZABLE_LANDSCAPE_ONLY || landscape);
1529 }
1530
Craig Mautner525f3d92013-05-07 14:01:50 -07001531 /**
Yorke Leebdef5372017-04-10 16:38:51 -07001532 * @return {@code true} if the task is being cleared for the purposes of being reused.
1533 */
1534 boolean isClearingToReuseTask() {
1535 return mReuseTask;
1536 }
1537
1538 /**
Craig Mautner525f3d92013-05-07 14:01:50 -07001539 * Find the activity in the history stack within the given task. Returns
1540 * the index within the history at which it's found, or < 0 if not found.
1541 */
1542 final ActivityRecord findActivityInHistoryLocked(ActivityRecord r) {
1543 final ComponentName realActivity = r.realActivity;
1544 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
1545 ActivityRecord candidate = mActivities.get(activityNdx);
1546 if (candidate.finishing) {
1547 continue;
1548 }
1549 if (candidate.realActivity.equals(realActivity)) {
1550 return candidate;
1551 }
1552 }
1553 return null;
1554 }
1555
Winson Chunga449dc02014-05-16 11:15:04 -07001556 /** Updates the last task description values. */
1557 void updateTaskDescription() {
1558 // Traverse upwards looking for any break between main task activities and
1559 // utility activities.
1560 int activityNdx;
1561 final int numActivities = mActivities.size();
Wale Ogunwale3eadad72016-10-13 09:16:59 -07001562 final boolean relinquish = numActivities != 0 &&
1563 (mActivities.get(0).info.flags & FLAG_RELINQUISH_TASK_IDENTITY) != 0;
Winson Chunga449dc02014-05-16 11:15:04 -07001564 for (activityNdx = Math.min(numActivities, 1); activityNdx < numActivities;
Craig Mautner21d24a22014-04-23 11:45:37 -07001565 ++activityNdx) {
Winson Chunga449dc02014-05-16 11:15:04 -07001566 final ActivityRecord r = mActivities.get(activityNdx);
Wale Ogunwale3eadad72016-10-13 09:16:59 -07001567 if (relinquish && (r.info.flags & FLAG_RELINQUISH_TASK_IDENTITY) == 0) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001568 // This will be the top activity for determining taskDescription. Pre-inc to
1569 // overcome initial decrement below.
1570 ++activityNdx;
1571 break;
1572 }
Winson Chunga449dc02014-05-16 11:15:04 -07001573 if (r.intent != null &&
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001574 (r.intent.getFlags() & Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET) != 0) {
Winson Chunga449dc02014-05-16 11:15:04 -07001575 break;
1576 }
1577 }
1578 if (activityNdx > 0) {
1579 // Traverse downwards starting below break looking for set label, icon.
1580 // Note that if there are activities in the task but none of them set the
1581 // recent activity values, then we do not fall back to the last set
1582 // values in the TaskRecord.
1583 String label = null;
Craig Mautner648f69b2014-09-18 14:16:26 -07001584 String iconFilename = null;
Matthew Ng54bc9422017-10-02 17:16:28 -07001585 int iconResource = -1;
Winson Chunga449dc02014-05-16 11:15:04 -07001586 int colorPrimary = 0;
Winson Chung1af8eda2016-02-05 17:55:56 +00001587 int colorBackground = 0;
Jorim Jaggi30d64f32017-04-07 16:33:17 +02001588 int statusBarColor = 0;
1589 int navigationBarColor = 0;
1590 boolean topActivity = true;
Winson Chunga449dc02014-05-16 11:15:04 -07001591 for (--activityNdx; activityNdx >= 0; --activityNdx) {
1592 final ActivityRecord r = mActivities.get(activityNdx);
1593 if (r.taskDescription != null) {
1594 if (label == null) {
1595 label = r.taskDescription.getLabel();
1596 }
Matthew Ng54bc9422017-10-02 17:16:28 -07001597 if (iconResource == -1) {
1598 iconResource = r.taskDescription.getIconResource();
1599 }
Craig Mautner648f69b2014-09-18 14:16:26 -07001600 if (iconFilename == null) {
1601 iconFilename = r.taskDescription.getIconFilename();
Winson Chunga449dc02014-05-16 11:15:04 -07001602 }
1603 if (colorPrimary == 0) {
1604 colorPrimary = r.taskDescription.getPrimaryColor();
Winson Chunga449dc02014-05-16 11:15:04 -07001605 }
Jorim Jaggi30d64f32017-04-07 16:33:17 +02001606 if (topActivity) {
Winson Chung1af8eda2016-02-05 17:55:56 +00001607 colorBackground = r.taskDescription.getBackgroundColor();
Jorim Jaggi30d64f32017-04-07 16:33:17 +02001608 statusBarColor = r.taskDescription.getStatusBarColor();
1609 navigationBarColor = r.taskDescription.getNavigationBarColor();
Winson Chung1af8eda2016-02-05 17:55:56 +00001610 }
Winson Chunga449dc02014-05-16 11:15:04 -07001611 }
Jorim Jaggi30d64f32017-04-07 16:33:17 +02001612 topActivity = false;
Winson Chunga449dc02014-05-16 11:15:04 -07001613 }
Matthew Ng54bc9422017-10-02 17:16:28 -07001614 lastTaskDescription = new TaskDescription(label, null, iconResource, iconFilename,
1615 colorPrimary, colorBackground, statusBarColor, navigationBarColor);
Jorim Jaggi829b9cd2017-01-23 16:20:53 +01001616 if (mWindowContainerController != null) {
1617 mWindowContainerController.setTaskDescription(lastTaskDescription);
1618 }
Winson Chungec396d62014-08-06 17:08:00 -07001619 // Update the task affiliation color if we are the parent of the group
1620 if (taskId == mAffiliatedTaskId) {
1621 mAffiliatedTaskColor = lastTaskDescription.getPrimaryColor();
1622 }
Winson Chunga449dc02014-05-16 11:15:04 -07001623 }
1624 }
1625
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001626 int findEffectiveRootIndex() {
Craig Mautner4767f4b2014-09-18 15:38:33 -07001627 int effectiveNdx = 0;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001628 final int topActivityNdx = mActivities.size() - 1;
Dianne Hackborn39569af2014-09-23 10:56:58 -07001629 for (int activityNdx = 0; activityNdx <= topActivityNdx; ++activityNdx) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001630 final ActivityRecord r = mActivities.get(activityNdx);
1631 if (r.finishing) {
1632 continue;
1633 }
Craig Mautner4767f4b2014-09-18 15:38:33 -07001634 effectiveNdx = activityNdx;
Wale Ogunwale3eadad72016-10-13 09:16:59 -07001635 if ((r.info.flags & FLAG_RELINQUISH_TASK_IDENTITY) == 0) {
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001636 break;
1637 }
1638 }
Craig Mautner4767f4b2014-09-18 15:38:33 -07001639 return effectiveNdx;
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001640 }
1641
1642 void updateEffectiveIntent() {
1643 final int effectiveRootIndex = findEffectiveRootIndex();
1644 final ActivityRecord r = mActivities.get(effectiveRootIndex);
Winson Chungfee26772014-08-05 12:21:52 -07001645 setIntent(r);
Winson Chung8d9009e2017-11-16 15:43:05 -08001646
1647 // Update the task description when the activities change
1648 updateTaskDescription();
Craig Mautner9d4e9bc2014-06-18 18:34:56 -07001649 }
1650
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001651 private void adjustForMinimalTaskDimensions(Rect bounds) {
1652 if (bounds == null) {
1653 return;
1654 }
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001655 int minWidth = mMinWidth;
1656 int minHeight = mMinHeight;
Robert Carr9c5867d2016-03-10 15:52:46 -08001657 // If the task has no requested minimal size, we'd like to enforce a minimal size
1658 // so that the user can not render the task too small to manipulate. We don't need
1659 // to do this for the pinned stack as the bounds are controlled by the system.
Wale Ogunwale44f036f2017-09-29 05:09:09 -07001660 if (!inPinnedWindowingMode()) {
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001661 if (minWidth == INVALID_MIN_SIZE) {
1662 minWidth = mService.mStackSupervisor.mDefaultMinSizeOfResizeableTask;
Andrii Kulian2e751b82016-03-16 16:59:32 -07001663 }
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001664 if (minHeight == INVALID_MIN_SIZE) {
1665 minHeight = mService.mStackSupervisor.mDefaultMinSizeOfResizeableTask;
Andrii Kulian2e751b82016-03-16 16:59:32 -07001666 }
Robert Carr9c5867d2016-03-10 15:52:46 -08001667 }
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001668 final boolean adjustWidth = minWidth > bounds.width();
1669 final boolean adjustHeight = minHeight > bounds.height();
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001670 if (!(adjustWidth || adjustHeight)) {
1671 return;
1672 }
1673
Bryce Leef3c6a472017-11-14 14:53:06 -08001674 final Rect configBounds = getOverrideBounds();
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001675 if (adjustWidth) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001676 if (!configBounds.isEmpty() && bounds.right == configBounds.right) {
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001677 bounds.left = bounds.right - minWidth;
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001678 } else {
1679 // Either left bounds match, or neither match, or the previous bounds were
1680 // fullscreen and we default to keeping left.
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001681 bounds.right = bounds.left + minWidth;
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001682 }
1683 }
1684 if (adjustHeight) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001685 if (!configBounds.isEmpty() && bounds.bottom == configBounds.bottom) {
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001686 bounds.top = bounds.bottom - minHeight;
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001687 } else {
1688 // Either top bounds match, or neither match, or the previous bounds were
1689 // fullscreen and we default to keeping top.
Andrii Kulianf66a83d2016-05-17 12:17:44 -07001690 bounds.bottom = bounds.top + minHeight;
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001691 }
1692 }
1693 }
1694
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001695 /**
Winson Chung5af42fc2017-03-24 17:11:33 -07001696 * @return a new Configuration for this Task, given the provided {@param bounds} and
1697 * {@param insetBounds}.
1698 */
1699 Configuration computeNewOverrideConfigurationForBounds(Rect bounds, Rect insetBounds) {
1700 // Compute a new override configuration for the given bounds, if fullscreen bounds
1701 // (bounds == null), then leave the override config unset
1702 final Configuration newOverrideConfig = new Configuration();
1703 if (bounds != null) {
1704 newOverrideConfig.setTo(getOverrideConfiguration());
1705 mTmpRect.set(bounds);
1706 adjustForMinimalTaskDimensions(mTmpRect);
1707 computeOverrideConfiguration(newOverrideConfig, mTmpRect, insetBounds,
1708 mTmpRect.right != bounds.right, mTmpRect.bottom != bounds.bottom);
1709 }
1710
1711 return newOverrideConfig;
1712 }
1713
1714 /**
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001715 * Update task's override configuration based on the bounds.
Jorim Jaggi0a932142016-02-01 17:42:25 -08001716 * @param bounds The bounds of the task.
Andrii Kulian8072d112016-09-16 11:11:01 -07001717 * @return True if the override configuration was updated.
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001718 */
Andrii Kulian8072d112016-09-16 11:11:01 -07001719 boolean updateOverrideConfiguration(Rect bounds) {
Jorim Jaggi0a932142016-02-01 17:42:25 -08001720 return updateOverrideConfiguration(bounds, null /* insetBounds */);
1721 }
1722
1723 /**
1724 * Update task's override configuration based on the bounds.
1725 * @param bounds The bounds of the task.
1726 * @param insetBounds The bounds used to calculate the system insets, which is used here to
1727 * subtract the navigation bar/status bar size from the screen size reported
1728 * to the application. See {@link IActivityManager#resizeDockedStack}.
Andrii Kulian8072d112016-09-16 11:11:01 -07001729 * @return True if the override configuration was updated.
Jorim Jaggi0a932142016-02-01 17:42:25 -08001730 */
Andrii Kulian8072d112016-09-16 11:11:01 -07001731 boolean updateOverrideConfiguration(Rect bounds, @Nullable Rect insetBounds) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001732 if (equivalentOverrideBounds(bounds)) {
Andrii Kulian8072d112016-09-16 11:11:01 -07001733 return false;
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001734 }
Bryce Leef3c6a472017-11-14 14:53:06 -08001735 final Rect currentBounds = getOverrideBounds();
1736
Andrii Kulian1779e612016-10-12 21:58:25 -07001737 mTmpConfig.setTo(getOverrideConfiguration());
Andrii Kulian1779e612016-10-12 21:58:25 -07001738 final Configuration newConfig = getOverrideConfiguration();
Filip Gruszczynskiebcc8752015-08-25 16:51:05 -07001739
Bryce Leef3c6a472017-11-14 14:53:06 -08001740 final boolean matchParentBounds = bounds == null || bounds.isEmpty();
Wale Ogunwale3382ab12017-07-27 08:55:03 -07001741 final boolean persistBounds = getWindowConfiguration().persistTaskBounds();
Bryce Leef3c6a472017-11-14 14:53:06 -08001742 if (matchParentBounds) {
1743 if (!currentBounds.isEmpty() && persistBounds) {
1744 mLastNonFullscreenBounds = currentBounds;
Wale Ogunwale706ed792015-08-02 10:29:44 -07001745 }
Bryce Leef3c6a472017-11-14 14:53:06 -08001746 setBounds(null);
Andrii Kulian1779e612016-10-12 21:58:25 -07001747 newConfig.unset();
Wale Ogunwale706ed792015-08-02 10:29:44 -07001748 } else {
Wale Ogunwale9a08f822016-02-17 19:03:58 -08001749 mTmpRect.set(bounds);
1750 adjustForMinimalTaskDimensions(mTmpRect);
Bryce Leef3c6a472017-11-14 14:53:06 -08001751 setBounds(mTmpRect);
1752
Wale Ogunwale3382ab12017-07-27 08:55:03 -07001753 if (mStack == null || persistBounds) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001754 mLastNonFullscreenBounds = getOverrideBounds();
Wale Ogunwaleb34a7ad2015-08-14 11:05:30 -07001755 }
Wale Ogunwale55ddf8f2017-03-20 08:56:38 -07001756 computeOverrideConfiguration(newConfig, mTmpRect, insetBounds,
Andrii Kuliandaea3572016-04-08 13:20:51 -07001757 mTmpRect.right != bounds.right, mTmpRect.bottom != bounds.bottom);
Wale Ogunwale706ed792015-08-02 10:29:44 -07001758 }
Andrii Kulian1779e612016-10-12 21:58:25 -07001759 onOverrideConfigurationChanged(newConfig);
Wale Ogunwaleeb76b762017-11-17 10:08:04 -08001760 return !mTmpConfig.equals(newConfig);
1761 }
Wale Ogunwale5f986092015-12-04 15:35:38 -08001762
Wale Ogunwaleeb76b762017-11-17 10:08:04 -08001763 @Override
1764 public void onConfigurationChanged(Configuration newParentConfig) {
1765 final boolean wasInMultiWindowMode = inMultiWindowMode();
1766 super.onConfigurationChanged(newParentConfig);
1767 if (wasInMultiWindowMode != inMultiWindowMode()) {
Winson Chung5af42fc2017-03-24 17:11:33 -07001768 mService.mStackSupervisor.scheduleUpdateMultiWindowMode(this);
Wale Ogunwale5f986092015-12-04 15:35:38 -08001769 }
Wale Ogunwaleeb76b762017-11-17 10:08:04 -08001770 // TODO: Should also take care of Pip mode changes here.
Wale Ogunwalee4a0c572015-06-30 08:40:31 -07001771 }
1772
Andrii Kulian1779e612016-10-12 21:58:25 -07001773 /** Clears passed config and fills it with new override values. */
Wale Ogunwale55ddf8f2017-03-20 08:56:38 -07001774 // TODO(b/36505427): TaskRecord.computeOverrideConfiguration() is a utility method that doesn't
1775 // depend on task or stacks, but uses those object to get the display to base the calculation
1776 // on. Probably best to centralize calculations like this in ConfigurationContainer.
1777 void computeOverrideConfiguration(Configuration config, Rect bounds, Rect insetBounds,
Andrii Kuliana8a9bc52016-10-14 11:00:13 -07001778 boolean overrideWidth, boolean overrideHeight) {
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001779 mTmpNonDecorBounds.set(bounds);
1780 mTmpStableBounds.set(bounds);
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001781
Andrii Kulian1779e612016-10-12 21:58:25 -07001782 config.unset();
Winson Chungbdc646f2017-02-13 12:12:22 -08001783 final Configuration parentConfig = getParent().getConfiguration();
Bryce Lee7566d762017-03-30 09:34:15 -07001784
Andrii Kulian1779e612016-10-12 21:58:25 -07001785 final float density = parentConfig.densityDpi * DisplayMetrics.DENSITY_DEFAULT_SCALE;
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001786
Winson Chungbdc646f2017-02-13 12:12:22 -08001787 if (mStack != null) {
1788 final StackWindowController stackController = mStack.getWindowContainerController();
1789 stackController.adjustConfigurationForBounds(bounds, insetBounds,
1790 mTmpNonDecorBounds, mTmpStableBounds, overrideWidth, overrideHeight, density,
1791 config, parentConfig);
1792 } else {
Bryce Lee7566d762017-03-30 09:34:15 -07001793 throw new IllegalArgumentException("Expected stack when calculating override config");
Winson Chungbdc646f2017-02-13 12:12:22 -08001794 }
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001795
Winson Chung60c1aba2017-03-14 17:47:42 -07001796 config.orientation = (config.screenWidthDp <= config.screenHeightDp)
1797 ? Configuration.ORIENTATION_PORTRAIT
1798 : Configuration.ORIENTATION_LANDSCAPE;
1799
Jorim Jaggi85639432016-05-06 17:27:55 -07001800 // For calculating screen layout, we need to use the non-decor inset screen area for the
1801 // calculation for compatibility reasons, i.e. screen area without system bars that could
1802 // never go away in Honeycomb.
Winson Chungbdc646f2017-02-13 12:12:22 -08001803 final int compatScreenWidthDp = (int) (mTmpNonDecorBounds.width() / density);
1804 final int compatScreenHeightDp = (int) (mTmpNonDecorBounds.height() / density);
Andrii Kulian1779e612016-10-12 21:58:25 -07001805 // We're only overriding LONG, SIZE and COMPAT parts of screenLayout, so we start override
1806 // calculation with partial default.
1807 final int sl = Configuration.SCREENLAYOUT_LONG_YES | Configuration.SCREENLAYOUT_SIZE_XLARGE;
Jorim Jaggi82c9dc92016-02-05 15:10:33 -08001808 final int longSize = Math.max(compatScreenHeightDp, compatScreenWidthDp);
Andrii Kulian1779e612016-10-12 21:58:25 -07001809 final int shortSize = Math.min(compatScreenHeightDp, compatScreenWidthDp);
Jorim Jaggi85639432016-05-06 17:27:55 -07001810 config.screenLayout = Configuration.reduceScreenLayout(sl, longSize, shortSize);
Jorim Jaggia95ca8d2016-01-15 22:54:46 -08001811 }
1812
Filip Gruszczynskidce2d162016-01-12 15:40:13 -08001813 Rect updateOverrideConfigurationFromLaunchBounds() {
Bryce Leef3c6a472017-11-14 14:53:06 -08001814 final Rect bounds = getLaunchBounds();
Filip Gruszczynskidce2d162016-01-12 15:40:13 -08001815 updateOverrideConfiguration(bounds);
Bryce Leef3c6a472017-11-14 14:53:06 -08001816 if (bounds != null && !bounds.isEmpty()) {
1817 // TODO: Review if we actually want to do this - we are setting the launch bounds
1818 // directly here.
1819 bounds.set(getOverrideBounds());
Andrii Kulian73336d812016-03-24 12:56:08 -07001820 }
Filip Gruszczynskidce2d162016-01-12 15:40:13 -08001821 return bounds;
1822 }
1823
Wale Ogunwale935e5022015-11-10 12:36:10 -08001824 /** Updates the task's bounds and override configuration to match what is expected for the
1825 * input stack. */
1826 void updateOverrideConfigurationForStack(ActivityStack inStack) {
Andrii Kulian02b7a832016-10-06 23:11:56 -07001827 if (mStack != null && mStack == inStack) {
Wale Ogunwale935e5022015-11-10 12:36:10 -08001828 return;
1829 }
1830
Wale Ogunwale44f036f2017-09-29 05:09:09 -07001831 if (inStack.inFreeformWindowingMode()) {
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001832 if (!isResizeable()) {
Wale Ogunwale935e5022015-11-10 12:36:10 -08001833 throw new IllegalArgumentException("Can not position non-resizeable task="
1834 + this + " in stack=" + inStack);
1835 }
Bryce Leef3c6a472017-11-14 14:53:06 -08001836 if (!matchParentBounds()) {
Wale Ogunwale935e5022015-11-10 12:36:10 -08001837 return;
1838 }
1839 if (mLastNonFullscreenBounds != null) {
1840 updateOverrideConfiguration(mLastNonFullscreenBounds);
1841 } else {
Bryce Leedacefc42017-10-10 12:56:02 -07001842 mService.mStackSupervisor.getLaunchingBoundsController().layoutTask(this, null);
Wale Ogunwale935e5022015-11-10 12:36:10 -08001843 }
1844 } else {
Bryce Leef3c6a472017-11-14 14:53:06 -08001845 updateOverrideConfiguration(inStack.getOverrideBounds());
Wale Ogunwale935e5022015-11-10 12:36:10 -08001846 }
1847 }
1848
Wale Ogunwale706ed792015-08-02 10:29:44 -07001849 /** Returns the bounds that should be used to launch this task. */
Wale Ogunwale30e441d2017-11-09 08:28:45 -08001850 Rect getLaunchBounds() {
Andrii Kulian02b7a832016-10-06 23:11:56 -07001851 if (mStack == null) {
Chong Zhang7d5f5102016-01-13 10:29:24 -08001852 return null;
1853 }
1854
Wale Ogunwale04a05ac2017-09-17 21:35:02 -07001855 final int windowingMode = getWindowingMode();
1856 if (!isActivityTypeStandardOrUndefined()
1857 || windowingMode == WINDOWING_MODE_FULLSCREEN
1858 || (windowingMode == WINDOWING_MODE_SPLIT_SCREEN_PRIMARY && !isResizeable())) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001859 return isResizeable() ? mStack.getOverrideBounds() : null;
Wale Ogunwale3382ab12017-07-27 08:55:03 -07001860 } else if (!getWindowConfiguration().persistTaskBounds()) {
Bryce Leef3c6a472017-11-14 14:53:06 -08001861 return mStack.getOverrideBounds();
Wale Ogunwale706ed792015-08-02 10:29:44 -07001862 }
1863 return mLastNonFullscreenBounds;
1864 }
1865
Jorim Jaggi8b702ed2017-01-20 16:59:03 +01001866 void addStartingWindowsForVisibleActivities(boolean taskSwitch) {
1867 for (int activityNdx = mActivities.size() - 1; activityNdx >= 0; --activityNdx) {
1868 final ActivityRecord r = mActivities.get(activityNdx);
1869 if (r.visible) {
1870 r.showStartingWindow(null /* prev */, false /* newTask */, taskSwitch);
1871 }
1872 }
1873 }
1874
Dianne Hackborn68a06332017-11-15 17:54:18 -08001875 void setRootProcess(ProcessRecord proc) {
1876 clearRootProcess();
1877 if (intent != null &&
1878 (intent.getFlags() & Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS) == 0) {
1879 mRootProcess = proc;
1880 proc.recentTasks.add(this);
1881 }
1882 }
1883
1884 void clearRootProcess() {
1885 if (mRootProcess != null) {
1886 mRootProcess.recentTasks.remove(this);
1887 mRootProcess = null;
1888 }
1889 }
1890
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001891 void dump(PrintWriter pw, String prefix) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001892 pw.print(prefix); pw.print("userId="); pw.print(userId);
Dianne Hackborn885fbe52014-08-23 15:23:58 -07001893 pw.print(" effectiveUid="); UserHandle.formatUid(pw, effectiveUid);
1894 pw.print(" mCallingUid="); UserHandle.formatUid(pw, mCallingUid);
Suprabh Shukla7745c142016-03-07 18:21:10 -08001895 pw.print(" mUserSetupComplete="); pw.print(mUserSetupComplete);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001896 pw.print(" mCallingPackage="); pw.println(mCallingPackage);
Dianne Hackborn79228822014-09-16 11:11:23 -07001897 if (affinity != null || rootAffinity != null) {
1898 pw.print(prefix); pw.print("affinity="); pw.print(affinity);
1899 if (affinity == null || !affinity.equals(rootAffinity)) {
1900 pw.print(" root="); pw.println(rootAffinity);
1901 } else {
1902 pw.println();
1903 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001904 }
Dianne Hackborn91097de2014-04-04 18:02:06 -07001905 if (voiceSession != null || voiceInteractor != null) {
1906 pw.print(prefix); pw.print("VOICE: session=0x");
1907 pw.print(Integer.toHexString(System.identityHashCode(voiceSession)));
1908 pw.print(" interactor=0x");
1909 pw.println(Integer.toHexString(System.identityHashCode(voiceInteractor)));
1910 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001911 if (intent != null) {
1912 StringBuilder sb = new StringBuilder(128);
1913 sb.append(prefix); sb.append("intent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -08001914 intent.toShortString(sb, false, true, false, true);
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001915 sb.append('}');
1916 pw.println(sb.toString());
1917 }
1918 if (affinityIntent != null) {
1919 StringBuilder sb = new StringBuilder(128);
1920 sb.append(prefix); sb.append("affinityIntent={");
Dianne Hackborn21c241e2012-03-08 13:57:23 -08001921 affinityIntent.toShortString(sb, false, true, false, true);
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001922 sb.append('}');
1923 pw.println(sb.toString());
1924 }
1925 if (origActivity != null) {
1926 pw.print(prefix); pw.print("origActivity=");
1927 pw.println(origActivity.flattenToShortString());
1928 }
1929 if (realActivity != null) {
1930 pw.print(prefix); pw.print("realActivity=");
1931 pw.println(realActivity.flattenToShortString());
1932 }
Wale Ogunwale66e16852017-10-19 13:35:52 -07001933 if (autoRemoveRecents || isPersistable || !isActivityTypeStandard() || numFullscreen != 0) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001934 pw.print(prefix); pw.print("autoRemoveRecents="); pw.print(autoRemoveRecents);
Dianne Hackborn852975d2014-08-22 17:42:43 -07001935 pw.print(" isPersistable="); pw.print(isPersistable);
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001936 pw.print(" numFullscreen="); pw.print(numFullscreen);
Wale Ogunwale66e16852017-10-19 13:35:52 -07001937 pw.print(" activityType="); pw.println(getActivityType());
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001938 }
Craig Mautner432f64e2015-05-20 14:59:57 -07001939 if (rootWasReset || mNeverRelinquishIdentity || mReuseTask
1940 || mLockTaskAuth != LOCK_TASK_AUTH_PINNABLE) {
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001941 pw.print(prefix); pw.print("rootWasReset="); pw.print(rootWasReset);
1942 pw.print(" mNeverRelinquishIdentity="); pw.print(mNeverRelinquishIdentity);
Craig Mautner432f64e2015-05-20 14:59:57 -07001943 pw.print(" mReuseTask="); pw.print(mReuseTask);
1944 pw.print(" mLockTaskAuth="); pw.println(lockTaskAuthToString());
Dianne Hackbornaec68bb2014-08-20 15:25:13 -07001945 }
Wale Ogunwale18795a22014-12-03 11:38:33 -08001946 if (mAffiliatedTaskId != taskId || mPrevAffiliateTaskId != INVALID_TASK_ID
1947 || mPrevAffiliate != null || mNextAffiliateTaskId != INVALID_TASK_ID
1948 || mNextAffiliate != null) {
Dianne Hackborn852975d2014-08-22 17:42:43 -07001949 pw.print(prefix); pw.print("affiliation="); pw.print(mAffiliatedTaskId);
1950 pw.print(" prevAffiliation="); pw.print(mPrevAffiliateTaskId);
1951 pw.print(" (");
1952 if (mPrevAffiliate == null) {
1953 pw.print("null");
1954 } else {
1955 pw.print(Integer.toHexString(System.identityHashCode(mPrevAffiliate)));
1956 }
1957 pw.print(") nextAffiliation="); pw.print(mNextAffiliateTaskId);
1958 pw.print(" (");
1959 if (mNextAffiliate == null) {
1960 pw.print("null");
1961 } else {
1962 pw.print(Integer.toHexString(System.identityHashCode(mNextAffiliate)));
1963 }
1964 pw.println(")");
1965 }
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001966 pw.print(prefix); pw.print("Activities="); pw.println(mActivities);
Dianne Hackborn852975d2014-08-22 17:42:43 -07001967 if (!askedCompatMode || !inRecents || !isAvailable) {
1968 pw.print(prefix); pw.print("askedCompatMode="); pw.print(askedCompatMode);
1969 pw.print(" inRecents="); pw.print(inRecents);
1970 pw.print(" isAvailable="); pw.println(isAvailable);
Dianne Hackborn36cd41f2011-05-25 21:00:46 -07001971 }
Dianne Hackborn852975d2014-08-22 17:42:43 -07001972 if (lastDescription != null) {
1973 pw.print(prefix); pw.print("lastDescription="); pw.println(lastDescription);
1974 }
Dianne Hackborn68a06332017-11-15 17:54:18 -08001975 if (mRootProcess != null) {
1976 pw.print(prefix); pw.print("mRootProcess="); pw.println(mRootProcess);
1977 }
Andrii Kulian02b7a832016-10-06 23:11:56 -07001978 pw.print(prefix); pw.print("stackId="); pw.println(getStackId());
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001979 pw.print(prefix + "hasBeenVisible=" + hasBeenVisible);
1980 pw.print(" mResizeMode=" + ActivityInfo.resizeModeToString(mResizeMode));
Winson Chungd3395382016-12-13 11:49:09 -08001981 pw.print(" mSupportsPictureInPicture=" + mSupportsPictureInPicture);
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001982 pw.print(" isResizeable=" + isResizeable());
Wale Ogunwaleb1faf602016-01-27 09:12:31 -08001983 pw.print(" lastActiveTime=" + lastActiveTime);
1984 pw.println(" (inactive for " + (getInactiveDuration() / 1000) + "s)");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001985 }
1986
Craig Mautner5d9c7be2013-02-15 14:02:56 -08001987 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001988 public String toString() {
Dianne Hackborn1d442e02009-04-20 18:14:05 -07001989 StringBuilder sb = new StringBuilder(128);
Craig Mautnerde4ef022013-04-07 19:01:33 -07001990 if (stringName != null) {
1991 sb.append(stringName);
1992 sb.append(" U=");
1993 sb.append(userId);
Wale Ogunwale6c5eb1c2015-11-10 07:52:22 -08001994 sb.append(" StackId=");
Andrii Kulian02b7a832016-10-06 23:11:56 -07001995 sb.append(getStackId());
Craig Mautnerde4ef022013-04-07 19:01:33 -07001996 sb.append(" sz=");
1997 sb.append(mActivities.size());
1998 sb.append('}');
1999 return sb.toString();
2000 }
Dianne Hackborn1d442e02009-04-20 18:14:05 -07002001 sb.append("TaskRecord{");
2002 sb.append(Integer.toHexString(System.identityHashCode(this)));
2003 sb.append(" #");
2004 sb.append(taskId);
2005 if (affinity != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08002006 sb.append(" A=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07002007 sb.append(affinity);
2008 } else if (intent != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08002009 sb.append(" I=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07002010 sb.append(intent.getComponent().flattenToShortString());
2011 } else if (affinityIntent != null) {
Craig Mautner5d9c7be2013-02-15 14:02:56 -08002012 sb.append(" aI=");
Dianne Hackborn1d442e02009-04-20 18:14:05 -07002013 sb.append(affinityIntent.getComponent().flattenToShortString());
2014 } else {
2015 sb.append(" ??");
2016 }
Craig Mautnerde4ef022013-04-07 19:01:33 -07002017 stringName = sb.toString();
2018 return toString();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002019 }
Steven Timotius4346f0a2017-09-12 11:07:21 -07002020
2021 public void writeToProto(ProtoOutputStream proto, long fieldId) {
2022 final long token = proto.start(fieldId);
Adrian Roos4921ccf2017-09-28 16:54:06 +02002023 super.writeToProto(proto, CONFIGURATION_CONTAINER, false /* trim */);
Steven Timotius4346f0a2017-09-12 11:07:21 -07002024 proto.write(ID, taskId);
2025 for (int i = mActivities.size() - 1; i >= 0; i--) {
2026 ActivityRecord activity = mActivities.get(i);
2027 activity.writeToProto(proto, ACTIVITIES);
2028 }
2029 proto.write(STACK_ID, mStack.mStackId);
2030 if (mLastNonFullscreenBounds != null) {
2031 mLastNonFullscreenBounds.writeToProto(proto, LAST_NON_FULLSCREEN_BOUNDS);
2032 }
2033 if (realActivity != null) {
2034 proto.write(REAL_ACTIVITY, realActivity.flattenToShortString());
2035 }
2036 if (origActivity != null) {
2037 proto.write(ORIG_ACTIVITY, origActivity.flattenToShortString());
2038 }
2039 proto.write(ACTIVITY_TYPE, getActivityType());
Steven Timotius4346f0a2017-09-12 11:07:21 -07002040 proto.write(RESIZE_MODE, mResizeMode);
Bryce Leef3c6a472017-11-14 14:53:06 -08002041 // TODO: Remove, no longer needed with windowingMode.
2042 proto.write(FULLSCREEN, matchParentBounds());
2043
2044 if (!matchParentBounds()) {
2045 final Rect bounds = getOverrideBounds();
2046 bounds.writeToProto(proto, BOUNDS);
Steven Timotius4346f0a2017-09-12 11:07:21 -07002047 }
2048 proto.write(MIN_WIDTH, mMinWidth);
2049 proto.write(MIN_HEIGHT, mMinHeight);
2050 proto.end(token);
2051 }
Winson Chung61c9e5a2017-10-11 10:39:32 -07002052
2053 /**
2054 * See {@link #getNumRunningActivities(TaskActivitiesReport)}.
2055 */
2056 static class TaskActivitiesReport {
2057 int numRunning;
2058 int numActivities;
2059 ActivityRecord top;
2060 ActivityRecord base;
2061
2062 void reset() {
2063 numRunning = numActivities = 0;
2064 top = base = null;
2065 }
2066 }
Garfield Tan9b1efea2017-12-05 16:43:46 -08002067
2068 /**
2069 * Saves this {@link TaskRecord} to XML using given serializer.
2070 */
2071 void saveToXml(XmlSerializer out) throws IOException, XmlPullParserException {
2072 if (DEBUG_RECENTS) Slog.i(TAG_RECENTS, "Saving task=" + this);
2073
2074 out.attribute(null, ATTR_TASKID, String.valueOf(taskId));
2075 if (realActivity != null) {
2076 out.attribute(null, ATTR_REALACTIVITY, realActivity.flattenToShortString());
2077 }
2078 out.attribute(null, ATTR_REALACTIVITY_SUSPENDED, String.valueOf(realActivitySuspended));
2079 if (origActivity != null) {
2080 out.attribute(null, ATTR_ORIGACTIVITY, origActivity.flattenToShortString());
2081 }
2082 // Write affinity, and root affinity if it is different from affinity.
2083 // We use the special string "@" for a null root affinity, so we can identify
2084 // later whether we were given a root affinity or should just make it the
2085 // same as the affinity.
2086 if (affinity != null) {
2087 out.attribute(null, ATTR_AFFINITY, affinity);
2088 if (!affinity.equals(rootAffinity)) {
2089 out.attribute(null, ATTR_ROOT_AFFINITY, rootAffinity != null ? rootAffinity : "@");
2090 }
2091 } else if (rootAffinity != null) {
2092 out.attribute(null, ATTR_ROOT_AFFINITY, rootAffinity != null ? rootAffinity : "@");
2093 }
2094 out.attribute(null, ATTR_ROOTHASRESET, String.valueOf(rootWasReset));
2095 out.attribute(null, ATTR_AUTOREMOVERECENTS, String.valueOf(autoRemoveRecents));
2096 out.attribute(null, ATTR_ASKEDCOMPATMODE, String.valueOf(askedCompatMode));
2097 out.attribute(null, ATTR_USERID, String.valueOf(userId));
2098 out.attribute(null, ATTR_USER_SETUP_COMPLETE, String.valueOf(mUserSetupComplete));
2099 out.attribute(null, ATTR_EFFECTIVE_UID, String.valueOf(effectiveUid));
2100 out.attribute(null, ATTR_LASTTIMEMOVED, String.valueOf(mLastTimeMoved));
2101 out.attribute(null, ATTR_NEVERRELINQUISH, String.valueOf(mNeverRelinquishIdentity));
2102 if (lastDescription != null) {
2103 out.attribute(null, ATTR_LASTDESCRIPTION, lastDescription.toString());
2104 }
2105 if (lastTaskDescription != null) {
2106 lastTaskDescription.saveToXml(out);
2107 }
2108 out.attribute(null, ATTR_TASK_AFFILIATION_COLOR, String.valueOf(mAffiliatedTaskColor));
2109 out.attribute(null, ATTR_TASK_AFFILIATION, String.valueOf(mAffiliatedTaskId));
2110 out.attribute(null, ATTR_PREV_AFFILIATION, String.valueOf(mPrevAffiliateTaskId));
2111 out.attribute(null, ATTR_NEXT_AFFILIATION, String.valueOf(mNextAffiliateTaskId));
2112 out.attribute(null, ATTR_CALLING_UID, String.valueOf(mCallingUid));
2113 out.attribute(null, ATTR_CALLING_PACKAGE, mCallingPackage == null ? "" : mCallingPackage);
2114 out.attribute(null, ATTR_RESIZE_MODE, String.valueOf(mResizeMode));
2115 out.attribute(null, ATTR_SUPPORTS_PICTURE_IN_PICTURE,
2116 String.valueOf(mSupportsPictureInPicture));
2117 if (mLastNonFullscreenBounds != null) {
2118 out.attribute(
2119 null, ATTR_NON_FULLSCREEN_BOUNDS, mLastNonFullscreenBounds.flattenToString());
2120 }
2121 out.attribute(null, ATTR_MIN_WIDTH, String.valueOf(mMinWidth));
2122 out.attribute(null, ATTR_MIN_HEIGHT, String.valueOf(mMinHeight));
2123 out.attribute(null, ATTR_PERSIST_TASK_VERSION, String.valueOf(PERSIST_TASK_VERSION));
2124
2125 if (affinityIntent != null) {
2126 out.startTag(null, TAG_AFFINITYINTENT);
2127 affinityIntent.saveToXml(out);
2128 out.endTag(null, TAG_AFFINITYINTENT);
2129 }
2130
2131 out.startTag(null, TAG_INTENT);
2132 intent.saveToXml(out);
2133 out.endTag(null, TAG_INTENT);
2134
2135 final ArrayList<ActivityRecord> activities = mActivities;
2136 final int numActivities = activities.size();
2137 for (int activityNdx = 0; activityNdx < numActivities; ++activityNdx) {
2138 final ActivityRecord r = activities.get(activityNdx);
2139 if (r.info.persistableMode == ActivityInfo.PERSIST_ROOT_ONLY || !r.isPersistable() ||
2140 ((r.intent.getFlags() & FLAG_ACTIVITY_NEW_DOCUMENT
2141 | FLAG_ACTIVITY_RETAIN_IN_RECENTS) == FLAG_ACTIVITY_NEW_DOCUMENT) &&
2142 activityNdx > 0) {
2143 // Stop at first non-persistable or first break in task (CLEAR_WHEN_TASK_RESET).
2144 break;
2145 }
2146 out.startTag(null, TAG_ACTIVITY);
2147 r.saveToXml(out);
2148 out.endTag(null, TAG_ACTIVITY);
2149 }
2150 }
2151
2152 @VisibleForTesting
2153 static TaskRecordFactory getTaskRecordFactory() {
2154 if (sTaskRecordFactory == null) {
2155 setTaskRecordFactory(new TaskRecordFactory());
2156 }
2157 return sTaskRecordFactory;
2158 }
2159
2160 static void setTaskRecordFactory(TaskRecordFactory factory) {
2161 sTaskRecordFactory = factory;
2162 }
2163
2164 static TaskRecord create(ActivityManagerService service, int taskId, ActivityInfo info,
2165 Intent intent, IVoiceInteractionSession voiceSession,
2166 IVoiceInteractor voiceInteractor) {
2167 return getTaskRecordFactory().create(
2168 service, taskId, info, intent, voiceSession, voiceInteractor);
2169 }
2170
2171 static TaskRecord create(ActivityManagerService service, int taskId, ActivityInfo info,
2172 Intent intent, TaskDescription taskDescription) {
2173 return getTaskRecordFactory().create(service, taskId, info, intent, taskDescription);
2174 }
2175
2176 static TaskRecord restoreFromXml(XmlPullParser in, ActivityStackSupervisor stackSupervisor)
2177 throws IOException, XmlPullParserException {
2178 return getTaskRecordFactory().restoreFromXml(in, stackSupervisor);
2179 }
2180
2181 /**
2182 * A factory class used to create {@link TaskRecord} or its subclass if any. This can be
2183 * specified when system boots by setting it with
2184 * {@link #setTaskRecordFactory(TaskRecordFactory)}.
2185 */
2186 static class TaskRecordFactory {
2187
2188 TaskRecord create(ActivityManagerService service, int taskId, ActivityInfo info,
2189 Intent intent, IVoiceInteractionSession voiceSession,
2190 IVoiceInteractor voiceInteractor) {
2191 return new TaskRecord(
2192 service, taskId, info, intent, voiceSession, voiceInteractor);
2193 }
2194
2195 TaskRecord create(ActivityManagerService service, int taskId, ActivityInfo info,
2196 Intent intent, TaskDescription taskDescription) {
2197 return new TaskRecord(service, taskId, info, intent, taskDescription);
2198 }
2199
2200 /**
2201 * Should only be used when we're restoring {@link TaskRecord} from storage.
2202 */
2203 TaskRecord create(ActivityManagerService service, int taskId, Intent intent,
2204 Intent affinityIntent, String affinity, String rootAffinity,
2205 ComponentName realActivity, ComponentName origActivity, boolean rootWasReset,
2206 boolean autoRemoveRecents, boolean askedCompatMode, int userId,
2207 int effectiveUid, String lastDescription, ArrayList<ActivityRecord> activities,
2208 long lastTimeMoved, boolean neverRelinquishIdentity,
2209 TaskDescription lastTaskDescription, int taskAffiliation, int prevTaskId,
2210 int nextTaskId, int taskAffiliationColor, int callingUid, String callingPackage,
2211 int resizeMode, boolean supportsPictureInPicture, boolean realActivitySuspended,
2212 boolean userSetupComplete, int minWidth, int minHeight) {
2213 return new TaskRecord(service, taskId, intent, affinityIntent, affinity,
2214 rootAffinity, realActivity, origActivity, rootWasReset, autoRemoveRecents,
2215 askedCompatMode, userId, effectiveUid, lastDescription, activities,
2216 lastTimeMoved, neverRelinquishIdentity, lastTaskDescription, taskAffiliation,
2217 prevTaskId, nextTaskId, taskAffiliationColor, callingUid, callingPackage,
2218 resizeMode, supportsPictureInPicture, realActivitySuspended, userSetupComplete,
2219 minWidth, minHeight);
2220 }
2221
2222 TaskRecord restoreFromXml(XmlPullParser in, ActivityStackSupervisor stackSupervisor)
2223 throws IOException, XmlPullParserException {
2224 Intent intent = null;
2225 Intent affinityIntent = null;
2226 ArrayList<ActivityRecord> activities = new ArrayList<>();
2227 ComponentName realActivity = null;
2228 boolean realActivitySuspended = false;
2229 ComponentName origActivity = null;
2230 String affinity = null;
2231 String rootAffinity = null;
2232 boolean hasRootAffinity = false;
2233 boolean rootHasReset = false;
2234 boolean autoRemoveRecents = false;
2235 boolean askedCompatMode = false;
2236 int taskType = 0;
2237 int userId = 0;
2238 boolean userSetupComplete = true;
2239 int effectiveUid = -1;
2240 String lastDescription = null;
2241 long lastTimeOnTop = 0;
2242 boolean neverRelinquishIdentity = true;
2243 int taskId = INVALID_TASK_ID;
2244 final int outerDepth = in.getDepth();
2245 TaskDescription taskDescription = new TaskDescription();
2246 int taskAffiliation = INVALID_TASK_ID;
2247 int taskAffiliationColor = 0;
2248 int prevTaskId = INVALID_TASK_ID;
2249 int nextTaskId = INVALID_TASK_ID;
2250 int callingUid = -1;
2251 String callingPackage = "";
2252 int resizeMode = RESIZE_MODE_FORCE_RESIZEABLE;
2253 boolean supportsPictureInPicture = false;
2254 Rect bounds = null;
2255 int minWidth = INVALID_MIN_SIZE;
2256 int minHeight = INVALID_MIN_SIZE;
2257 int persistTaskVersion = 0;
2258
2259 for (int attrNdx = in.getAttributeCount() - 1; attrNdx >= 0; --attrNdx) {
2260 final String attrName = in.getAttributeName(attrNdx);
2261 final String attrValue = in.getAttributeValue(attrNdx);
2262 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG, "TaskRecord: attribute name=" +
2263 attrName + " value=" + attrValue);
2264 switch (attrName) {
2265 case ATTR_TASKID:
2266 if (taskId == INVALID_TASK_ID) taskId = Integer.parseInt(attrValue);
2267 break;
2268 case ATTR_REALACTIVITY:
2269 realActivity = ComponentName.unflattenFromString(attrValue);
2270 break;
2271 case ATTR_REALACTIVITY_SUSPENDED:
2272 realActivitySuspended = Boolean.valueOf(attrValue);
2273 break;
2274 case ATTR_ORIGACTIVITY:
2275 origActivity = ComponentName.unflattenFromString(attrValue);
2276 break;
2277 case ATTR_AFFINITY:
2278 affinity = attrValue;
2279 break;
2280 case ATTR_ROOT_AFFINITY:
2281 rootAffinity = attrValue;
2282 hasRootAffinity = true;
2283 break;
2284 case ATTR_ROOTHASRESET:
2285 rootHasReset = Boolean.parseBoolean(attrValue);
2286 break;
2287 case ATTR_AUTOREMOVERECENTS:
2288 autoRemoveRecents = Boolean.parseBoolean(attrValue);
2289 break;
2290 case ATTR_ASKEDCOMPATMODE:
2291 askedCompatMode = Boolean.parseBoolean(attrValue);
2292 break;
2293 case ATTR_USERID:
2294 userId = Integer.parseInt(attrValue);
2295 break;
2296 case ATTR_USER_SETUP_COMPLETE:
2297 userSetupComplete = Boolean.parseBoolean(attrValue);
2298 break;
2299 case ATTR_EFFECTIVE_UID:
2300 effectiveUid = Integer.parseInt(attrValue);
2301 break;
2302 case ATTR_TASKTYPE:
2303 taskType = Integer.parseInt(attrValue);
2304 break;
2305 case ATTR_LASTDESCRIPTION:
2306 lastDescription = attrValue;
2307 break;
2308 case ATTR_LASTTIMEMOVED:
2309 lastTimeOnTop = Long.parseLong(attrValue);
2310 break;
2311 case ATTR_NEVERRELINQUISH:
2312 neverRelinquishIdentity = Boolean.parseBoolean(attrValue);
2313 break;
2314 case ATTR_TASK_AFFILIATION:
2315 taskAffiliation = Integer.parseInt(attrValue);
2316 break;
2317 case ATTR_PREV_AFFILIATION:
2318 prevTaskId = Integer.parseInt(attrValue);
2319 break;
2320 case ATTR_NEXT_AFFILIATION:
2321 nextTaskId = Integer.parseInt(attrValue);
2322 break;
2323 case ATTR_TASK_AFFILIATION_COLOR:
2324 taskAffiliationColor = Integer.parseInt(attrValue);
2325 break;
2326 case ATTR_CALLING_UID:
2327 callingUid = Integer.parseInt(attrValue);
2328 break;
2329 case ATTR_CALLING_PACKAGE:
2330 callingPackage = attrValue;
2331 break;
2332 case ATTR_RESIZE_MODE:
2333 resizeMode = Integer.parseInt(attrValue);
2334 break;
2335 case ATTR_SUPPORTS_PICTURE_IN_PICTURE:
2336 supportsPictureInPicture = Boolean.parseBoolean(attrValue);
2337 break;
2338 case ATTR_NON_FULLSCREEN_BOUNDS:
2339 bounds = Rect.unflattenFromString(attrValue);
2340 break;
2341 case ATTR_MIN_WIDTH:
2342 minWidth = Integer.parseInt(attrValue);
2343 break;
2344 case ATTR_MIN_HEIGHT:
2345 minHeight = Integer.parseInt(attrValue);
2346 break;
2347 case ATTR_PERSIST_TASK_VERSION:
2348 persistTaskVersion = Integer.parseInt(attrValue);
2349 break;
2350 default:
2351 if (attrName.startsWith(TaskDescription.ATTR_TASKDESCRIPTION_PREFIX)) {
2352 taskDescription.restoreFromXml(attrName, attrValue);
2353 } else {
2354 Slog.w(TAG, "TaskRecord: Unknown attribute=" + attrName);
2355 }
2356 }
2357 }
2358
2359 int event;
2360 while (((event = in.next()) != XmlPullParser.END_DOCUMENT) &&
2361 (event != XmlPullParser.END_TAG || in.getDepth() >= outerDepth)) {
2362 if (event == XmlPullParser.START_TAG) {
2363 final String name = in.getName();
2364 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG,
2365 "TaskRecord: START_TAG name=" + name);
2366 if (TAG_AFFINITYINTENT.equals(name)) {
2367 affinityIntent = Intent.restoreFromXml(in);
2368 } else if (TAG_INTENT.equals(name)) {
2369 intent = Intent.restoreFromXml(in);
2370 } else if (TAG_ACTIVITY.equals(name)) {
2371 ActivityRecord activity =
2372 ActivityRecord.restoreFromXml(in, stackSupervisor);
2373 if (TaskPersister.DEBUG) Slog.d(TaskPersister.TAG, "TaskRecord: activity=" +
2374 activity);
2375 if (activity != null) {
2376 activities.add(activity);
2377 }
2378 } else {
2379 Slog.e(TAG, "restoreTask: Unexpected name=" + name);
2380 XmlUtils.skipCurrentTag(in);
2381 }
2382 }
2383 }
2384 if (!hasRootAffinity) {
2385 rootAffinity = affinity;
2386 } else if ("@".equals(rootAffinity)) {
2387 rootAffinity = null;
2388 }
2389 if (effectiveUid <= 0) {
2390 Intent checkIntent = intent != null ? intent : affinityIntent;
2391 effectiveUid = 0;
2392 if (checkIntent != null) {
2393 IPackageManager pm = AppGlobals.getPackageManager();
2394 try {
2395 ApplicationInfo ai = pm.getApplicationInfo(
2396 checkIntent.getComponent().getPackageName(),
2397 PackageManager.MATCH_UNINSTALLED_PACKAGES
2398 | PackageManager.MATCH_DISABLED_COMPONENTS, userId);
2399 if (ai != null) {
2400 effectiveUid = ai.uid;
2401 }
2402 } catch (RemoteException e) {
2403 }
2404 }
2405 Slog.w(TAG, "Updating task #" + taskId + " for " + checkIntent
2406 + ": effectiveUid=" + effectiveUid);
2407 }
2408
2409 if (persistTaskVersion < 1) {
2410 // We need to convert the resize mode of home activities saved before version one if
2411 // they are marked as RESIZE_MODE_RESIZEABLE to
2412 // RESIZE_MODE_RESIZEABLE_VIA_SDK_VERSION since we didn't have that differentiation
2413 // before version 1 and the system didn't resize home activities before then.
2414 if (taskType == 1 /* old home type */ && resizeMode == RESIZE_MODE_RESIZEABLE) {
2415 resizeMode = RESIZE_MODE_RESIZEABLE_VIA_SDK_VERSION;
2416 }
2417 } else {
2418 // This activity has previously marked itself explicitly as both resizeable and
2419 // supporting picture-in-picture. Since there is no longer a requirement for
2420 // picture-in-picture activities to be resizeable, we can mark this simply as
2421 // resizeable and supporting picture-in-picture separately.
2422 if (resizeMode == RESIZE_MODE_RESIZEABLE_AND_PIPABLE_DEPRECATED) {
2423 resizeMode = RESIZE_MODE_RESIZEABLE;
2424 supportsPictureInPicture = true;
2425 }
2426 }
2427
2428 final TaskRecord task = create(stackSupervisor.mService, taskId, intent, affinityIntent,
2429 affinity, rootAffinity, realActivity, origActivity, rootHasReset,
2430 autoRemoveRecents, askedCompatMode, userId, effectiveUid, lastDescription,
2431 activities, lastTimeOnTop, neverRelinquishIdentity, taskDescription,
2432 taskAffiliation, prevTaskId, nextTaskId, taskAffiliationColor, callingUid,
2433 callingPackage, resizeMode, supportsPictureInPicture, realActivitySuspended,
2434 userSetupComplete, minWidth, minHeight);
2435 task.updateOverrideConfiguration(bounds);
2436
2437 for (int activityNdx = activities.size() - 1; activityNdx >=0; --activityNdx) {
2438 activities.get(activityNdx).setTask(task);
2439 }
2440
2441 if (DEBUG_RECENTS) Slog.d(TAG_RECENTS, "Restored task=" + task);
2442 return task;
2443 }
2444 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002445}