1 /*
2  * Copyright (C) 2018 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.app;
18 
19 import android.annotation.RequiresPermission;
20 import android.annotation.SystemService;
21 import android.annotation.TestApi;
22 import android.annotation.UnsupportedAppUsage;
23 import android.content.Context;
24 import android.content.pm.PackageManager;
25 import android.content.res.Resources;
26 import android.graphics.Rect;
27 import android.os.Handler;
28 import android.os.IBinder;
29 import android.os.RemoteException;
30 import android.os.ServiceManager;
31 import android.util.Singleton;
32 
33 import java.util.List;
34 
35 /**
36  * This class gives information about, and interacts with activities and their containers like task,
37  * stacks, and displays.
38  *
39  * @hide
40  */
41 @TestApi
42 @SystemService(Context.ACTIVITY_TASK_SERVICE)
43 public class ActivityTaskManager {
44 
45     /** Invalid stack ID. */
46     public static final int INVALID_STACK_ID = -1;
47 
48     /**
49      * Invalid task ID.
50      * @hide
51      */
52     public static final int INVALID_TASK_ID = -1;
53 
54     /**
55      * Parameter to {@link IActivityTaskManager#setTaskWindowingModeSplitScreenPrimary} which
56      * specifies the position of the created docked stack at the top half of the screen if
57      * in portrait mode or at the left half of the screen if in landscape mode.
58      */
59     public static final int SPLIT_SCREEN_CREATE_MODE_TOP_OR_LEFT = 0;
60 
61     /**
62      * Parameter to {@link IActivityTaskManager#setTaskWindowingModeSplitScreenPrimary} which
63      * specifies the position of the created docked stack at the bottom half of the screen if
64      * in portrait mode or at the right half of the screen if in landscape mode.
65      */
66     public static final int SPLIT_SCREEN_CREATE_MODE_BOTTOM_OR_RIGHT = 1;
67 
68     /**
69      * Input parameter to {@link IActivityTaskManager#resizeTask} which indicates
70      * that the resize doesn't need to preserve the window, and can be skipped if bounds
71      * is unchanged. This mode is used by window manager in most cases.
72      * @hide
73      */
74     public static final int RESIZE_MODE_SYSTEM = 0;
75 
76     /**
77      * Input parameter to {@link IActivityTaskManager#resizeTask} which indicates
78      * that the resize should preserve the window if possible.
79      * @hide
80      */
81     public static final int RESIZE_MODE_PRESERVE_WINDOW   = (0x1 << 0);
82 
83     /**
84      * Input parameter to {@link IActivityTaskManager#resizeTask} used when the
85      * resize is due to a drag action.
86      * @hide
87      */
88     public static final int RESIZE_MODE_USER = RESIZE_MODE_PRESERVE_WINDOW;
89 
90     /**
91      * Input parameter to {@link IActivityTaskManager#resizeTask} used by window
92      * manager during a screen rotation.
93      * @hide
94      */
95     public static final int RESIZE_MODE_SYSTEM_SCREEN_ROTATION = RESIZE_MODE_PRESERVE_WINDOW;
96 
97     /**
98      * Input parameter to {@link IActivityTaskManager#resizeTask} which indicates
99      * that the resize should be performed even if the bounds appears unchanged.
100      * @hide
101      */
102     public static final int RESIZE_MODE_FORCED = (0x1 << 1);
103 
104     /**
105      * Input parameter to {@link IActivityTaskManager#resizeTask} which indicates
106      * that the resize should preserve the window if possible, and should not be skipped
107      * even if the bounds is unchanged. Usually used to force a resizing when a drag action
108      * is ending.
109      * @hide
110      */
111     public static final int RESIZE_MODE_USER_FORCED =
112             RESIZE_MODE_PRESERVE_WINDOW | RESIZE_MODE_FORCED;
113 
114     /**
115      * Extra included on intents that are delegating the call to
116      * ActivityManager#startActivityAsCaller to another app.  This token is necessary for that call
117      * to succeed.  Type is IBinder.
118      * @hide
119      */
120     public static final String EXTRA_PERMISSION_TOKEN = "android.app.extra.PERMISSION_TOKEN";
121 
122     /**
123      * Extra included on intents that contain an EXTRA_INTENT, with options that the contained
124      * intent may want to be started with.  Type is Bundle.
125      * TODO: remove once the ChooserActivity moves to systemui
126      * @hide
127      */
128     public static final String EXTRA_OPTIONS = "android.app.extra.OPTIONS";
129 
130     /**
131      * Extra included on intents that contain an EXTRA_INTENT, use this boolean value for the
132      * parameter of the same name when starting the contained intent.
133      * TODO: remove once the ChooserActivity moves to systemui
134      * @hide
135      */
136     public static final String EXTRA_IGNORE_TARGET_SECURITY =
137             "android.app.extra.EXTRA_IGNORE_TARGET_SECURITY";
138 
139 
140     private static int sMaxRecentTasks = -1;
141 
ActivityTaskManager(Context context, Handler handler)142     ActivityTaskManager(Context context, Handler handler) {
143     }
144 
145     /** @hide */
getService()146     public static IActivityTaskManager getService() {
147         return IActivityTaskManagerSingleton.get();
148     }
149 
150     @UnsupportedAppUsage(trackingBug = 129726065)
151     private static final Singleton<IActivityTaskManager> IActivityTaskManagerSingleton =
152             new Singleton<IActivityTaskManager>() {
153                 @Override
154                 protected IActivityTaskManager create() {
155                     final IBinder b = ServiceManager.getService(Context.ACTIVITY_TASK_SERVICE);
156                     return IActivityTaskManager.Stub.asInterface(b);
157                 }
158             };
159 
160     /**
161      * Sets the windowing mode for a specific task. Only works on tasks of type
162      * {@link WindowConfiguration#ACTIVITY_TYPE_STANDARD}
163      * @param taskId The id of the task to set the windowing mode for.
164      * @param windowingMode The windowing mode to set for the task.
165      * @param toTop If the task should be moved to the top once the windowing mode changes.
166      */
167     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
setTaskWindowingMode(int taskId, int windowingMode, boolean toTop)168     public void setTaskWindowingMode(int taskId, int windowingMode, boolean toTop)
169             throws SecurityException {
170         try {
171             getService().setTaskWindowingMode(taskId, windowingMode, toTop);
172         } catch (RemoteException e) {
173             throw e.rethrowFromSystemServer();
174         }
175     }
176 
177     /**
178      * Moves the input task to the primary-split-screen stack.
179      * @param taskId Id of task to move.
180      * @param createMode The mode the primary split screen stack should be created in if it doesn't
181      *                   exist already. See
182      *                   {@link ActivityTaskManager#SPLIT_SCREEN_CREATE_MODE_TOP_OR_LEFT}
183      *                   and
184      *                   {@link android.app.ActivityManager
185      *                        #SPLIT_SCREEN_CREATE_MODE_BOTTOM_OR_RIGHT}
186      * @param toTop If the task and stack should be moved to the top.
187      * @param animate Whether we should play an animation for the moving the task
188      * @param initialBounds If the primary stack gets created, it will use these bounds for the
189      *                      docked stack. Pass {@code null} to use default bounds.
190      * @param showRecents If the recents activity should be shown on the other side of the task
191      *                    going into split-screen mode.
192      */
193     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
setTaskWindowingModeSplitScreenPrimary(int taskId, int createMode, boolean toTop, boolean animate, Rect initialBounds, boolean showRecents)194     public void setTaskWindowingModeSplitScreenPrimary(int taskId, int createMode, boolean toTop,
195             boolean animate, Rect initialBounds, boolean showRecents) throws SecurityException {
196         try {
197             getService().setTaskWindowingModeSplitScreenPrimary(taskId, createMode, toTop, animate,
198                     initialBounds, showRecents);
199         } catch (RemoteException e) {
200             throw e.rethrowFromSystemServer();
201         }
202     }
203 
204     /**
205      * Resizes the input stack id to the given bounds.
206      * @param stackId Id of the stack to resize.
207      * @param bounds Bounds to resize the stack to or {@code null} for fullscreen.
208      */
209     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
resizeStack(int stackId, Rect bounds)210     public void resizeStack(int stackId, Rect bounds) throws SecurityException {
211         try {
212             getService().resizeStack(stackId, bounds, false /* allowResizeInDockedMode */,
213                     false /* preserveWindows */, false /* animate */, -1 /* animationDuration */);
214         } catch (RemoteException e) {
215             throw e.rethrowFromSystemServer();
216         }
217     }
218 
219     /**
220      * Removes stacks in the windowing modes from the system if they are of activity type
221      * ACTIVITY_TYPE_STANDARD or ACTIVITY_TYPE_UNDEFINED
222      */
223     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
removeStacksInWindowingModes(int[] windowingModes)224     public void removeStacksInWindowingModes(int[] windowingModes) throws SecurityException {
225         try {
226             getService().removeStacksInWindowingModes(windowingModes);
227         } catch (RemoteException e) {
228             throw e.rethrowFromSystemServer();
229         }
230     }
231 
232     /** Removes stack of the activity types from the system. */
233     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
removeStacksWithActivityTypes(int[] activityTypes)234     public void removeStacksWithActivityTypes(int[] activityTypes) throws SecurityException {
235         try {
236             getService().removeStacksWithActivityTypes(activityTypes);
237         } catch (RemoteException e) {
238             throw e.rethrowFromSystemServer();
239         }
240     }
241 
242     /**
243      * Removes all visible recent tasks from the system.
244      * @hide
245      */
246     @RequiresPermission(android.Manifest.permission.REMOVE_TASKS)
removeAllVisibleRecentTasks()247     public void removeAllVisibleRecentTasks() {
248         try {
249             getService().removeAllVisibleRecentTasks();
250         } catch (RemoteException e) {
251             throw e.rethrowFromSystemServer();
252         }
253     }
254 
255     /**
256      * Return the maximum number of recents entries that we will maintain and show.
257      * @hide
258      */
getMaxRecentTasksStatic()259     public static int getMaxRecentTasksStatic() {
260         if (sMaxRecentTasks < 0) {
261             return sMaxRecentTasks = ActivityManager.isLowRamDeviceStatic() ? 36 : 48;
262         }
263         return sMaxRecentTasks;
264     }
265 
266     /**
267      * Return the default limit on the number of recents that an app can make.
268      * @hide
269      */
getDefaultAppRecentsLimitStatic()270     public static int getDefaultAppRecentsLimitStatic() {
271         return getMaxRecentTasksStatic() / 6;
272     }
273 
274     /**
275      * Return the maximum limit on the number of recents that an app can make.
276      * @hide
277      */
getMaxAppRecentsLimitStatic()278     public static int getMaxAppRecentsLimitStatic() {
279         return getMaxRecentTasksStatic() / 2;
280     }
281 
282     /**
283      * Returns true if the system supports at least one form of multi-window.
284      * E.g. freeform, split-screen, picture-in-picture.
285      */
supportsMultiWindow(Context context)286     public static boolean supportsMultiWindow(Context context) {
287         // On watches, multi-window is used to present essential system UI, and thus it must be
288         // supported regardless of device memory characteristics.
289         boolean isWatch = context.getPackageManager().hasSystemFeature(
290                 PackageManager.FEATURE_WATCH);
291         return (!ActivityManager.isLowRamDeviceStatic() || isWatch)
292                 && Resources.getSystem().getBoolean(
293                 com.android.internal.R.bool.config_supportsMultiWindow);
294     }
295 
296     /** Returns true if the system supports split screen multi-window. */
supportsSplitScreenMultiWindow(Context context)297     public static boolean supportsSplitScreenMultiWindow(Context context) {
298         return supportsMultiWindow(context)
299                 && Resources.getSystem().getBoolean(
300                 com.android.internal.R.bool.config_supportsSplitScreenMultiWindow);
301     }
302 
303     /**
304      * Moves the top activity in the input stackId to the pinned stack.
305      * @param stackId Id of stack to move the top activity to pinned stack.
306      * @param bounds Bounds to use for pinned stack.
307      * @return True if the top activity of stack was successfully moved to the pinned stack.
308      */
309     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
moveTopActivityToPinnedStack(int stackId, Rect bounds)310     public boolean moveTopActivityToPinnedStack(int stackId, Rect bounds) {
311         try {
312             return getService().moveTopActivityToPinnedStack(stackId, bounds);
313         } catch (RemoteException e) {
314             throw e.rethrowFromSystemServer();
315         }
316     }
317 
318     /**
319      * Start to enter lock task mode for given task by system(UI).
320      * @param taskId Id of task to lock.
321      */
322     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
startSystemLockTaskMode(int taskId)323     public void startSystemLockTaskMode(int taskId) {
324         try {
325             getService().startSystemLockTaskMode(taskId);
326         } catch (RemoteException e) {
327             throw e.rethrowFromSystemServer();
328         }
329     }
330 
331     /**
332      * Stop lock task mode by system(UI).
333      */
334     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
stopSystemLockTaskMode()335     public void stopSystemLockTaskMode() {
336         try {
337             getService().stopSystemLockTaskMode();
338         } catch (RemoteException e) {
339             throw e.rethrowFromSystemServer();
340         }
341     }
342 
343     /**
344      * Move task to stack with given id.
345      * @param taskId Id of the task to move.
346      * @param stackId Id of the stack for task moving.
347      * @param toTop Whether the given task should shown to top of stack.
348      */
349     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
moveTaskToStack(int taskId, int stackId, boolean toTop)350     public void moveTaskToStack(int taskId, int stackId, boolean toTop) {
351         try {
352             getService().moveTaskToStack(taskId, stackId, toTop);
353         } catch (RemoteException e) {
354             throw e.rethrowFromSystemServer();
355         }
356     }
357 
358     /**
359      * Resize the input stack id to the given bounds with animate setting.
360      * @param stackId Id of the stack to resize.
361      * @param bounds Bounds to resize the stack to or {@code null} for fullscreen.
362      * @param animate Whether we should play an animation for resizing stack.
363      */
364     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
resizeStack(int stackId, Rect bounds, boolean animate)365     public void resizeStack(int stackId, Rect bounds, boolean animate) {
366         try {
367             getService().resizeStack(stackId, bounds, false, false, animate /* animate */,
368                     -1 /* animationDuration */);
369         } catch (RemoteException e) {
370             throw e.rethrowFromSystemServer();
371         }
372     }
373 
374     /**
375      * Resize task to given bounds.
376      * @param taskId Id of task to resize.
377      * @param bounds Bounds to resize task.
378      */
379     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
resizeTask(int taskId, Rect bounds)380     public void resizeTask(int taskId, Rect bounds) {
381         try {
382             getService().resizeTask(taskId, bounds, RESIZE_MODE_SYSTEM);
383         } catch (RemoteException e) {
384             throw e.rethrowFromSystemServer();
385         }
386     }
387 
388     /**
389      * Resize docked stack & its task to given stack & task bounds.
390      * @param stackBounds Bounds to resize stack.
391      * @param taskBounds Bounds to resize task.
392      */
393     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
resizeDockedStack(Rect stackBounds, Rect taskBounds)394     public void resizeDockedStack(Rect stackBounds, Rect taskBounds) {
395         try {
396             getService().resizeDockedStack(stackBounds, taskBounds, null, null, null);
397         } catch (RemoteException e) {
398             throw e.rethrowFromSystemServer();
399         }
400     }
401 
402     /**
403      * List all activity stacks information.
404      */
405     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
listAllStacks()406     public String listAllStacks() {
407         final List<ActivityManager.StackInfo> stacks;
408         try {
409             stacks = getService().getAllStackInfos();
410         } catch (RemoteException e) {
411             throw e.rethrowFromSystemServer();
412         }
413 
414         final StringBuilder sb = new StringBuilder();
415         if (stacks != null) {
416             for (ActivityManager.StackInfo info : stacks) {
417                 sb.append(info).append("\n");
418             }
419         }
420         return sb.toString();
421     }
422 
423     /**
424      * Clears launch params for the given package.
425      * @param packageNames the names of the packages of which the launch params are to be cleared
426      */
427     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
clearLaunchParamsForPackages(List<String> packageNames)428     public void clearLaunchParamsForPackages(List<String> packageNames) {
429         try {
430             getService().clearLaunchParamsForPackages(packageNames);
431         } catch (RemoteException e) {
432             e.rethrowFromSystemServer();
433         }
434     }
435 
436     /**
437      * Makes the display with the given id a single task instance display. I.e the display can only
438      * contain one task.
439      */
440     @RequiresPermission(android.Manifest.permission.MANAGE_ACTIVITY_STACKS)
setDisplayToSingleTaskInstance(int displayId)441     public void setDisplayToSingleTaskInstance(int displayId) {
442         try {
443             getService().setDisplayToSingleTaskInstance(displayId);
444         } catch (RemoteException e) {
445             throw e.rethrowFromSystemServer();
446         }
447     }
448 }
449