blob: 33fe04cc8225966088ae72ca3098f3c6d4529a2c [file] [log] [blame]
Mady Mellorc3d6f7d2018-11-07 09:36:56 -08001/*
2 * Copyright (C) 2018 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
17package com.android.systemui.bubbles;
18
Mady Mellord1c78b262018-11-06 18:04:40 -080019import static android.view.View.INVISIBLE;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080020import static android.view.View.VISIBLE;
Joshua Tsujib1a796b2019-01-16 15:43:12 -080021import static android.view.ViewGroup.LayoutParams.MATCH_PARENT;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080022
Mady Mellor3f2efdb2018-11-21 11:30:45 -080023import static com.android.systemui.statusbar.StatusBarState.SHADE;
24import static com.android.systemui.statusbar.notification.NotificationAlertingManager.alertAgain;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080025
Mark Renouf08bc42a2019-03-07 13:01:59 -050026import static java.lang.annotation.RetentionPolicy.SOURCE;
27
Mady Mellorb4991e62019-01-10 15:14:51 -080028import android.annotation.Nullable;
Mady Mellore80930e2019-03-21 16:00:45 -070029import android.app.ActivityManager;
Mark Renoufc808f062019-02-07 15:20:37 -050030import android.app.ActivityManager.RunningTaskInfo;
Mark Renoufcecc77b2019-01-30 16:32:24 -050031import android.app.ActivityTaskManager;
32import android.app.IActivityTaskManager;
Mady Mellorb4991e62019-01-10 15:14:51 -080033import android.app.INotificationManager;
Mady Mellor5549dd22018-11-06 18:07:34 -080034import android.app.Notification;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080035import android.content.Context;
Joshua Tsujia19515f2019-02-13 18:02:29 -050036import android.content.pm.ParceledListSlice;
Mady Mellord1c78b262018-11-06 18:04:40 -080037import android.graphics.Rect;
Mark Renoufcecc77b2019-01-30 16:32:24 -050038import android.os.RemoteException;
Mady Mellorb4991e62019-01-10 15:14:51 -080039import android.os.ServiceManager;
Mady Mellorceced172018-11-27 11:18:39 -080040import android.provider.Settings;
Mady Mellor5549dd22018-11-06 18:07:34 -080041import android.service.notification.StatusBarNotification;
Mark Renoufcecc77b2019-01-30 16:32:24 -050042import android.view.Display;
Joshua Tsujia19515f2019-02-13 18:02:29 -050043import android.view.IPinnedStackController;
44import android.view.IPinnedStackListener;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080045import android.view.ViewGroup;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080046import android.widget.FrameLayout;
47
Mark Renouf08bc42a2019-03-07 13:01:59 -050048import androidx.annotation.IntDef;
Mark Renouf658c6bc2019-01-30 10:26:54 -050049import androidx.annotation.MainThread;
50
Mady Mellorebdbbb92018-11-15 14:36:48 -080051import com.android.internal.annotations.VisibleForTesting;
Mady Mellor3f2efdb2018-11-21 11:30:45 -080052import com.android.internal.statusbar.NotificationVisibility;
Ned Burns01e38212019-01-03 16:32:52 -050053import com.android.systemui.Dependency;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080054import com.android.systemui.R;
Beverly8fdb5332019-02-04 14:29:49 -050055import com.android.systemui.plugins.statusbar.StatusBarStateController;
Mark Renoufcecc77b2019-01-30 16:32:24 -050056import com.android.systemui.shared.system.ActivityManagerWrapper;
57import com.android.systemui.shared.system.TaskStackChangeListener;
Joshua Tsujia19515f2019-02-13 18:02:29 -050058import com.android.systemui.shared.system.WindowManagerWrapper;
Ned Burns01e38212019-01-03 16:32:52 -050059import com.android.systemui.statusbar.notification.NotificationEntryListener;
60import com.android.systemui.statusbar.notification.NotificationEntryManager;
Mady Mellor3f2efdb2018-11-21 11:30:45 -080061import com.android.systemui.statusbar.notification.NotificationInterruptionStateProvider;
Ned Burnsf81c4c42019-01-07 14:10:43 -050062import com.android.systemui.statusbar.notification.collection.NotificationEntry;
Ned Burns1a5e22f2019-02-14 15:11:52 -050063import com.android.systemui.statusbar.notification.row.NotificationContentInflater.InflationFlag;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080064import com.android.systemui.statusbar.phone.StatusBarWindowController;
Lyn Hanf1c9b8b2019-03-14 16:49:48 -070065import com.android.systemui.statusbar.policy.ConfigurationController;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080066
Mark Renouf08bc42a2019-03-07 13:01:59 -050067import java.lang.annotation.Retention;
Mady Mellore80930e2019-03-21 16:00:45 -070068import java.util.List;
Mark Renouf08bc42a2019-03-07 13:01:59 -050069
Jason Monk27d01a622018-12-10 15:57:09 -050070import javax.inject.Inject;
71import javax.inject.Singleton;
72
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080073/**
74 * Bubbles are a special type of content that can "float" on top of other apps or System UI.
75 * Bubbles can be expanded to show more content.
76 *
77 * The controller manages addition, removal, and visible state of bubbles on screen.
78 */
Jason Monk27d01a622018-12-10 15:57:09 -050079@Singleton
Lyn Hanf1c9b8b2019-03-14 16:49:48 -070080public class BubbleController implements BubbleExpandedView.OnBubbleBlockedListener,
81 ConfigurationController.ConfigurationListener {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080082
83 private static final String TAG = "BubbleController";
84
Mark Renouf08bc42a2019-03-07 13:01:59 -050085 @Retention(SOURCE)
86 @IntDef({DISMISS_USER_GESTURE, DISMISS_AGED, DISMISS_TASK_FINISHED, DISMISS_BLOCKED,
87 DISMISS_NOTIF_CANCEL, DISMISS_ACCESSIBILITY_ACTION})
88 @interface DismissReason {}
Lyn Hanf1c9b8b2019-03-14 16:49:48 -070089
Mark Renouf08bc42a2019-03-07 13:01:59 -050090 static final int DISMISS_USER_GESTURE = 1;
91 static final int DISMISS_AGED = 2;
92 static final int DISMISS_TASK_FINISHED = 3;
93 static final int DISMISS_BLOCKED = 4;
94 static final int DISMISS_NOTIF_CANCEL = 5;
95 static final int DISMISS_ACCESSIBILITY_ACTION = 6;
96
Joshua Tsuji25a4b7b2019-03-22 14:11:06 -040097 static final int MAX_BUBBLES = 5; // TODO: actually enforce this
98
Mady Mellor5549dd22018-11-06 18:07:34 -080099 // Enables some subset of notifs to automatically become bubbles
Ned Burns01e38212019-01-03 16:32:52 -0500100 private static final boolean DEBUG_ENABLE_AUTO_BUBBLE = false;
Mady Mellor5549dd22018-11-06 18:07:34 -0800101
Mady Mellor44ee2fe2019-01-30 17:51:16 -0800102 /** Flag to enable or disable the entire feature */
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800103 private static final String ENABLE_BUBBLES = "experiment_enable_bubbles";
Mady Mellor44ee2fe2019-01-30 17:51:16 -0800104 /** Auto bubble flags set whether different notif types should be presented as a bubble */
Mady Mellorceced172018-11-27 11:18:39 -0800105 private static final String ENABLE_AUTO_BUBBLE_MESSAGES = "experiment_autobubble_messaging";
106 private static final String ENABLE_AUTO_BUBBLE_ONGOING = "experiment_autobubble_ongoing";
107 private static final String ENABLE_AUTO_BUBBLE_ALL = "experiment_autobubble_all";
Mady Mellor44ee2fe2019-01-30 17:51:16 -0800108
109 /** Use an activityView for an auto-bubbled notifs if it has an appropriate content intent */
Mark Renouf89b1a4a2018-12-04 14:59:45 -0500110 private static final String ENABLE_BUBBLE_CONTENT_INTENT = "experiment_bubble_content_intent";
Mady Mellorceced172018-11-27 11:18:39 -0800111
Joshua Tsuji010c2b12019-02-25 18:11:25 -0500112 private static final String BUBBLE_STIFFNESS = "experiment_bubble_stiffness";
113 private static final String BUBBLE_BOUNCINESS = "experiment_bubble_bounciness";
114
Ned Burns01e38212019-01-03 16:32:52 -0500115 private final Context mContext;
116 private final NotificationEntryManager mNotificationEntryManager;
Mark Renoufcecc77b2019-01-30 16:32:24 -0500117 private final IActivityTaskManager mActivityTaskManager;
118 private final BubbleTaskStackListener mTaskStackListener;
Mady Mellord1c78b262018-11-06 18:04:40 -0800119 private BubbleStateChangeListener mStateChangeListener;
Mady Mellorcd9b1302018-11-06 18:08:04 -0800120 private BubbleExpandListener mExpandListener;
Issei Suzukic0387542019-03-08 17:31:14 +0100121 @Nullable private BubbleStackView.SurfaceSynchronizer mSurfaceSynchronizer;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800122
Mady Mellor3dff9e62019-02-05 18:12:53 -0800123 private BubbleData mBubbleData;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800124 private BubbleStackView mStackView;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800125
126 // Bubbles get added to the status bar view
Ned Burns01e38212019-01-03 16:32:52 -0500127 private final StatusBarWindowController mStatusBarWindowController;
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800128 private StatusBarStateListener mStatusBarStateListener;
129
130 private final NotificationInterruptionStateProvider mNotificationInterruptionStateProvider =
131 Dependency.get(NotificationInterruptionStateProvider.class);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800132
Mady Mellorb4991e62019-01-10 15:14:51 -0800133 private INotificationManager mNotificationManagerService;
134
Mady Mellord1c78b262018-11-06 18:04:40 -0800135 // Used for determining view rect for touch interaction
136 private Rect mTempRect = new Rect();
137
Mady Mellor5549dd22018-11-06 18:07:34 -0800138 /**
Mady Mellord1c78b262018-11-06 18:04:40 -0800139 * Listener to be notified when some states of the bubbles change.
140 */
141 public interface BubbleStateChangeListener {
142 /**
143 * Called when the stack has bubbles or no longer has bubbles.
144 */
145 void onHasBubblesChanged(boolean hasBubbles);
146 }
147
Mady Mellorcd9b1302018-11-06 18:08:04 -0800148 /**
149 * Listener to find out about stack expansion / collapse events.
150 */
151 public interface BubbleExpandListener {
152 /**
153 * Called when the expansion state of the bubble stack changes.
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700154 *
Mady Mellorcd9b1302018-11-06 18:08:04 -0800155 * @param isExpanding whether it's expanding or collapsing
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800156 * @param key the notification key associated with bubble being expanded
Mady Mellorcd9b1302018-11-06 18:08:04 -0800157 */
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800158 void onBubbleExpandChanged(boolean isExpanding, String key);
159 }
160
161 /**
162 * Listens for the current state of the status bar and updates the visibility state
163 * of bubbles as needed.
164 */
165 private class StatusBarStateListener implements StatusBarStateController.StateListener {
166 private int mState;
167 /**
168 * Returns the current status bar state.
169 */
170 public int getCurrentState() {
171 return mState;
172 }
173
174 @Override
175 public void onStateChanged(int newState) {
176 mState = newState;
177 updateVisibility();
178 }
Mady Mellorcd9b1302018-11-06 18:08:04 -0800179 }
180
Jason Monk27d01a622018-12-10 15:57:09 -0500181 @Inject
Mady Mellorcfd06c12019-02-13 14:32:12 -0800182 public BubbleController(Context context, StatusBarWindowController statusBarWindowController,
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700183 BubbleData data, ConfigurationController configurationController) {
184 this(context, statusBarWindowController, data, null /* synchronizer */,
185 configurationController);
Issei Suzukic0387542019-03-08 17:31:14 +0100186 }
187
188 public BubbleController(Context context, StatusBarWindowController statusBarWindowController,
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700189 BubbleData data, @Nullable BubbleStackView.SurfaceSynchronizer synchronizer,
190 ConfigurationController configurationController) {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800191 mContext = context;
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700192 configurationController.addCallback(this /* configurationListener */);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800193
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800194 mNotificationEntryManager = Dependency.get(NotificationEntryManager.class);
Ned Burns01e38212019-01-03 16:32:52 -0500195 mNotificationEntryManager.addNotificationEntryListener(mEntryListener);
Mady Mellorb4991e62019-01-10 15:14:51 -0800196
197 try {
198 mNotificationManagerService = INotificationManager.Stub.asInterface(
199 ServiceManager.getServiceOrThrow(Context.NOTIFICATION_SERVICE));
200 } catch (ServiceManager.ServiceNotFoundException e) {
201 e.printStackTrace();
202 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800203
204 mStatusBarWindowController = statusBarWindowController;
205 mStatusBarStateListener = new StatusBarStateListener();
206 Dependency.get(StatusBarStateController.class).addCallback(mStatusBarStateListener);
Mark Renoufcecc77b2019-01-30 16:32:24 -0500207
208 mActivityTaskManager = ActivityTaskManager.getService();
209 mTaskStackListener = new BubbleTaskStackListener();
210 ActivityManagerWrapper.getInstance().registerTaskStackListener(mTaskStackListener);
Mady Mellor3dff9e62019-02-05 18:12:53 -0800211
Joshua Tsujia19515f2019-02-13 18:02:29 -0500212 try {
213 WindowManagerWrapper.getInstance().addPinnedStackListener(new BubblesImeListener());
214 } catch (RemoteException e) {
215 e.printStackTrace();
216 }
217
Mady Mellorcfd06c12019-02-13 14:32:12 -0800218 mBubbleData = data;
Issei Suzukic0387542019-03-08 17:31:14 +0100219 mSurfaceSynchronizer = synchronizer;
Mady Mellor5549dd22018-11-06 18:07:34 -0800220 }
221
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700222 @Override
223 public void onUiModeChanged() {
224 if (mStackView != null) {
225 mStackView.onConfigChanged();
226 }
227 }
228
229 @Override
230 public void onOverlayChanged() {
231 if (mStackView != null) {
232 mStackView.onConfigChanged();
233 }
234 }
235
Mady Mellor5549dd22018-11-06 18:07:34 -0800236 /**
Mady Mellord1c78b262018-11-06 18:04:40 -0800237 * Set a listener to be notified when some states of the bubbles change.
238 */
239 public void setBubbleStateChangeListener(BubbleStateChangeListener listener) {
240 mStateChangeListener = listener;
241 }
242
243 /**
Mady Mellorcd9b1302018-11-06 18:08:04 -0800244 * Set a listener to be notified of bubble expand events.
245 */
246 public void setExpandListener(BubbleExpandListener listener) {
Issei Suzukiac9fcb72019-02-04 17:45:57 +0100247 mExpandListener = ((isExpanding, key) -> {
248 if (listener != null) {
249 listener.onBubbleExpandChanged(isExpanding, key);
250 }
251 mStatusBarWindowController.setBubbleExpanded(isExpanding);
252 });
Mady Mellorcd9b1302018-11-06 18:08:04 -0800253 if (mStackView != null) {
254 mStackView.setExpandListener(mExpandListener);
255 }
256 }
257
258 /**
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800259 * Whether or not there are bubbles present, regardless of them being visible on the
260 * screen (e.g. if on AOD).
261 */
262 public boolean hasBubbles() {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800263 if (mStackView == null) {
264 return false;
265 }
266 for (Bubble bubble : mBubbleData.getBubbles()) {
267 if (!bubble.entry.isBubbleDismissed()) {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800268 return true;
269 }
270 }
271 return false;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800272 }
273
274 /**
275 * Whether the stack of bubbles is expanded or not.
276 */
277 public boolean isStackExpanded() {
278 return mStackView != null && mStackView.isExpanded();
279 }
280
281 /**
282 * Tell the stack of bubbles to collapse.
283 */
284 public void collapseStack() {
285 if (mStackView != null) {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800286 mStackView.collapseStack();
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800287 }
288 }
289
290 /**
Mark Renouffec45da2019-03-13 13:24:27 -0400291 * Request the stack expand if needed, then select the specified Bubble as current.
292 *
293 * @param notificationKey the notification key for the bubble to be selected
294 */
295 public void expandStackAndSelectBubble(String notificationKey) {
296 if (mStackView != null && mBubbleData.getBubble(notificationKey) != null) {
297 mStackView.setExpandedBubble(notificationKey);
298 }
299 }
300
301 /**
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800302 * Tell the stack of bubbles to be dismissed, this will remove all of the bubbles in the stack.
303 */
Mark Renouf08bc42a2019-03-07 13:01:59 -0500304 void dismissStack(@DismissReason int reason) {
Mady Mellord1c78b262018-11-06 18:04:40 -0800305 if (mStackView == null) {
306 return;
307 }
Mark Renouf08bc42a2019-03-07 13:01:59 -0500308 mStackView.stackDismissed(reason);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800309
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800310 updateVisibility();
Ned Burns01e38212019-01-03 16:32:52 -0500311 mNotificationEntryManager.updateNotifications();
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800312 }
313
314 /**
Mark Renouf041d7262019-02-06 12:09:41 -0500315 * Directs a back gesture at the bubble stack. When opened, the current expanded bubble
316 * is forwarded a back key down/up pair.
317 */
318 public void performBackPressIfNeeded() {
319 if (mStackView != null) {
320 mStackView.performBackPressIfNeeded();
321 }
322 }
323
324 /**
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800325 * Adds or updates a bubble associated with the provided notification entry.
326 *
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700327 * @param notif the notification associated with this bubble.
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800328 */
Mark Renouff97ed462019-04-05 13:46:24 -0400329 void updateBubble(NotificationEntry notif) {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800330 if (mStackView != null && mBubbleData.getBubble(notif.key) != null) {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800331 // It's an update
Mark Renouff97ed462019-04-05 13:46:24 -0400332 mStackView.updateBubble(notif);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800333 } else {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800334 if (mStackView == null) {
Issei Suzukic0387542019-03-08 17:31:14 +0100335 mStackView = new BubbleStackView(mContext, mBubbleData, mSurfaceSynchronizer);
Mady Mellord1c78b262018-11-06 18:04:40 -0800336 ViewGroup sbv = mStatusBarWindowController.getStatusBarView();
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800337 // XXX: Bug when you expand the shade on top of expanded bubble, there is no scrim
338 // between bubble and the shade
339 int bubblePosition = sbv.indexOfChild(sbv.findViewById(R.id.scrim_behind)) + 1;
340 sbv.addView(mStackView, bubblePosition,
Joshua Tsujib1a796b2019-01-16 15:43:12 -0800341 new FrameLayout.LayoutParams(MATCH_PARENT, MATCH_PARENT));
Mady Mellorcd9b1302018-11-06 18:08:04 -0800342 if (mExpandListener != null) {
343 mStackView.setExpandListener(mExpandListener);
344 }
Mady Mellore8e07712019-01-23 12:45:33 -0800345 mStackView.setOnBlockedListener(this);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800346 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800347 // It's new
Mady Mellor3dff9e62019-02-05 18:12:53 -0800348 mStackView.addBubble(notif);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800349 }
Mady Mellore80930e2019-03-21 16:00:45 -0700350 if (shouldAutoExpand(notif)) {
351 mStackView.setExpandedBubble(notif);
352 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800353 updateVisibility();
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800354 }
355
356 /**
357 * Removes the bubble associated with the {@param uri}.
Mark Renouf658c6bc2019-01-30 10:26:54 -0500358 * <p>
359 * Must be called from the main thread.
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800360 */
Mark Renouf658c6bc2019-01-30 10:26:54 -0500361 @MainThread
Mark Renouf08bc42a2019-03-07 13:01:59 -0500362 void removeBubble(String key, int reason) {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800363 if (mStackView != null) {
Mark Renouf08bc42a2019-03-07 13:01:59 -0500364 mStackView.removeBubble(key, reason);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800365 }
Mady Mellor3dff9e62019-02-05 18:12:53 -0800366 mNotificationEntryManager.updateNotifications();
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800367 updateVisibility();
Mady Mellord1c78b262018-11-06 18:04:40 -0800368 }
369
Mady Mellore8e07712019-01-23 12:45:33 -0800370 @Override
371 public void onBubbleBlocked(NotificationEntry entry) {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800372 Object[] bubbles = mBubbleData.getBubbles().toArray();
Mady Mellore8e07712019-01-23 12:45:33 -0800373 for (int i = 0; i < bubbles.length; i++) {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800374 NotificationEntry e = ((Bubble) bubbles[i]).entry;
Mady Mellore8e07712019-01-23 12:45:33 -0800375 boolean samePackage = entry.notification.getPackageName().equals(
376 e.notification.getPackageName());
377 if (samePackage) {
Mark Renouf08bc42a2019-03-07 13:01:59 -0500378 removeBubble(entry.key, DISMISS_BLOCKED);
Mady Mellore8e07712019-01-23 12:45:33 -0800379 }
380 }
381 }
382
Ned Burns01e38212019-01-03 16:32:52 -0500383 @SuppressWarnings("FieldCanBeLocal")
384 private final NotificationEntryListener mEntryListener = new NotificationEntryListener() {
385 @Override
Ned Burnsf81c4c42019-01-07 14:10:43 -0500386 public void onPendingEntryAdded(NotificationEntry entry) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800387 if (!areBubblesEnabled(mContext)) {
388 return;
389 }
390 if (shouldAutoBubbleForFlags(mContext, entry) || shouldBubble(entry)) {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800391 // TODO: handle group summaries
Mady Mellor3ed46202019-03-26 20:22:35 -0700392 boolean suppressNotification = entry.getBubbleMetadata() != null
393 && entry.getBubbleMetadata().getSuppressInitialNotification()
394 && isForegroundApp(entry.notification.getPackageName());
395 entry.setShowInShadeWhenBubble(!suppressNotification);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800396 }
397 }
398
399 @Override
Ned Burns1a5e22f2019-02-14 15:11:52 -0500400 public void onEntryInflated(NotificationEntry entry, @InflationFlag int inflatedFlags) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800401 if (!areBubblesEnabled(mContext)) {
402 return;
403 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800404 if (entry.isBubble() && mNotificationInterruptionStateProvider.shouldBubbleUp(entry)) {
Mark Renouff97ed462019-04-05 13:46:24 -0400405 updateBubble(entry);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800406 }
407 }
408
409 @Override
410 public void onPreEntryUpdated(NotificationEntry entry) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800411 if (!areBubblesEnabled(mContext)) {
412 return;
413 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800414 if (mNotificationInterruptionStateProvider.shouldBubbleUp(entry)
415 && alertAgain(entry, entry.notification.getNotification())) {
416 entry.setShowInShadeWhenBubble(true);
417 entry.setBubbleDismissed(false); // updates come back as bubbles even if dismissed
Mark Renouff97ed462019-04-05 13:46:24 -0400418 updateBubble(entry);
Mady Mellor3dff9e62019-02-05 18:12:53 -0800419 mStackView.updateDotVisibility(entry.key);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800420 }
421 }
422
423 @Override
424 public void onEntryRemoved(NotificationEntry entry,
425 @Nullable NotificationVisibility visibility,
426 boolean removedByUser) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800427 if (!areBubblesEnabled(mContext)) {
428 return;
429 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800430 entry.setShowInShadeWhenBubble(false);
Mady Mellor3dff9e62019-02-05 18:12:53 -0800431 if (mStackView != null) {
432 mStackView.updateDotVisibility(entry.key);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800433 }
434 if (!removedByUser) {
435 // This was a cancel so we should remove the bubble
Mark Renouf08bc42a2019-03-07 13:01:59 -0500436 removeBubble(entry.key, DISMISS_NOTIF_CANCEL);
Ned Burns01e38212019-01-03 16:32:52 -0500437 }
438 }
439 };
440
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800441 /**
442 * Lets any listeners know if bubble state has changed.
443 */
Mady Mellord1c78b262018-11-06 18:04:40 -0800444 private void updateBubblesShowing() {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800445 if (mStackView == null) {
446 return;
Mady Mellord1c78b262018-11-06 18:04:40 -0800447 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800448
Mady Mellord1c78b262018-11-06 18:04:40 -0800449 boolean hadBubbles = mStatusBarWindowController.getBubblesShowing();
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800450 boolean hasBubblesShowing = hasBubbles() && mStackView.getVisibility() == VISIBLE;
Mady Mellord1c78b262018-11-06 18:04:40 -0800451 mStatusBarWindowController.setBubblesShowing(hasBubblesShowing);
Mady Mellord1c78b262018-11-06 18:04:40 -0800452 if (mStateChangeListener != null && hadBubbles != hasBubblesShowing) {
453 mStateChangeListener.onHasBubblesChanged(hasBubblesShowing);
454 }
Mady Mellor5549dd22018-11-06 18:07:34 -0800455 }
456
457 /**
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800458 * Updates the visibility of the bubbles based on current state.
459 * Does not un-bubble, just hides or un-hides. Will notify any
460 * {@link BubbleStateChangeListener}s if visibility changes.
Mady Mellor5549dd22018-11-06 18:07:34 -0800461 */
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800462 public void updateVisibility() {
463 if (mStatusBarStateListener.getCurrentState() == SHADE && hasBubbles()) {
464 // Bubbles only appear in unlocked shade
465 mStackView.setVisibility(hasBubbles() ? VISIBLE : INVISIBLE);
466 } else if (mStackView != null) {
467 mStackView.setVisibility(INVISIBLE);
468 collapseStack();
Mady Mellor5549dd22018-11-06 18:07:34 -0800469 }
Mady Mellord1c78b262018-11-06 18:04:40 -0800470 updateBubblesShowing();
471 }
472
473 /**
474 * Rect indicating the touchable region for the bubble stack / expanded stack.
475 */
476 public Rect getTouchableRegion() {
477 if (mStackView == null || mStackView.getVisibility() != VISIBLE) {
478 return null;
479 }
480 mStackView.getBoundsOnScreen(mTempRect);
481 return mTempRect;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800482 }
483
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800484 @VisibleForTesting
485 BubbleStackView getStackView() {
486 return mStackView;
487 }
488
Mady Mellor5549dd22018-11-06 18:07:34 -0800489 /**
Mady Mellorb4991e62019-01-10 15:14:51 -0800490 * Whether the notification has been developer configured to bubble and is allowed by the user.
491 */
Mady Mellorc18ba962019-01-29 11:11:56 -0800492 @VisibleForTesting
493 protected boolean shouldBubble(NotificationEntry entry) {
Mady Mellorb4991e62019-01-10 15:14:51 -0800494 StatusBarNotification n = entry.notification;
Mady Mellorc39b4ae2019-01-09 17:11:37 -0800495 boolean hasOverlayIntent = n.getNotification().getBubbleMetadata() != null
496 && n.getNotification().getBubbleMetadata().getIntent() != null;
Julia Reynolds4509ce72019-01-31 13:12:43 -0500497 return hasOverlayIntent && entry.canBubble;
Mady Mellorb4991e62019-01-10 15:14:51 -0800498 }
499
500 /**
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800501 * Whether the notification should automatically bubble or not. Gated by secure settings flags.
Mady Mellor5549dd22018-11-06 18:07:34 -0800502 */
Mady Mellor9bad2242019-01-28 11:21:51 -0800503 @VisibleForTesting
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800504 protected boolean shouldAutoBubbleForFlags(Context context, NotificationEntry entry) {
Mady Mellorceced172018-11-27 11:18:39 -0800505 if (entry.isBubbleDismissed()) {
Mady Mellor5549dd22018-11-06 18:07:34 -0800506 return false;
507 }
Mady Mellorb4991e62019-01-10 15:14:51 -0800508 StatusBarNotification n = entry.notification;
Mady Mellorceced172018-11-27 11:18:39 -0800509
510 boolean autoBubbleMessages = shouldAutoBubbleMessages(context) || DEBUG_ENABLE_AUTO_BUBBLE;
511 boolean autoBubbleOngoing = shouldAutoBubbleOngoing(context) || DEBUG_ENABLE_AUTO_BUBBLE;
512 boolean autoBubbleAll = shouldAutoBubbleAll(context) || DEBUG_ENABLE_AUTO_BUBBLE;
513
Mady Mellor5549dd22018-11-06 18:07:34 -0800514 boolean hasRemoteInput = false;
515 if (n.getNotification().actions != null) {
516 for (Notification.Action action : n.getNotification().actions) {
517 if (action.getRemoteInputs() != null) {
518 hasRemoteInput = true;
519 break;
520 }
521 }
522 }
Mady Mellor711f9562018-12-05 14:53:46 -0800523 boolean isCall = Notification.CATEGORY_CALL.equals(n.getNotification().category)
524 && n.isOngoing();
525 boolean isMusic = n.getNotification().hasMediaSession();
526 boolean isImportantOngoing = isMusic || isCall;
Mady Mellorceced172018-11-27 11:18:39 -0800527
Mady Mellor5549dd22018-11-06 18:07:34 -0800528 Class<? extends Notification.Style> style = n.getNotification().getNotificationStyle();
Mady Mellore3175372018-12-04 17:05:11 -0800529 boolean isMessageType = Notification.CATEGORY_MESSAGE.equals(n.getNotification().category);
530 boolean isMessageStyle = Notification.MessagingStyle.class.equals(style);
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800531 return (((isMessageType && hasRemoteInput) || isMessageStyle) && autoBubbleMessages)
Mady Mellor711f9562018-12-05 14:53:46 -0800532 || (isImportantOngoing && autoBubbleOngoing)
Mady Mellorceced172018-11-27 11:18:39 -0800533 || autoBubbleAll;
534 }
535
Mady Mellore80930e2019-03-21 16:00:45 -0700536 private boolean shouldAutoExpand(NotificationEntry entry) {
537 Notification.BubbleMetadata metadata = entry.getBubbleMetadata();
538 return metadata != null && metadata.getAutoExpandBubble()
539 && isForegroundApp(entry.notification.getPackageName());
540 }
541
542 /**
543 * Return true if the applications with the package name is running in foreground.
544 *
545 * @param pkgName application package name.
546 */
547 private boolean isForegroundApp(String pkgName) {
548 ActivityManager am = mContext.getSystemService(ActivityManager.class);
549 List<RunningTaskInfo> tasks = am.getRunningTasks(1 /* maxNum */);
550 return !tasks.isEmpty() && pkgName.equals(tasks.get(0).topActivity.getPackageName());
551 }
552
Mark Renoufcecc77b2019-01-30 16:32:24 -0500553 /**
554 * This task stack listener is responsible for responding to tasks moved to the front
555 * which are on the default (main) display. When this happens, expanded bubbles must be
556 * collapsed so the user may interact with the app which was just moved to the front.
557 * <p>
558 * This listener is registered with SystemUI's ActivityManagerWrapper which dispatches
559 * these calls via a main thread Handler.
560 */
561 @MainThread
562 private class BubbleTaskStackListener extends TaskStackChangeListener {
563
Mark Renoufcecc77b2019-01-30 16:32:24 -0500564 @Override
Mark Renoufc808f062019-02-07 15:20:37 -0500565 public void onTaskMovedToFront(RunningTaskInfo taskInfo) {
566 if (mStackView != null && taskInfo.displayId == Display.DEFAULT_DISPLAY) {
Mark Renoufcecc77b2019-01-30 16:32:24 -0500567 mStackView.collapseStack();
568 }
569 }
570
Mark Renoufcecc77b2019-01-30 16:32:24 -0500571 @Override
Mark Renoufa12e8762019-03-07 15:43:01 -0500572 public void onActivityLaunchOnSecondaryDisplayRerouted() {
Mark Renoufcecc77b2019-01-30 16:32:24 -0500573 if (mStackView != null) {
574 mStackView.collapseStack();
575 }
576 }
577 }
578
Mady Mellorceced172018-11-27 11:18:39 -0800579 private static boolean shouldAutoBubbleMessages(Context context) {
580 return Settings.Secure.getInt(context.getContentResolver(),
581 ENABLE_AUTO_BUBBLE_MESSAGES, 0) != 0;
582 }
583
584 private static boolean shouldAutoBubbleOngoing(Context context) {
585 return Settings.Secure.getInt(context.getContentResolver(),
586 ENABLE_AUTO_BUBBLE_ONGOING, 0) != 0;
587 }
588
589 private static boolean shouldAutoBubbleAll(Context context) {
590 return Settings.Secure.getInt(context.getContentResolver(),
591 ENABLE_AUTO_BUBBLE_ALL, 0) != 0;
Mady Mellor5549dd22018-11-06 18:07:34 -0800592 }
Mark Renouf89b1a4a2018-12-04 14:59:45 -0500593
Mady Mellor3dff9e62019-02-05 18:12:53 -0800594 static boolean shouldUseContentIntent(Context context) {
Mark Renouf89b1a4a2018-12-04 14:59:45 -0500595 return Settings.Secure.getInt(context.getContentResolver(),
596 ENABLE_BUBBLE_CONTENT_INTENT, 0) != 0;
597 }
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800598
599 private static boolean areBubblesEnabled(Context context) {
600 return Settings.Secure.getInt(context.getContentResolver(),
601 ENABLE_BUBBLES, 1) != 0;
602 }
Joshua Tsujia19515f2019-02-13 18:02:29 -0500603
Joshua Tsuji010c2b12019-02-25 18:11:25 -0500604 /** Default stiffness to use for bubble physics animations. */
605 public static int getBubbleStiffness(Context context, int defaultStiffness) {
606 return Settings.Secure.getInt(
607 context.getContentResolver(), BUBBLE_STIFFNESS, defaultStiffness);
608 }
609
610 /** Default bounciness/damping ratio to use for bubble physics animations. */
611 public static float getBubbleBounciness(Context context, float defaultBounciness) {
612 return Settings.Secure.getInt(
613 context.getContentResolver(),
614 BUBBLE_BOUNCINESS,
615 (int) (defaultBounciness * 100)) / 100f;
616 }
617
Joshua Tsujia19515f2019-02-13 18:02:29 -0500618 /** PinnedStackListener that dispatches IME visibility updates to the stack. */
619 private class BubblesImeListener extends IPinnedStackListener.Stub {
620
621 @Override
622 public void onListenerRegistered(IPinnedStackController controller) throws RemoteException {
623 }
624
625 @Override
626 public void onMovementBoundsChanged(Rect insetBounds, Rect normalBounds,
627 Rect animatingBounds, boolean fromImeAdjustment, boolean fromShelfAdjustment,
628 int displayRotation) throws RemoteException {}
629
630 @Override
Joshua Tsujid9422832019-03-05 13:32:37 -0500631 public void onImeVisibilityChanged(boolean imeVisible, int imeHeight) {
632 if (mStackView != null && mStackView.getBubbleCount() > 0) {
633 mStackView.post(() -> mStackView.onImeVisibilityChanged(imeVisible, imeHeight));
Joshua Tsujia19515f2019-02-13 18:02:29 -0500634 }
635 }
636
637 @Override
638 public void onShelfVisibilityChanged(boolean shelfVisible, int shelfHeight)
639 throws RemoteException {}
640
641 @Override
642 public void onMinimizedStateChanged(boolean isMinimized) throws RemoteException {}
643
644 @Override
645 public void onActionsChanged(ParceledListSlice actions) throws RemoteException {}
646 }
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800647}