|
| 1 | +package me.ycdev.android.lib.common.activity |
| 2 | + |
| 3 | +import android.app.Activity |
| 4 | +import android.app.Application |
| 5 | +import android.os.Bundle |
| 6 | +import androidx.annotation.GuardedBy |
| 7 | +import androidx.annotation.VisibleForTesting |
| 8 | +import timber.log.Timber |
| 9 | + |
| 10 | +object ActivityTaskTracker { |
| 11 | + private const val TAG = "ActivityTaskTracker" |
| 12 | + |
| 13 | + internal val lifecycleCallback = MyLifecycleCallback() |
| 14 | + |
| 15 | + private val allTasks: HashMap<Int, ActivityTask> = hashMapOf() |
| 16 | + private var focusedTaskId: Int = -1 |
| 17 | + private var resumedActivity: Activity? = null |
| 18 | + |
| 19 | + private var debugLog: Boolean = false |
| 20 | + |
| 21 | + fun enableDebugLog(enable: Boolean) { |
| 22 | + debugLog = enable |
| 23 | + } |
| 24 | + |
| 25 | + fun init(app: Application) { |
| 26 | + app.registerActivityLifecycleCallbacks(lifecycleCallback) |
| 27 | + } |
| 28 | + |
| 29 | + fun getFocusedTask(): ActivityTask? { |
| 30 | + synchronized(allTasks) { |
| 31 | + if (focusedTaskId != -1) { |
| 32 | + return allTasks[focusedTaskId]?.makeCopy() |
| 33 | + } |
| 34 | + return null |
| 35 | + } |
| 36 | + } |
| 37 | + |
| 38 | + /** |
| 39 | + * Return all tasks. The focused task will be the first element in returned list. |
| 40 | + */ |
| 41 | + fun getAllTasks(): List<ActivityTask> { |
| 42 | + synchronized(allTasks) { |
| 43 | + val result = ArrayList<ActivityTask>(allTasks.size) |
| 44 | + // always put the focused task at index 0 |
| 45 | + val focusedTask = getFocusedTask() |
| 46 | + if (focusedTask != null) { |
| 47 | + result.add(focusedTask) |
| 48 | + } |
| 49 | + allTasks.values.forEach { |
| 50 | + if (it.taskId != focusedTaskId) { |
| 51 | + result.add(it.makeCopy()) |
| 52 | + } |
| 53 | + } |
| 54 | + return result |
| 55 | + } |
| 56 | + } |
| 57 | + |
| 58 | + @GuardedBy("allTasks") |
| 59 | + private fun getOrCreateTaskLocked(taskId: Int): ActivityTask { |
| 60 | + var task = allTasks[taskId] |
| 61 | + if (task == null) { |
| 62 | + task = ActivityTask(taskId) |
| 63 | + allTasks[taskId] = task |
| 64 | + } |
| 65 | + return task |
| 66 | + } |
| 67 | + |
| 68 | + private fun updateLastActivityState(activity: Activity, state: ActivityInfo.State) { |
| 69 | + synchronized(allTasks) { |
| 70 | + val task = getOrCreateTaskLocked(activity.taskId) |
| 71 | + val appActivity = task.lastActivity(activity.componentName) |
| 72 | + appActivity.state = state |
| 73 | + } |
| 74 | + } |
| 75 | + |
| 76 | + @VisibleForTesting |
| 77 | + internal class MyLifecycleCallback : Application.ActivityLifecycleCallbacks { |
| 78 | + override fun onActivityCreated(activity: Activity, bundle: Bundle?) { |
| 79 | + if (debugLog) Timber.tag(TAG).d("onCreate: %s", activity.componentName) |
| 80 | + synchronized(allTasks) { |
| 81 | + val appActivity = ActivityInfo(activity.componentName, activity.taskId) |
| 82 | + appActivity.state = ActivityInfo.State.Created |
| 83 | + val task = getOrCreateTaskLocked(activity.taskId) |
| 84 | + task.addActivity(appActivity) |
| 85 | + } |
| 86 | + } |
| 87 | + |
| 88 | + override fun onActivityStarted(activity: Activity) { |
| 89 | + if (debugLog) Timber.tag(TAG).d("onStarted: %s", activity.componentName) |
| 90 | + updateLastActivityState(activity, ActivityInfo.State.Started) |
| 91 | + } |
| 92 | + |
| 93 | + override fun onActivityResumed(activity: Activity) { |
| 94 | + if (debugLog) Timber.tag(TAG).d("onResumed: %s", activity.componentName) |
| 95 | + updateLastActivityState(activity, ActivityInfo.State.Resumed) |
| 96 | + focusedTaskId = activity.taskId |
| 97 | + resumedActivity = activity |
| 98 | + } |
| 99 | + |
| 100 | + override fun onActivityPaused(activity: Activity) { |
| 101 | + if (debugLog) Timber.tag(TAG).d("onPaused: %s", activity.componentName) |
| 102 | + updateLastActivityState(activity, ActivityInfo.State.Paused) |
| 103 | + if (activity == resumedActivity) { |
| 104 | + focusedTaskId = -1 |
| 105 | + resumedActivity = null |
| 106 | + } |
| 107 | + } |
| 108 | + |
| 109 | + override fun onActivitySaveInstanceState(activity: Activity, bundle: Bundle) { |
| 110 | + if (debugLog) Timber.tag(TAG).d("onSaveState: %s", activity.componentName) |
| 111 | + } |
| 112 | + |
| 113 | + override fun onActivityStopped(activity: Activity) { |
| 114 | + if (debugLog) Timber.tag(TAG).d("onStopped: %s", activity.componentName) |
| 115 | + updateLastActivityState(activity, ActivityInfo.State.Stopped) |
| 116 | + } |
| 117 | + |
| 118 | + override fun onActivityDestroyed(activity: Activity) { |
| 119 | + if (debugLog) Timber.tag(TAG).d("onDestroyed: %s", activity.componentName) |
| 120 | + synchronized(allTasks) { |
| 121 | + val task = getOrCreateTaskLocked(activity.taskId) |
| 122 | + task.popActivity(activity.componentName) |
| 123 | + if (task.isEmpty()) { |
| 124 | + allTasks.remove(task.taskId) |
| 125 | + } |
| 126 | + } |
| 127 | + } |
| 128 | + } |
| 129 | +} |
0 commit comments