summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author Jeff DeCew <jeffdq@google.com> 2023-04-14 14:37:55 -0400
committer Jeff DeCew <jeffdq@google.com> 2023-04-24 23:47:29 +0000
commit1000166c2059f9542aa8b406aa824a09a3fb5d1d (patch)
treee653a4f7c9b95498e893316aaf59b30562f68ade
parentb5c0e2843fb53db37099f65e7162a167d83a058e (diff)
Formalize CoreCoordinators
* Adds the CoreCoordinator interface * Stores core coordinators in a separate list * Includes CoreCoordinator details in NotifPipeline dump * Makes DataStoreCoordinator and NotifLiveDataStore PipelineDumpable * Make ListenerSet a Collection so that it can be dumped by PipelineDumper Test: atest SystemUITests Fixes: 208866714 Change-Id: I2e1d290f280514e2ab2b0571bbabc874c56c5197
-rw-r--r--packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/NotifLiveDataStoreImpl.kt15
-rw-r--r--packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/CoreCoordinator.kt24
-rw-r--r--packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/DataStoreCoordinator.kt9
-rw-r--r--packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/NotifCoordinators.kt16
-rw-r--r--packages/SystemUI/src/com/android/systemui/util/ListenerSet.kt35
5 files changed, 66 insertions, 33 deletions
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/NotifLiveDataStoreImpl.kt b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/NotifLiveDataStoreImpl.kt
index 8aa6b81eede0..d95d593778a9 100644
--- a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/NotifLiveDataStoreImpl.kt
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/NotifLiveDataStoreImpl.kt
@@ -32,7 +32,7 @@ import javax.inject.Inject
@SysUISingleton
class NotifLiveDataStoreImpl @Inject constructor(
@Main private val mainExecutor: Executor
-) : NotifLiveDataStore {
+) : NotifLiveDataStore, PipelineDumpable {
private val hasActiveNotifsPrivate = NotifLiveDataImpl(
name = "hasActiveNotifs",
initialValue = false,
@@ -66,6 +66,12 @@ class NotifLiveDataStoreImpl @Inject constructor(
).forEach { dispatcher -> dispatcher.invoke() }
}
}
+
+ override fun dumpPipeline(d: PipelineDumper) {
+ d.dump("activeNotifListPrivate", activeNotifListPrivate)
+ d.dump("activeNotifCountPrivate", activeNotifCountPrivate)
+ d.dump("hasActiveNotifsPrivate", hasActiveNotifsPrivate)
+ }
}
/** Read-write implementation of [NotifLiveData] */
@@ -73,7 +79,7 @@ class NotifLiveDataImpl<T>(
private val name: String,
initialValue: T,
@Main private val mainExecutor: Executor
-) : NotifLiveData<T> {
+) : NotifLiveData<T>, PipelineDumpable {
private val syncObservers = ListenerSet<Observer<T>>()
private val asyncObservers = ListenerSet<Observer<T>>()
private val atomicValue = AtomicReference(initialValue)
@@ -134,4 +140,9 @@ class NotifLiveDataImpl<T>(
syncObservers.remove(observer)
asyncObservers.remove(observer)
}
+
+ override fun dumpPipeline(d: PipelineDumper) {
+ d.dump("syncObservers", syncObservers)
+ d.dump("asyncObservers", asyncObservers)
+ }
} \ No newline at end of file
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/CoreCoordinator.kt b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/CoreCoordinator.kt
new file mode 100644
index 000000000000..75e461c262cd
--- /dev/null
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/CoreCoordinator.kt
@@ -0,0 +1,24 @@
+/*
+ * Copyright (C) 2023 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.statusbar.notification.collection.coordinator
+
+import com.android.systemui.statusbar.notification.collection.PipelineDumpable
+
+/**
+ * A special type of [Coordinator] that is also a core part of the pipeline, and so is also a
+ * [PipelineDumpable].
+ */
+interface CoreCoordinator : Coordinator, PipelineDumpable
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/DataStoreCoordinator.kt b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/DataStoreCoordinator.kt
index 8e307ecd896d..dc8ff63865a2 100644
--- a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/DataStoreCoordinator.kt
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/DataStoreCoordinator.kt
@@ -21,6 +21,7 @@ import com.android.systemui.statusbar.notification.collection.ListEntry
import com.android.systemui.statusbar.notification.collection.NotifLiveDataStoreImpl
import com.android.systemui.statusbar.notification.collection.NotifPipeline
import com.android.systemui.statusbar.notification.collection.NotificationEntry
+import com.android.systemui.statusbar.notification.collection.PipelineDumper
import com.android.systemui.statusbar.notification.collection.coordinator.dagger.CoordinatorScope
import com.android.systemui.statusbar.notification.collection.render.requireSummary
import javax.inject.Inject
@@ -32,13 +33,17 @@ import javax.inject.Inject
@CoordinatorScope
class DataStoreCoordinator @Inject internal constructor(
private val notifLiveDataStoreImpl: NotifLiveDataStoreImpl
-) : Coordinator {
+) : CoreCoordinator {
override fun attach(pipeline: NotifPipeline) {
pipeline.addOnAfterRenderListListener { entries, _ -> onAfterRenderList(entries) }
}
- fun onAfterRenderList(entries: List<ListEntry>) {
+ override fun dumpPipeline(d: PipelineDumper) {
+ d.dump("notifLiveDataStoreImpl", notifLiveDataStoreImpl)
+ }
+
+ private fun onAfterRenderList(entries: List<ListEntry>) {
val flatEntryList = flattenedEntryList(entries)
notifLiveDataStoreImpl.setActiveNotifList(flatEntryList)
}
diff --git a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/NotifCoordinators.kt b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/NotifCoordinators.kt
index 02ce0d46ead8..e5953cfc07cd 100644
--- a/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/NotifCoordinators.kt
+++ b/packages/SystemUI/src/com/android/systemui/statusbar/notification/collection/coordinator/NotifCoordinators.kt
@@ -15,7 +15,6 @@
*/
package com.android.systemui.statusbar.notification.collection.coordinator
-import com.android.systemui.statusbar.notification.NotifPipelineFlags
import com.android.systemui.statusbar.notification.collection.NotifPipeline
import com.android.systemui.statusbar.notification.collection.PipelineDumpable
import com.android.systemui.statusbar.notification.collection.PipelineDumper
@@ -32,7 +31,6 @@ interface NotifCoordinators : Coordinator, PipelineDumpable
@CoordinatorScope
class NotifCoordinatorsImpl @Inject constructor(
- notifPipelineFlags: NotifPipelineFlags,
sectionStyleProvider: SectionStyleProvider,
dataStoreCoordinator: DataStoreCoordinator,
hideLocallyDismissedNotifsCoordinator: HideLocallyDismissedNotifsCoordinator,
@@ -61,6 +59,7 @@ class NotifCoordinatorsImpl @Inject constructor(
dismissibilityCoordinator: DismissibilityCoordinator,
) : NotifCoordinators {
+ private val mCoreCoordinators: MutableList<CoreCoordinator> = ArrayList()
private val mCoordinators: MutableList<Coordinator> = ArrayList()
private val mOrderedSections: MutableList<NotifSectioner> = ArrayList()
@@ -68,11 +67,8 @@ class NotifCoordinatorsImpl @Inject constructor(
* Creates all the coordinators.
*/
init {
- // TODO(b/208866714): formalize the system by which some coordinators may be required by the
- // pipeline, such as this DataStoreCoordinator which cannot be removed, as it's a critical
- // glue between the pipeline and parts of SystemUI which depend on pipeline output via the
- // NotifLiveDataStore.
- mCoordinators.add(dataStoreCoordinator)
+ // Attach core coordinators.
+ mCoreCoordinators.add(dataStoreCoordinator)
// Attach normal coordinators.
mCoordinators.add(hideLocallyDismissedNotifsCoordinator)
@@ -122,6 +118,9 @@ class NotifCoordinatorsImpl @Inject constructor(
* [Pluggable]s, [NotifCollectionListener]s and [NotifLifetimeExtender]s.
*/
override fun attach(pipeline: NotifPipeline) {
+ for (c in mCoreCoordinators) {
+ c.attach(pipeline)
+ }
for (c in mCoordinators) {
c.attach(pipeline)
}
@@ -133,7 +132,8 @@ class NotifCoordinatorsImpl @Inject constructor(
* as they are dumped in the RenderStageManager instead.
*/
override fun dumpPipeline(d: PipelineDumper) = with(d) {
- dump("coordinators", mCoordinators)
+ dump("core coordinators", mCoreCoordinators)
+ dump("normal coordinators", mCoordinators)
}
companion object {
diff --git a/packages/SystemUI/src/com/android/systemui/util/ListenerSet.kt b/packages/SystemUI/src/com/android/systemui/util/ListenerSet.kt
index 4f20067dc0ed..a47e61441c4c 100644
--- a/packages/SystemUI/src/com/android/systemui/util/ListenerSet.kt
+++ b/packages/SystemUI/src/com/android/systemui/util/ListenerSet.kt
@@ -22,34 +22,27 @@ import java.util.concurrent.CopyOnWriteArrayList
* A collection of listeners, observers, callbacks, etc.
*
* This container is optimized for infrequent mutation and frequent iteration, with thread safety
- * and reentrant-safety guarantees as well.
+ * and reentrant-safety guarantees as well. Specifically, to ensure that
+ * [ConcurrentModificationException] is never thrown, this iterator will not reflect changes made to
+ * the set after the iterator is constructed.
*/
-class ListenerSet<E> : Iterable<E> {
- private val listeners: CopyOnWriteArrayList<E> = CopyOnWriteArrayList()
+class ListenerSet<E : Any>
+/** Private constructor takes the internal list so that we can use auto-delegation */
+private constructor(private val listeners: CopyOnWriteArrayList<E>) :
+ Collection<E> by listeners, Set<E> {
+
+ /** Create a new instance */
+ constructor() : this(CopyOnWriteArrayList())
/**
- * A thread-safe, reentrant-safe method to add a listener.
- * Does nothing if the listener is already in the set.
+ * A thread-safe, reentrant-safe method to add a listener. Does nothing if the listener is
+ * already in the set.
*/
fun addIfAbsent(element: E): Boolean = listeners.addIfAbsent(element)
- /**
- * A thread-safe, reentrant-safe method to remove a listener.
- */
+ /** A thread-safe, reentrant-safe method to remove a listener. */
fun remove(element: E): Boolean = listeners.remove(element)
-
- /**
- * Determine if the listener set is empty
- */
- fun isEmpty(): Boolean = listeners.isEmpty()
-
- /**
- * Returns an iterator over the listeners currently in the set. Note that to ensure
- * [ConcurrentModificationException] is never thrown, this iterator will not reflect changes
- * made to the set after the iterator is constructed.
- */
- override fun iterator(): Iterator<E> = listeners.iterator()
}
/** Extension to match Collection which is implemented to only be (easily) accessible in kotlin */
-fun <T> ListenerSet<T>.isNotEmpty(): Boolean = !isEmpty()
+fun <T : Any> ListenerSet<T>.isNotEmpty(): Boolean = !isEmpty()