1 /* 2 * Copyright (C) 2017 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 com.android.systemui.shared.system; 18 19 import android.annotation.NonNull; 20 import android.app.ActivityManager.RunningTaskInfo; 21 import android.app.ActivityTaskManager; 22 import android.app.TaskStackListener; 23 import android.content.ComponentName; 24 import android.os.Handler; 25 import android.os.Looper; 26 import android.os.Message; 27 import android.os.Trace; 28 import android.util.Log; 29 import android.window.TaskSnapshot; 30 31 import androidx.annotation.VisibleForTesting; 32 33 import com.android.internal.os.SomeArgs; 34 import com.android.systemui.shared.recents.model.ThumbnailData; 35 36 import java.util.ArrayList; 37 import java.util.List; 38 39 /** 40 * Tracks all the task stack listeners 41 */ 42 public class TaskStackChangeListeners { 43 44 private static final String TAG = TaskStackChangeListeners.class.getSimpleName(); 45 private static final TaskStackChangeListeners INSTANCE = new TaskStackChangeListeners(); 46 47 private final Impl mImpl; 48 49 /** 50 * Proxies calls to the given handler callback synchronously for testing purposes. 51 */ 52 private static class TestSyncHandler extends Handler { 53 private Handler.Callback mCb; 54 TestSyncHandler()55 public TestSyncHandler() { 56 super(Looper.getMainLooper()); 57 } 58 setCallback(Handler.Callback cb)59 public void setCallback(Handler.Callback cb) { 60 mCb = cb; 61 } 62 63 @Override sendMessageAtTime(@onNull Message msg, long uptimeMillis)64 public boolean sendMessageAtTime(@NonNull Message msg, long uptimeMillis) { 65 return mCb.handleMessage(msg); 66 } 67 } 68 TaskStackChangeListeners()69 private TaskStackChangeListeners() { 70 mImpl = new Impl(Looper.getMainLooper()); 71 } 72 TaskStackChangeListeners(Handler h)73 private TaskStackChangeListeners(Handler h) { 74 mImpl = new Impl(h); 75 } 76 getInstance()77 public static TaskStackChangeListeners getInstance() { 78 return INSTANCE; 79 } 80 81 /** 82 * Returns an instance of the listeners that can be called upon synchronously for testsing 83 * purposes. 84 */ 85 @VisibleForTesting getTestInstance()86 public static TaskStackChangeListeners getTestInstance() { 87 TestSyncHandler h = new TestSyncHandler(); 88 TaskStackChangeListeners l = new TaskStackChangeListeners(h); 89 h.setCallback(l.mImpl); 90 return l; 91 } 92 93 /** 94 * Registers a task stack listener with the system. 95 * This should be called on the main thread. 96 */ registerTaskStackListener(TaskStackChangeListener listener)97 public void registerTaskStackListener(TaskStackChangeListener listener) { 98 synchronized (mImpl) { 99 mImpl.addListener(listener); 100 } 101 } 102 103 /** 104 * Unregisters a task stack listener with the system. 105 * This should be called on the main thread. 106 */ unregisterTaskStackListener(TaskStackChangeListener listener)107 public void unregisterTaskStackListener(TaskStackChangeListener listener) { 108 synchronized (mImpl) { 109 mImpl.removeListener(listener); 110 } 111 } 112 113 /** 114 * Returns an instance of the listener to call upon from tests. 115 */ 116 @VisibleForTesting getListenerImpl()117 public TaskStackListener getListenerImpl() { 118 return mImpl; 119 } 120 121 private class Impl extends TaskStackListener implements Handler.Callback { 122 123 private static final int ON_TASK_STACK_CHANGED = 1; 124 private static final int ON_TASK_SNAPSHOT_CHANGED = 2; 125 private static final int ON_ACTIVITY_PINNED = 3; 126 private static final int ON_ACTIVITY_RESTART_ATTEMPT = 4; 127 private static final int ON_ACTIVITY_FORCED_RESIZABLE = 6; 128 private static final int ON_ACTIVITY_DISMISSING_DOCKED_STACK = 7; 129 private static final int ON_TASK_PROFILE_LOCKED = 8; 130 private static final int ON_ACTIVITY_UNPINNED = 10; 131 private static final int ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_FAILED = 11; 132 private static final int ON_TASK_CREATED = 12; 133 private static final int ON_TASK_REMOVED = 13; 134 private static final int ON_TASK_MOVED_TO_FRONT = 14; 135 private static final int ON_ACTIVITY_REQUESTED_ORIENTATION_CHANGE = 15; 136 private static final int ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_REROUTED = 16; 137 private static final int ON_BACK_PRESSED_ON_TASK_ROOT = 17; 138 private static final int ON_TASK_DISPLAY_CHANGED = 18; 139 private static final int ON_TASK_LIST_UPDATED = 19; 140 private static final int ON_TASK_LIST_FROZEN_UNFROZEN = 20; 141 private static final int ON_TASK_DESCRIPTION_CHANGED = 21; 142 private static final int ON_ACTIVITY_ROTATION = 22; 143 private static final int ON_LOCK_TASK_MODE_CHANGED = 23; 144 145 /** 146 * List of {@link TaskStackChangeListener} registered from {@link #addListener}. 147 */ 148 private final List<TaskStackChangeListener> mTaskStackListeners = new ArrayList<>(); 149 private final List<TaskStackChangeListener> mTmpListeners = new ArrayList<>(); 150 151 private final Handler mHandler; 152 private boolean mRegistered; 153 Impl(Looper looper)154 private Impl(Looper looper) { 155 mHandler = new Handler(looper, this); 156 } 157 Impl(Handler handler)158 private Impl(Handler handler) { 159 mHandler = handler; 160 } 161 addListener(TaskStackChangeListener listener)162 public void addListener(TaskStackChangeListener listener) { 163 synchronized (mTaskStackListeners) { 164 mTaskStackListeners.add(listener); 165 } 166 if (!mRegistered) { 167 // Register mTaskStackListener to IActivityManager only once if needed. 168 try { 169 ActivityTaskManager.getService().registerTaskStackListener(this); 170 mRegistered = true; 171 } catch (Exception e) { 172 Log.w(TAG, "Failed to call registerTaskStackListener", e); 173 } 174 } 175 } 176 removeListener(TaskStackChangeListener listener)177 public void removeListener(TaskStackChangeListener listener) { 178 boolean isEmpty; 179 synchronized (mTaskStackListeners) { 180 mTaskStackListeners.remove(listener); 181 isEmpty = mTaskStackListeners.isEmpty(); 182 } 183 if (isEmpty && mRegistered) { 184 // Unregister mTaskStackListener once we have no more listeners 185 try { 186 ActivityTaskManager.getService().unregisterTaskStackListener(this); 187 mRegistered = false; 188 } catch (Exception e) { 189 Log.w(TAG, "Failed to call unregisterTaskStackListener", e); 190 } 191 } 192 } 193 194 @Override onTaskStackChanged()195 public void onTaskStackChanged() { 196 // Call the task changed callback for the non-ui thread listeners first. Copy to a set 197 // of temp listeners so that we don't lock on mTaskStackListeners while calling all the 198 // callbacks. This call is always on the same binder thread, so we can just synchronize 199 // on the copying of the listener list. 200 synchronized (mTaskStackListeners) { 201 mTmpListeners.addAll(mTaskStackListeners); 202 } 203 for (int i = mTmpListeners.size() - 1; i >= 0; i--) { 204 mTmpListeners.get(i).onTaskStackChangedBackground(); 205 } 206 mTmpListeners.clear(); 207 208 mHandler.removeMessages(ON_TASK_STACK_CHANGED); 209 mHandler.sendEmptyMessage(ON_TASK_STACK_CHANGED); 210 } 211 212 @Override onActivityPinned(String packageName, int userId, int taskId, int stackId)213 public void onActivityPinned(String packageName, int userId, int taskId, int stackId) { 214 mHandler.removeMessages(ON_ACTIVITY_PINNED); 215 mHandler.obtainMessage(ON_ACTIVITY_PINNED, 216 new PinnedActivityInfo(packageName, userId, taskId, stackId)).sendToTarget(); 217 } 218 219 @Override onActivityUnpinned()220 public void onActivityUnpinned() { 221 mHandler.removeMessages(ON_ACTIVITY_UNPINNED); 222 mHandler.sendEmptyMessage(ON_ACTIVITY_UNPINNED); 223 } 224 225 @Override onActivityRestartAttempt(RunningTaskInfo task, boolean homeTaskVisible, boolean clearedTask, boolean wasVisible)226 public void onActivityRestartAttempt(RunningTaskInfo task, boolean homeTaskVisible, 227 boolean clearedTask, boolean wasVisible) { 228 final SomeArgs args = SomeArgs.obtain(); 229 args.arg1 = task; 230 args.argi1 = homeTaskVisible ? 1 : 0; 231 args.argi2 = clearedTask ? 1 : 0; 232 args.argi3 = wasVisible ? 1 : 0; 233 mHandler.removeMessages(ON_ACTIVITY_RESTART_ATTEMPT); 234 mHandler.obtainMessage(ON_ACTIVITY_RESTART_ATTEMPT, args).sendToTarget(); 235 } 236 237 @Override onActivityForcedResizable(String packageName, int taskId, int reason)238 public void onActivityForcedResizable(String packageName, int taskId, int reason) { 239 mHandler.obtainMessage(ON_ACTIVITY_FORCED_RESIZABLE, taskId, reason, packageName) 240 .sendToTarget(); 241 } 242 243 @Override onActivityDismissingDockedTask()244 public void onActivityDismissingDockedTask() { 245 mHandler.sendEmptyMessage(ON_ACTIVITY_DISMISSING_DOCKED_STACK); 246 } 247 248 @Override onActivityLaunchOnSecondaryDisplayFailed(RunningTaskInfo taskInfo, int requestedDisplayId)249 public void onActivityLaunchOnSecondaryDisplayFailed(RunningTaskInfo taskInfo, 250 int requestedDisplayId) { 251 mHandler.obtainMessage(ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_FAILED, 252 requestedDisplayId, 253 0 /* unused */, 254 taskInfo).sendToTarget(); 255 } 256 257 @Override onActivityLaunchOnSecondaryDisplayRerouted(RunningTaskInfo taskInfo, int requestedDisplayId)258 public void onActivityLaunchOnSecondaryDisplayRerouted(RunningTaskInfo taskInfo, 259 int requestedDisplayId) { 260 mHandler.obtainMessage(ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_REROUTED, 261 requestedDisplayId, 0 /* unused */, taskInfo).sendToTarget(); 262 } 263 264 @Override onTaskProfileLocked(RunningTaskInfo taskInfo)265 public void onTaskProfileLocked(RunningTaskInfo taskInfo) { 266 mHandler.obtainMessage(ON_TASK_PROFILE_LOCKED, taskInfo).sendToTarget(); 267 } 268 269 @Override onTaskSnapshotChanged(int taskId, TaskSnapshot snapshot)270 public void onTaskSnapshotChanged(int taskId, TaskSnapshot snapshot) { 271 mHandler.obtainMessage(ON_TASK_SNAPSHOT_CHANGED, taskId, 0, snapshot).sendToTarget(); 272 } 273 274 @Override onTaskCreated(int taskId, ComponentName componentName)275 public void onTaskCreated(int taskId, ComponentName componentName) { 276 mHandler.obtainMessage(ON_TASK_CREATED, taskId, 0, componentName).sendToTarget(); 277 } 278 279 @Override onTaskRemoved(int taskId)280 public void onTaskRemoved(int taskId) { 281 mHandler.obtainMessage(ON_TASK_REMOVED, taskId, 0).sendToTarget(); 282 } 283 284 @Override onTaskMovedToFront(RunningTaskInfo taskInfo)285 public void onTaskMovedToFront(RunningTaskInfo taskInfo) { 286 mHandler.obtainMessage(ON_TASK_MOVED_TO_FRONT, taskInfo).sendToTarget(); 287 } 288 289 @Override onBackPressedOnTaskRoot(RunningTaskInfo taskInfo)290 public void onBackPressedOnTaskRoot(RunningTaskInfo taskInfo) { 291 mHandler.obtainMessage(ON_BACK_PRESSED_ON_TASK_ROOT, taskInfo).sendToTarget(); 292 } 293 294 @Override onActivityRequestedOrientationChanged(int taskId, int requestedOrientation)295 public void onActivityRequestedOrientationChanged(int taskId, int requestedOrientation) { 296 mHandler.obtainMessage(ON_ACTIVITY_REQUESTED_ORIENTATION_CHANGE, taskId, 297 requestedOrientation).sendToTarget(); 298 } 299 300 @Override onTaskDisplayChanged(int taskId, int newDisplayId)301 public void onTaskDisplayChanged(int taskId, int newDisplayId) { 302 mHandler.obtainMessage(ON_TASK_DISPLAY_CHANGED, taskId, newDisplayId).sendToTarget(); 303 } 304 305 @Override onRecentTaskListUpdated()306 public void onRecentTaskListUpdated() { 307 mHandler.obtainMessage(ON_TASK_LIST_UPDATED).sendToTarget(); 308 } 309 310 @Override onRecentTaskListFrozenChanged(boolean frozen)311 public void onRecentTaskListFrozenChanged(boolean frozen) { 312 mHandler.obtainMessage(ON_TASK_LIST_FROZEN_UNFROZEN, frozen ? 1 : 0, 0 /* unused */) 313 .sendToTarget(); 314 } 315 316 @Override onTaskDescriptionChanged(RunningTaskInfo taskInfo)317 public void onTaskDescriptionChanged(RunningTaskInfo taskInfo) { 318 mHandler.obtainMessage(ON_TASK_DESCRIPTION_CHANGED, taskInfo).sendToTarget(); 319 } 320 321 @Override onActivityRotation(int displayId)322 public void onActivityRotation(int displayId) { 323 mHandler.obtainMessage(ON_ACTIVITY_ROTATION, displayId, 0 /* unused */) 324 .sendToTarget(); 325 } 326 327 @Override onLockTaskModeChanged(int mode)328 public void onLockTaskModeChanged(int mode) { 329 mHandler.obtainMessage(ON_LOCK_TASK_MODE_CHANGED, mode, 0 /* unused */).sendToTarget(); 330 } 331 332 @Override handleMessage(Message msg)333 public boolean handleMessage(Message msg) { 334 synchronized (mTaskStackListeners) { 335 switch (msg.what) { 336 case ON_TASK_STACK_CHANGED: { 337 Trace.beginSection("onTaskStackChanged"); 338 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 339 mTaskStackListeners.get(i).onTaskStackChanged(); 340 } 341 Trace.endSection(); 342 break; 343 } 344 case ON_TASK_SNAPSHOT_CHANGED: { 345 Trace.beginSection("onTaskSnapshotChanged"); 346 final TaskSnapshot snapshot = (TaskSnapshot) msg.obj; 347 final ThumbnailData thumbnail = new ThumbnailData(snapshot); 348 boolean snapshotConsumed = false; 349 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 350 boolean consumed = mTaskStackListeners.get(i).onTaskSnapshotChanged( 351 msg.arg1, thumbnail); 352 snapshotConsumed |= consumed; 353 } 354 if (!snapshotConsumed) { 355 thumbnail.recycleBitmap(); 356 if (snapshot.getHardwareBuffer() != null) { 357 snapshot.getHardwareBuffer().close(); 358 } 359 } 360 Trace.endSection(); 361 break; 362 } 363 case ON_ACTIVITY_PINNED: { 364 final PinnedActivityInfo info = (PinnedActivityInfo) msg.obj; 365 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 366 mTaskStackListeners.get(i).onActivityPinned( 367 info.mPackageName, info.mUserId, info.mTaskId, 368 info.mStackId); 369 } 370 break; 371 } 372 case ON_ACTIVITY_UNPINNED: { 373 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 374 mTaskStackListeners.get(i).onActivityUnpinned(); 375 } 376 break; 377 } 378 case ON_ACTIVITY_RESTART_ATTEMPT: { 379 final SomeArgs args = (SomeArgs) msg.obj; 380 final RunningTaskInfo task = (RunningTaskInfo) args.arg1; 381 final boolean homeTaskVisible = args.argi1 != 0; 382 final boolean clearedTask = args.argi2 != 0; 383 final boolean wasVisible = args.argi3 != 0; 384 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 385 mTaskStackListeners.get(i).onActivityRestartAttempt(task, 386 homeTaskVisible, clearedTask, wasVisible); 387 } 388 break; 389 } 390 case ON_ACTIVITY_FORCED_RESIZABLE: { 391 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 392 mTaskStackListeners.get(i).onActivityForcedResizable( 393 (String) msg.obj, msg.arg1, msg.arg2); 394 } 395 break; 396 } 397 case ON_ACTIVITY_DISMISSING_DOCKED_STACK: { 398 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 399 mTaskStackListeners.get(i).onActivityDismissingDockedStack(); 400 } 401 break; 402 } 403 case ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_FAILED: { 404 final RunningTaskInfo info = (RunningTaskInfo) msg.obj; 405 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 406 mTaskStackListeners.get(i) 407 .onActivityLaunchOnSecondaryDisplayFailed(info); 408 } 409 break; 410 } 411 case ON_ACTIVITY_LAUNCH_ON_SECONDARY_DISPLAY_REROUTED: { 412 final RunningTaskInfo info = (RunningTaskInfo) msg.obj; 413 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 414 mTaskStackListeners.get(i) 415 .onActivityLaunchOnSecondaryDisplayRerouted(info); 416 } 417 break; 418 } 419 case ON_TASK_PROFILE_LOCKED: { 420 final RunningTaskInfo info = (RunningTaskInfo) msg.obj; 421 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 422 mTaskStackListeners.get(i).onTaskProfileLocked(info); 423 } 424 break; 425 } 426 case ON_TASK_CREATED: { 427 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 428 mTaskStackListeners.get(i).onTaskCreated(msg.arg1, 429 (ComponentName) msg.obj); 430 } 431 break; 432 } 433 case ON_TASK_REMOVED: { 434 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 435 mTaskStackListeners.get(i).onTaskRemoved(msg.arg1); 436 } 437 break; 438 } 439 case ON_TASK_MOVED_TO_FRONT: { 440 final RunningTaskInfo info = (RunningTaskInfo) msg.obj; 441 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 442 mTaskStackListeners.get(i).onTaskMovedToFront(info); 443 } 444 break; 445 } 446 case ON_ACTIVITY_REQUESTED_ORIENTATION_CHANGE: { 447 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 448 mTaskStackListeners.get(i) 449 .onActivityRequestedOrientationChanged(msg.arg1, msg.arg2); 450 } 451 break; 452 } 453 case ON_BACK_PRESSED_ON_TASK_ROOT: { 454 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 455 mTaskStackListeners.get(i).onBackPressedOnTaskRoot( 456 (RunningTaskInfo) msg.obj); 457 } 458 break; 459 } 460 case ON_TASK_DISPLAY_CHANGED: { 461 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 462 mTaskStackListeners.get(i).onTaskDisplayChanged(msg.arg1, msg.arg2); 463 } 464 break; 465 } 466 case ON_TASK_LIST_UPDATED: { 467 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 468 mTaskStackListeners.get(i).onRecentTaskListUpdated(); 469 } 470 break; 471 } 472 case ON_TASK_LIST_FROZEN_UNFROZEN: { 473 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 474 mTaskStackListeners.get(i).onRecentTaskListFrozenChanged( 475 msg.arg1 != 0); 476 } 477 break; 478 } 479 case ON_TASK_DESCRIPTION_CHANGED: { 480 final RunningTaskInfo info = (RunningTaskInfo) msg.obj; 481 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 482 mTaskStackListeners.get(i).onTaskDescriptionChanged(info); 483 } 484 break; 485 } 486 case ON_ACTIVITY_ROTATION: { 487 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 488 mTaskStackListeners.get(i).onActivityRotation(msg.arg1); 489 } 490 break; 491 } 492 case ON_LOCK_TASK_MODE_CHANGED: { 493 for (int i = mTaskStackListeners.size() - 1; i >= 0; i--) { 494 mTaskStackListeners.get(i).onLockTaskModeChanged(msg.arg1); 495 } 496 break; 497 } 498 } 499 } 500 if (msg.obj instanceof SomeArgs) { 501 ((SomeArgs) msg.obj).recycle(); 502 } 503 return true; 504 } 505 } 506 507 private static class PinnedActivityInfo { 508 final String mPackageName; 509 final int mUserId; 510 final int mTaskId; 511 final int mStackId; 512 PinnedActivityInfo(String packageName, int userId, int taskId, int stackId)513 PinnedActivityInfo(String packageName, int userId, int taskId, int stackId) { 514 mPackageName = packageName; 515 mUserId = userId; 516 mTaskId = taskId; 517 mStackId = stackId; 518 } 519 } 520 } 521