1 /*
2  * Copyright (C) 2020 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.statusbar.notification.row;
18 
19 import android.os.Handler;
20 import android.os.Looper;
21 import android.os.Message;
22 import android.util.ArrayMap;
23 import android.util.ArraySet;
24 import android.widget.FrameLayout;
25 
26 import androidx.annotation.MainThread;
27 import androidx.annotation.NonNull;
28 import androidx.annotation.Nullable;
29 import androidx.core.os.CancellationSignal;
30 
31 import com.android.systemui.dagger.qualifiers.Main;
32 import com.android.systemui.statusbar.notification.collection.NotificationEntry;
33 import com.android.systemui.statusbar.notification.collection.inflation.NotificationRowBinder;
34 import com.android.systemui.statusbar.notification.collection.notifcollection.CommonNotifCollection;
35 import com.android.systemui.statusbar.notification.collection.notifcollection.NotifCollectionListener;
36 import com.android.systemui.statusbar.notification.row.NotificationRowContentBinder.InflationFlag;
37 
38 import java.util.ArrayList;
39 import java.util.List;
40 import java.util.Map;
41 import java.util.Set;
42 
43 import javax.inject.Inject;
44 import javax.inject.Singleton;
45 
46 /**
47  * {@link NotifBindPipeline} is responsible for converting notifications from their data form to
48  * their actual inflated views. It is essentially a control class that composes notification view
49  * binding logic (i.e. {@link BindStage}) in response to explicit bind requests. At the end of the
50  * pipeline, the notification's bound views are guaranteed to be correct and up-to-date, and any
51  * registered callbacks will be called.
52  *
53  * The pipeline ensures that a notification's top-level view and its content views are bound.
54  * Currently, a notification's top-level view, the {@link ExpandableNotificationRow} is essentially
55  * just a {@link FrameLayout} for various different content views that are switched in and out as
56  * appropriate. These include a contracted view, expanded view, heads up view, and sensitive view on
57  * keyguard. See {@link InflationFlag}. These content views themselves can have child views added
58  * on depending on different factors. For example, notification actions and smart replies are views
59  * that are dynamically added to these content views after they're inflated. Finally, aside from
60  * the app provided content views, System UI itself also provides some content views that are shown
61  * occasionally (e.g. {@link NotificationGuts}). Many of these are business logic specific views
62  * and the requirements surrounding them may change over time, so the pipeline must handle
63  * composing the logic as necessary.
64  *
65  * Note that bind requests do not only occur from add/updates from updates from the app. For
66  * example, the user may make changes to device settings (e.g. sensitive notifications on lock
67  * screen) or we may want to make certain optimizations for the sake of memory or performance (e.g
68  * freeing views when not visible). Oftentimes, we also need to wait for these changes to complete
69  * before doing something else (e.g. moving a notification to the top of the screen to heads up).
70  * The pipeline thus handles bind requests from across the system and provides a way for
71  * requesters to know when the change is propagated to the view.
72  *
73  * Right now, we only support one attached {@link BindStage} which just does all the binding but we
74  * should eventually support multiple stages once content inflation is made more modular.
75  * In particular, row inflation/binding, which is handled by {@link NotificationRowBinder} should
76  * probably be moved here in the future as a stage. Right now, the pipeline just manages content
77  * views and assumes that a row is given to it when it's inflated.
78  */
79 @MainThread
80 @Singleton
81 public final class NotifBindPipeline {
82     private final Map<NotificationEntry, BindEntry> mBindEntries = new ArrayMap<>();
83     private final NotifBindPipelineLogger mLogger;
84     private final List<BindCallback> mScratchCallbacksList = new ArrayList<>();
85     private final Handler mMainHandler;
86     private BindStage mStage;
87 
88     @Inject
NotifBindPipeline( CommonNotifCollection collection, NotifBindPipelineLogger logger, @Main Looper mainLooper)89     NotifBindPipeline(
90             CommonNotifCollection collection,
91             NotifBindPipelineLogger logger,
92             @Main Looper mainLooper) {
93         collection.addCollectionListener(mCollectionListener);
94         mLogger = logger;
95         mMainHandler = new NotifBindPipelineHandler(mainLooper);
96     }
97 
98     /**
99      * Set the bind stage for binding notification row content.
100      */
setStage( BindStage stage)101     public void setStage(
102             BindStage stage) {
103         mLogger.logStageSet(stage.getClass().getName());
104 
105         mStage = stage;
106         mStage.setBindRequestListener(this::onBindRequested);
107     }
108 
109     /**
110      * Start managing the row's content for a given notification.
111      */
manageRow( @onNull NotificationEntry entry, @NonNull ExpandableNotificationRow row)112     public void manageRow(
113             @NonNull NotificationEntry entry,
114             @NonNull ExpandableNotificationRow row) {
115         mLogger.logManagedRow(entry.getKey());
116 
117         final BindEntry bindEntry = getBindEntry(entry);
118         if (bindEntry == null) {
119             return;
120         }
121         bindEntry.row = row;
122         if (bindEntry.invalidated) {
123             requestPipelineRun(entry);
124         }
125     }
126 
onBindRequested( @onNull NotificationEntry entry, @NonNull CancellationSignal signal, @Nullable BindCallback callback)127     private void onBindRequested(
128             @NonNull NotificationEntry entry,
129             @NonNull CancellationSignal signal,
130             @Nullable BindCallback callback) {
131         final BindEntry bindEntry = getBindEntry(entry);
132         if (bindEntry == null) {
133             // Invalidating views for a notification that is not active.
134             return;
135         }
136 
137         bindEntry.invalidated = true;
138 
139         // Put in new callback.
140         if (callback != null) {
141             final Set<BindCallback> callbacks = bindEntry.callbacks;
142             callbacks.add(callback);
143             signal.setOnCancelListener(() -> callbacks.remove(callback));
144         }
145 
146         requestPipelineRun(entry);
147     }
148 
149     /**
150      * Request pipeline to start.
151      *
152      * We avoid starting the pipeline immediately as multiple clients may request rebinds
153      * back-to-back due to a single change (e.g. notification update), and it's better to start
154      * the real work once rather than repeatedly start and cancel it.
155      */
requestPipelineRun(NotificationEntry entry)156     private void requestPipelineRun(NotificationEntry entry) {
157         mLogger.logRequestPipelineRun(entry.getKey());
158 
159         final BindEntry bindEntry = getBindEntry(entry);
160         if (bindEntry.row == null) {
161             // Row is not managed yet but may be soon. Stop for now.
162             mLogger.logRequestPipelineRowNotSet(entry.getKey());
163             return;
164         }
165 
166         // Abort any existing pipeline run
167         mStage.abortStage(entry, bindEntry.row);
168 
169         if (!mMainHandler.hasMessages(START_PIPELINE_MSG, entry)) {
170             Message msg = Message.obtain(mMainHandler, START_PIPELINE_MSG, entry);
171             mMainHandler.sendMessage(msg);
172         }
173     }
174 
175     /**
176      * Run the pipeline for the notification, ensuring all views are bound when finished. Call all
177      * callbacks when the run finishes. If a run is already in progress, it is restarted.
178      */
startPipeline(NotificationEntry entry)179     private void startPipeline(NotificationEntry entry) {
180         mLogger.logStartPipeline(entry.getKey());
181 
182         if (mStage == null) {
183             throw new IllegalStateException("No stage was ever set on the pipeline");
184         }
185 
186         final BindEntry bindEntry = mBindEntries.get(entry);
187         final ExpandableNotificationRow row = bindEntry.row;
188 
189         mStage.executeStage(entry, row, (en) -> onPipelineComplete(en));
190     }
191 
onPipelineComplete(NotificationEntry entry)192     private void onPipelineComplete(NotificationEntry entry) {
193         final BindEntry bindEntry = getBindEntry(entry);
194         final Set<BindCallback> callbacks = bindEntry.callbacks;
195 
196         mLogger.logFinishedPipeline(entry.getKey(), callbacks.size());
197 
198         bindEntry.invalidated = false;
199         // Move all callbacks to separate list as callbacks may themselves add/remove callbacks.
200         // TODO: Throw an exception for this re-entrant behavior once we deprecate
201         // NotificationGroupAlertTransferHelper
202         mScratchCallbacksList.addAll(callbacks);
203         callbacks.clear();
204         for (int i = 0; i < mScratchCallbacksList.size(); i++) {
205             mScratchCallbacksList.get(i).onBindFinished(entry);
206         }
207         mScratchCallbacksList.clear();
208     }
209 
210     private final NotifCollectionListener mCollectionListener = new NotifCollectionListener() {
211         @Override
212         public void onEntryInit(NotificationEntry entry) {
213             mBindEntries.put(entry, new BindEntry());
214             mStage.createStageParams(entry);
215         }
216 
217         @Override
218         public void onEntryCleanUp(NotificationEntry entry) {
219             BindEntry bindEntry = mBindEntries.remove(entry);
220             ExpandableNotificationRow row = bindEntry.row;
221             if (row != null) {
222                 mStage.abortStage(entry, row);
223             }
224             mStage.deleteStageParams(entry);
225             mMainHandler.removeMessages(START_PIPELINE_MSG, entry);
226         }
227     };
228 
getBindEntry(NotificationEntry entry)229     private @NonNull BindEntry getBindEntry(NotificationEntry entry) {
230         final BindEntry bindEntry = mBindEntries.get(entry);
231         return bindEntry;
232     }
233 
234     /**
235      * Interface for bind callback.
236      */
237     public interface BindCallback {
238         /**
239          * Called when all views are fully bound on the notification.
240          */
onBindFinished(NotificationEntry entry)241         void onBindFinished(NotificationEntry entry);
242     }
243 
244     private class BindEntry {
245         public ExpandableNotificationRow row;
246         public final Set<BindCallback> callbacks = new ArraySet<>();
247         public boolean invalidated;
248     }
249 
250     private static final int START_PIPELINE_MSG = 1;
251 
252     private class NotifBindPipelineHandler extends Handler {
253 
NotifBindPipelineHandler(Looper looper)254         NotifBindPipelineHandler(Looper looper) {
255             super(looper);
256         }
257 
258         @Override
handleMessage(Message msg)259         public void handleMessage(Message msg) {
260             switch (msg.what) {
261                 case START_PIPELINE_MSG:
262                     NotificationEntry entry = (NotificationEntry) msg.obj;
263                     startPipeline(entry);
264                     break;
265                 default:
266                     throw new IllegalArgumentException("Unknown message type: " + msg.what);
267             }
268         }
269     }
270 }
271