| /* |
| * Copyright (C) 2015 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.systemui.shared.system; |
| |
| import static android.app.ActivityManager.LOCK_TASK_MODE_LOCKED; |
| import static android.app.ActivityManager.LOCK_TASK_MODE_NONE; |
| import static android.app.ActivityManager.LOCK_TASK_MODE_PINNED; |
| import static android.app.ActivityManager.RECENT_IGNORE_UNAVAILABLE; |
| import static android.app.ActivityTaskManager.getService; |
| |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.app.Activity; |
| import android.app.ActivityClient; |
| import android.app.ActivityManager; |
| import android.app.ActivityManager.RecentTaskInfo; |
| import android.app.ActivityManager.RunningTaskInfo; |
| import android.app.ActivityOptions; |
| import android.app.ActivityTaskManager; |
| import android.app.AppGlobals; |
| import android.app.WindowConfiguration; |
| import android.content.ContentResolver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.pm.PackageManager; |
| import android.content.pm.UserInfo; |
| import android.graphics.Rect; |
| import android.os.Bundle; |
| import android.os.Handler; |
| import android.os.IBinder; |
| import android.os.RemoteException; |
| import android.os.ServiceManager; |
| import android.os.SystemClock; |
| import android.provider.Settings; |
| import android.util.Log; |
| import android.view.IRecentsAnimationController; |
| import android.view.IRecentsAnimationRunner; |
| import android.view.RemoteAnimationTarget; |
| import android.window.TaskSnapshot; |
| |
| import com.android.internal.app.IVoiceInteractionManagerService; |
| import com.android.systemui.shared.recents.model.Task; |
| import com.android.systemui.shared.recents.model.ThumbnailData; |
| |
| import java.util.List; |
| import java.util.function.Consumer; |
| |
| public class ActivityManagerWrapper { |
| |
| private static final String TAG = "ActivityManagerWrapper"; |
| private static final int NUM_RECENT_ACTIVITIES_REQUEST = 3; |
| private static final ActivityManagerWrapper sInstance = new ActivityManagerWrapper(); |
| |
| // Should match the values in PhoneWindowManager |
| public static final String CLOSE_SYSTEM_WINDOWS_REASON_RECENTS = "recentapps"; |
| public static final String CLOSE_SYSTEM_WINDOWS_REASON_HOME_KEY = "homekey"; |
| |
| // Should match the value in AssistManager |
| private static final String INVOCATION_TIME_MS_KEY = "invocation_time_ms"; |
| |
| private final ActivityTaskManager mAtm = ActivityTaskManager.getInstance(); |
| private ActivityManagerWrapper() { } |
| |
| public static ActivityManagerWrapper getInstance() { |
| return sInstance; |
| } |
| |
| /** |
| * @return the current user's id. |
| */ |
| public int getCurrentUserId() { |
| UserInfo ui; |
| try { |
| ui = ActivityManager.getService().getCurrentUser(); |
| return ui != null ? ui.id : 0; |
| } catch (RemoteException e) { |
| throw e.rethrowFromSystemServer(); |
| } |
| } |
| |
| /** |
| * @return the top running task (can be {@code null}). |
| */ |
| public ActivityManager.RunningTaskInfo getRunningTask() { |
| return getRunningTask(false /* filterVisibleRecents */); |
| } |
| |
| /** |
| * @return the top running task filtering only for tasks that can be visible in the recent tasks |
| * list (can be {@code null}). |
| */ |
| public ActivityManager.RunningTaskInfo getRunningTask(boolean filterOnlyVisibleRecents) { |
| // Note: The set of running tasks from the system is ordered by recency |
| List<ActivityManager.RunningTaskInfo> tasks = |
| mAtm.getTasks(1, filterOnlyVisibleRecents); |
| if (tasks.isEmpty()) { |
| return null; |
| } |
| return tasks.get(0); |
| } |
| |
| /** |
| * We ask for {@link #NUM_RECENT_ACTIVITIES_REQUEST} activities because when in split screen, |
| * we'll get back 2 activities for each split app and one for launcher. Launcher might be more |
| * "recently" used than one of the split apps so if we only request 2 tasks, then we might miss |
| * out on one of the split apps |
| * |
| * @return an array of up to {@link #NUM_RECENT_ACTIVITIES_REQUEST} running tasks |
| * filtering only for tasks that can be visible in the recent tasks list. |
| */ |
| public ActivityManager.RunningTaskInfo[] getRunningTasks(boolean filterOnlyVisibleRecents) { |
| // Note: The set of running tasks from the system is ordered by recency |
| List<ActivityManager.RunningTaskInfo> tasks = |
| mAtm.getTasks(NUM_RECENT_ACTIVITIES_REQUEST, filterOnlyVisibleRecents); |
| return tasks.toArray(new RunningTaskInfo[tasks.size()]); |
| } |
| |
| /** |
| * @return a list of the recents tasks. |
| */ |
| public List<RecentTaskInfo> getRecentTasks(int numTasks, int userId) { |
| return mAtm.getRecentTasks(numTasks, RECENT_IGNORE_UNAVAILABLE, userId); |
| } |
| |
| /** |
| * @return a {@link ThumbnailData} with {@link TaskSnapshot} for the given {@param taskId}. |
| * The snapshot will be triggered if no cached {@link TaskSnapshot} exists. |
| */ |
| public @NonNull ThumbnailData getTaskThumbnail(int taskId, boolean isLowResolution) { |
| TaskSnapshot snapshot = null; |
| try { |
| snapshot = getService().getTaskSnapshot(taskId, isLowResolution, |
| true /* takeSnapshotIfNeeded */); |
| } catch (RemoteException e) { |
| Log.w(TAG, "Failed to retrieve task snapshot", e); |
| } |
| if (snapshot != null) { |
| return new ThumbnailData(snapshot); |
| } else { |
| return new ThumbnailData(); |
| } |
| } |
| |
| /** |
| * Removes the outdated snapshot of home task. |
| * |
| * @param homeActivity The home task activity, or null if you have the |
| * {@link android.Manifest.permission#MANAGE_ACTIVITY_TASKS} permission and |
| * want us to find the home task for you. |
| */ |
| public void invalidateHomeTaskSnapshot(@Nullable final Activity homeActivity) { |
| try { |
| ActivityClient.getInstance().invalidateHomeTaskSnapshot( |
| homeActivity == null ? null : homeActivity.getActivityToken()); |
| } catch (Throwable e) { |
| Log.w(TAG, "Failed to invalidate home snapshot", e); |
| } |
| } |
| |
| /** |
| * Starts the recents activity. The caller should manage the thread on which this is called. |
| */ |
| public void startRecentsActivity(Intent intent, long eventTime, |
| final RecentsAnimationListener animationHandler, final Consumer<Boolean> resultCallback, |
| Handler resultCallbackHandler) { |
| boolean result = startRecentsActivity(intent, eventTime, animationHandler); |
| if (resultCallback != null) { |
| resultCallbackHandler.post(new Runnable() { |
| @Override |
| public void run() { |
| resultCallback.accept(result); |
| } |
| }); |
| } |
| } |
| |
| /** |
| * Starts the recents activity. The caller should manage the thread on which this is called. |
| */ |
| public boolean startRecentsActivity( |
| Intent intent, long eventTime, RecentsAnimationListener animationHandler) { |
| try { |
| IRecentsAnimationRunner runner = null; |
| if (animationHandler != null) { |
| runner = new IRecentsAnimationRunner.Stub() { |
| @Override |
| public void onAnimationStart(IRecentsAnimationController controller, |
| RemoteAnimationTarget[] apps, RemoteAnimationTarget[] wallpapers, |
| Rect homeContentInsets, Rect minimizedHomeBounds) { |
| final RecentsAnimationControllerCompat controllerCompat = |
| new RecentsAnimationControllerCompat(controller); |
| final RemoteAnimationTargetCompat[] appsCompat = |
| RemoteAnimationTargetCompat.wrap(apps); |
| final RemoteAnimationTargetCompat[] wallpapersCompat = |
| RemoteAnimationTargetCompat.wrap(wallpapers); |
| animationHandler.onAnimationStart(controllerCompat, appsCompat, |
| wallpapersCompat, homeContentInsets, minimizedHomeBounds); |
| } |
| |
| @Override |
| public void onAnimationCanceled(int[] taskIds, TaskSnapshot[] taskSnapshots) { |
| animationHandler.onAnimationCanceled( |
| ThumbnailData.wrap(taskIds, taskSnapshots)); |
| } |
| |
| @Override |
| public void onTasksAppeared(RemoteAnimationTarget[] apps) { |
| final RemoteAnimationTargetCompat[] compats = |
| new RemoteAnimationTargetCompat[apps.length]; |
| for (int i = 0; i < apps.length; ++i) { |
| compats[i] = new RemoteAnimationTargetCompat(apps[i]); |
| } |
| animationHandler.onTasksAppeared(compats); |
| } |
| }; |
| } |
| getService().startRecentsActivity(intent, eventTime, runner); |
| return true; |
| } catch (Exception e) { |
| return false; |
| } |
| } |
| |
| /** |
| * Cancels the remote recents animation started from {@link #startRecentsActivity}. |
| */ |
| public void cancelRecentsAnimation(boolean restoreHomeRootTaskPosition) { |
| try { |
| getService().cancelRecentsAnimation(restoreHomeRootTaskPosition); |
| } catch (RemoteException e) { |
| Log.e(TAG, "Failed to cancel recents animation", e); |
| } |
| } |
| |
| /** |
| * Starts a task from Recents. |
| * |
| * @param resultCallback The result success callback |
| * @param resultCallbackHandler The handler to receive the result callback |
| */ |
| public void startActivityFromRecentsAsync(Task.TaskKey taskKey, ActivityOptions options, |
| Consumer<Boolean> resultCallback, Handler resultCallbackHandler) { |
| final boolean result = startActivityFromRecents(taskKey, options); |
| if (resultCallback != null) { |
| resultCallbackHandler.post(new Runnable() { |
| @Override |
| public void run() { |
| resultCallback.accept(result); |
| } |
| }); |
| } |
| } |
| |
| /** |
| * Starts a task from Recents synchronously. |
| */ |
| public boolean startActivityFromRecents(Task.TaskKey taskKey, ActivityOptions options) { |
| return startActivityFromRecents(taskKey.id, options); |
| } |
| |
| /** |
| * Starts a task from Recents synchronously. |
| */ |
| public boolean startActivityFromRecents(int taskId, ActivityOptions options) { |
| try { |
| Bundle optsBundle = options == null ? null : options.toBundle(); |
| getService().startActivityFromRecents(taskId, optsBundle); |
| return true; |
| } catch (Exception e) { |
| return false; |
| } |
| } |
| |
| /** |
| * @deprecated use {@link TaskStackChangeListeners#registerTaskStackListener} |
| */ |
| public void registerTaskStackListener(TaskStackChangeListener listener) { |
| TaskStackChangeListeners.getInstance().registerTaskStackListener(listener); |
| } |
| |
| /** |
| * @deprecated use {@link TaskStackChangeListeners#unregisterTaskStackListener} |
| */ |
| public void unregisterTaskStackListener(TaskStackChangeListener listener) { |
| TaskStackChangeListeners.getInstance().unregisterTaskStackListener(listener); |
| } |
| |
| /** |
| * Requests that the system close any open system windows (including other SystemUI). |
| */ |
| public void closeSystemWindows(final String reason) { |
| try { |
| ActivityManager.getService().closeSystemDialogs(reason); |
| } catch (RemoteException e) { |
| Log.w(TAG, "Failed to close system windows", e); |
| } |
| } |
| |
| /** |
| * Removes a task by id. |
| */ |
| public void removeTask(final int taskId) { |
| try { |
| getService().removeTask(taskId); |
| } catch (RemoteException e) { |
| Log.w(TAG, "Failed to remove task=" + taskId, e); |
| } |
| } |
| |
| /** |
| * Removes all the recent tasks. |
| */ |
| public void removeAllRecentTasks() { |
| try { |
| getService().removeAllVisibleRecentTasks(); |
| } catch (RemoteException e) { |
| Log.w(TAG, "Failed to remove all tasks", e); |
| } |
| } |
| |
| /** |
| * @return whether screen pinning is active. |
| */ |
| public boolean isScreenPinningActive() { |
| try { |
| return getService().getLockTaskModeState() == LOCK_TASK_MODE_PINNED; |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| /** |
| * @return whether screen pinning is enabled. |
| */ |
| public boolean isScreenPinningEnabled() { |
| final ContentResolver cr = AppGlobals.getInitialApplication().getContentResolver(); |
| return Settings.System.getInt(cr, Settings.System.LOCK_TO_APP_ENABLED, 0) != 0; |
| } |
| |
| /** |
| * @return whether there is currently a locked task (ie. in screen pinning). |
| */ |
| public boolean isLockToAppActive() { |
| try { |
| return getService().getLockTaskModeState() != LOCK_TASK_MODE_NONE; |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| /** |
| * @return whether lock task mode is active in kiosk-mode (not screen pinning). |
| */ |
| public boolean isLockTaskKioskModeActive() { |
| try { |
| return getService().getLockTaskModeState() == LOCK_TASK_MODE_LOCKED; |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| /** |
| * Shows a voice session identified by {@code token} |
| * @return true if the session was shown, false otherwise |
| */ |
| public boolean showVoiceSession(IBinder token, Bundle args, int flags) { |
| IVoiceInteractionManagerService service = IVoiceInteractionManagerService.Stub.asInterface( |
| ServiceManager.getService(Context.VOICE_INTERACTION_MANAGER_SERVICE)); |
| if (service == null) { |
| return false; |
| } |
| args.putLong(INVOCATION_TIME_MS_KEY, SystemClock.elapsedRealtime()); |
| |
| try { |
| return service.showSessionFromSession(token, args, flags); |
| } catch (RemoteException e) { |
| return false; |
| } |
| } |
| |
| /** |
| * Returns true if the system supports freeform multi-window. |
| */ |
| public boolean supportsFreeformMultiWindow(Context context) { |
| final boolean freeformDevOption = Settings.Global.getInt(context.getContentResolver(), |
| Settings.Global.DEVELOPMENT_ENABLE_FREEFORM_WINDOWS_SUPPORT, 0) != 0; |
| return ActivityTaskManager.supportsMultiWindow(context) |
| && (context.getPackageManager().hasSystemFeature( |
| PackageManager.FEATURE_FREEFORM_WINDOW_MANAGEMENT) |
| || freeformDevOption); |
| } |
| |
| /** |
| * Returns true if the running task represents the home task |
| */ |
| public static boolean isHomeTask(RunningTaskInfo info) { |
| return info.configuration.windowConfiguration.getActivityType() |
| == WindowConfiguration.ACTIVITY_TYPE_HOME; |
| } |
| } |