blob: 5559d29a905065b1b72770d4d4bea2ace4df56cf [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 Mellor22f2f072019-04-18 13:26:18 -070019import static android.app.Notification.FLAG_AUTOGROUP_SUMMARY;
Mady Mellor3a0a1b42019-05-23 06:40:21 -070020import static android.app.Notification.FLAG_BUBBLE;
Mady Mellorca0c24c2019-05-16 16:14:32 -070021import static android.content.pm.ActivityInfo.DOCUMENT_LAUNCH_ALWAYS;
Mady Mellorc2ff0112019-03-28 14:18:06 -070022import static android.service.notification.NotificationListenerService.REASON_APP_CANCEL;
23import static android.service.notification.NotificationListenerService.REASON_APP_CANCEL_ALL;
24import static android.service.notification.NotificationListenerService.REASON_CANCEL;
25import static android.service.notification.NotificationListenerService.REASON_CANCEL_ALL;
Mady Mellor22f2f072019-04-18 13:26:18 -070026import static android.service.notification.NotificationListenerService.REASON_CLICK;
27import static android.service.notification.NotificationListenerService.REASON_GROUP_SUMMARY_CANCELED;
Mady Mellor390bff42019-04-05 15:09:01 -070028import static android.view.Display.DEFAULT_DISPLAY;
29import static android.view.Display.INVALID_DISPLAY;
Mady Mellord1c78b262018-11-06 18:04:40 -080030import static android.view.View.INVISIBLE;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080031import static android.view.View.VISIBLE;
Joshua Tsujib1a796b2019-01-16 15:43:12 -080032import static android.view.ViewGroup.LayoutParams.MATCH_PARENT;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080033
Issei Suzukia8d07312019-06-07 12:56:19 +020034import static com.android.systemui.bubbles.BubbleDebugConfig.DEBUG_BUBBLE_CONTROLLER;
35import static com.android.systemui.bubbles.BubbleDebugConfig.TAG_BUBBLES;
36import static com.android.systemui.bubbles.BubbleDebugConfig.TAG_WITH_CLASS_NAME;
Mady Mellor3f2efdb2018-11-21 11:30:45 -080037import static com.android.systemui.statusbar.StatusBarState.SHADE;
Mady Mellor1a4e86f2019-05-03 16:07:23 -070038import static com.android.systemui.statusbar.notification.NotificationEntryManager.UNDEFINED_DISMISS_REASON;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080039
Mark Renoufba5ab512019-05-02 15:21:01 -040040import static java.lang.annotation.ElementType.FIELD;
41import static java.lang.annotation.ElementType.LOCAL_VARIABLE;
42import static java.lang.annotation.ElementType.PARAMETER;
Mark Renouf08bc42a2019-03-07 13:01:59 -050043import static java.lang.annotation.RetentionPolicy.SOURCE;
44
Mark Renoufc19b4732019-06-26 12:08:33 -040045import android.annotation.UserIdInt;
Mark Renoufc808f062019-02-07 15:20:37 -050046import android.app.ActivityManager.RunningTaskInfo;
Mady Mellor66efd5e2019-05-15 13:38:11 -070047import android.app.NotificationManager;
Mady Mellorca0c24c2019-05-16 16:14:32 -070048import android.app.PendingIntent;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080049import android.content.Context;
Mady Mellorca0c24c2019-05-16 16:14:32 -070050import android.content.pm.ActivityInfo;
Joshua Tsujia19515f2019-02-13 18:02:29 -050051import android.content.pm.ParceledListSlice;
Joshua Tsujif418f9e2019-04-04 17:09:53 -040052import android.content.res.Configuration;
Mady Mellord1c78b262018-11-06 18:04:40 -080053import android.graphics.Rect;
Mark Renoufcecc77b2019-01-30 16:32:24 -050054import android.os.RemoteException;
Mady Mellorb4991e62019-01-10 15:14:51 -080055import android.os.ServiceManager;
Mady Mellorceced172018-11-27 11:18:39 -080056import android.provider.Settings;
Mark Renoufbbcf07f2019-05-09 10:42:43 -040057import android.service.notification.NotificationListenerService.RankingMap;
Joshua Tsujidd4d9f92019-05-13 13:57:38 -040058import android.service.notification.ZenModeConfig;
Mark Renoufc19b4732019-06-26 12:08:33 -040059import android.util.ArraySet;
Mark Renouf9ba6cea2019-04-17 11:53:50 -040060import android.util.Log;
Mark Renouf82a40e62019-05-23 16:16:24 -040061import android.util.Pair;
Mark Renoufc19b4732019-06-26 12:08:33 -040062import android.util.SparseSetArray;
Mark Renoufcecc77b2019-01-30 16:32:24 -050063import android.view.Display;
Joshua Tsujia19515f2019-02-13 18:02:29 -050064import android.view.IPinnedStackController;
65import android.view.IPinnedStackListener;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080066import android.view.ViewGroup;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080067import android.widget.FrameLayout;
68
Mark Renouf08bc42a2019-03-07 13:01:59 -050069import androidx.annotation.IntDef;
Mark Renouf658c6bc2019-01-30 10:26:54 -050070import androidx.annotation.MainThread;
Joshua Tsujic650a142019-05-22 11:31:19 -040071import androidx.annotation.Nullable;
Mark Renouf658c6bc2019-01-30 10:26:54 -050072
Mady Mellorebdbbb92018-11-15 14:36:48 -080073import com.android.internal.annotations.VisibleForTesting;
Mady Mellora54e9fa2019-04-18 13:26:18 -070074import com.android.internal.statusbar.IStatusBarService;
Ned Burns01e38212019-01-03 16:32:52 -050075import com.android.systemui.Dependency;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080076import com.android.systemui.R;
Beverly8fdb5332019-02-04 14:29:49 -050077import com.android.systemui.plugins.statusbar.StatusBarStateController;
Mark Renoufcecc77b2019-01-30 16:32:24 -050078import com.android.systemui.shared.system.ActivityManagerWrapper;
79import com.android.systemui.shared.system.TaskStackChangeListener;
Joshua Tsujia19515f2019-02-13 18:02:29 -050080import com.android.systemui.shared.system.WindowManagerWrapper;
Mark Renoufc19b4732019-06-26 12:08:33 -040081import com.android.systemui.statusbar.NotificationLockscreenUserManager;
Mady Mellorc2ff0112019-03-28 14:18:06 -070082import com.android.systemui.statusbar.NotificationRemoveInterceptor;
Ned Burns01e38212019-01-03 16:32:52 -050083import com.android.systemui.statusbar.notification.NotificationEntryListener;
84import com.android.systemui.statusbar.notification.NotificationEntryManager;
Mady Mellor3f2efdb2018-11-21 11:30:45 -080085import com.android.systemui.statusbar.notification.NotificationInterruptionStateProvider;
Mark Renoufc19b4732019-06-26 12:08:33 -040086import com.android.systemui.statusbar.notification.collection.NotificationData;
Ned Burnsf81c4c42019-01-07 14:10:43 -050087import com.android.systemui.statusbar.notification.collection.NotificationEntry;
Mady Mellor22f2f072019-04-18 13:26:18 -070088import com.android.systemui.statusbar.phone.NotificationGroupManager;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080089import com.android.systemui.statusbar.phone.StatusBarWindowController;
Lyn Hanf1c9b8b2019-03-14 16:49:48 -070090import com.android.systemui.statusbar.policy.ConfigurationController;
Joshua Tsujidd4d9f92019-05-13 13:57:38 -040091import com.android.systemui.statusbar.policy.ZenModeController;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -080092
Mady Mellor70cba7bb2019-07-02 15:06:07 -070093import java.io.FileDescriptor;
94import java.io.PrintWriter;
Mark Renouf08bc42a2019-03-07 13:01:59 -050095import java.lang.annotation.Retention;
Mark Renoufba5ab512019-05-02 15:21:01 -040096import java.lang.annotation.Target;
Mady Mellor22f2f072019-04-18 13:26:18 -070097import java.util.ArrayList;
Mady Mellore80930e2019-03-21 16:00:45 -070098import java.util.List;
Mark Renouf08bc42a2019-03-07 13:01:59 -050099
Jason Monk27d01a622018-12-10 15:57:09 -0500100import javax.inject.Inject;
101import javax.inject.Singleton;
102
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800103/**
104 * Bubbles are a special type of content that can "float" on top of other apps or System UI.
105 * Bubbles can be expanded to show more content.
106 *
107 * The controller manages addition, removal, and visible state of bubbles on screen.
108 */
Jason Monk27d01a622018-12-10 15:57:09 -0500109@Singleton
Mark Renouf71a3af62019-04-08 15:02:54 -0400110public class BubbleController implements ConfigurationController.ConfigurationListener {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800111
Issei Suzukia8d07312019-06-07 12:56:19 +0200112 private static final String TAG = TAG_WITH_CLASS_NAME ? "BubbleController" : TAG_BUBBLES;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800113
Mark Renouf08bc42a2019-03-07 13:01:59 -0500114 @Retention(SOURCE)
115 @IntDef({DISMISS_USER_GESTURE, DISMISS_AGED, DISMISS_TASK_FINISHED, DISMISS_BLOCKED,
Mark Renoufc19b4732019-06-26 12:08:33 -0400116 DISMISS_NOTIF_CANCEL, DISMISS_ACCESSIBILITY_ACTION, DISMISS_NO_LONGER_BUBBLE,
Mady Mellor8454ddf2019-08-15 11:16:23 -0700117 DISMISS_USER_CHANGED, DISMISS_GROUP_CANCELLED, DISMISS_INVALID_INTENT})
Mark Renoufba5ab512019-05-02 15:21:01 -0400118 @Target({FIELD, LOCAL_VARIABLE, PARAMETER})
Mark Renouf08bc42a2019-03-07 13:01:59 -0500119 @interface DismissReason {}
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700120
Mark Renouf08bc42a2019-03-07 13:01:59 -0500121 static final int DISMISS_USER_GESTURE = 1;
122 static final int DISMISS_AGED = 2;
123 static final int DISMISS_TASK_FINISHED = 3;
124 static final int DISMISS_BLOCKED = 4;
125 static final int DISMISS_NOTIF_CANCEL = 5;
126 static final int DISMISS_ACCESSIBILITY_ACTION = 6;
Mady Melloraa8fef22019-04-11 13:36:40 -0700127 static final int DISMISS_NO_LONGER_BUBBLE = 7;
Mark Renoufc19b4732019-06-26 12:08:33 -0400128 static final int DISMISS_USER_CHANGED = 8;
Mady Mellor22f2f072019-04-18 13:26:18 -0700129 static final int DISMISS_GROUP_CANCELLED = 9;
Mady Mellor8454ddf2019-08-15 11:16:23 -0700130 static final int DISMISS_INVALID_INTENT = 10;
Mark Renouf08bc42a2019-03-07 13:01:59 -0500131
Joshua Tsuji4accf5982019-04-22 17:36:11 -0400132 public static final int MAX_BUBBLES = 5; // TODO: actually enforce this
Joshua Tsuji25a4b7b2019-03-22 14:11:06 -0400133
Mady Mellor44ee2fe2019-01-30 17:51:16 -0800134 /** Flag to enable or disable the entire feature */
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800135 private static final String ENABLE_BUBBLES = "experiment_enable_bubbles";
Joshua Tsuji010c2b12019-02-25 18:11:25 -0500136
Ned Burns01e38212019-01-03 16:32:52 -0500137 private final Context mContext;
138 private final NotificationEntryManager mNotificationEntryManager;
Mark Renoufcecc77b2019-01-30 16:32:24 -0500139 private final BubbleTaskStackListener mTaskStackListener;
Mady Mellord1c78b262018-11-06 18:04:40 -0800140 private BubbleStateChangeListener mStateChangeListener;
Mady Mellorcd9b1302018-11-06 18:08:04 -0800141 private BubbleExpandListener mExpandListener;
Issei Suzukic0387542019-03-08 17:31:14 +0100142 @Nullable private BubbleStackView.SurfaceSynchronizer mSurfaceSynchronizer;
Mady Mellor22f2f072019-04-18 13:26:18 -0700143 private final NotificationGroupManager mNotificationGroupManager;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800144
Mady Mellor3dff9e62019-02-05 18:12:53 -0800145 private BubbleData mBubbleData;
Joshua Tsujic650a142019-05-22 11:31:19 -0400146 @Nullable private BubbleStackView mStackView;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800147
Mark Renoufc19b4732019-06-26 12:08:33 -0400148 // Tracks the id of the current (foreground) user.
149 private int mCurrentUserId;
150 // Saves notification keys of active bubbles when users are switched.
151 private final SparseSetArray<String> mSavedBubbleKeysPerUser;
152
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800153 // Bubbles get added to the status bar view
Ned Burns01e38212019-01-03 16:32:52 -0500154 private final StatusBarWindowController mStatusBarWindowController;
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400155 private final ZenModeController mZenModeController;
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800156 private StatusBarStateListener mStatusBarStateListener;
157
Mady Melloraa8fef22019-04-11 13:36:40 -0700158 private final NotificationInterruptionStateProvider mNotificationInterruptionStateProvider;
Mady Mellora54e9fa2019-04-18 13:26:18 -0700159 private IStatusBarService mBarService;
Mady Mellorb4991e62019-01-10 15:14:51 -0800160
Mady Mellord1c78b262018-11-06 18:04:40 -0800161 // Used for determining view rect for touch interaction
162 private Rect mTempRect = new Rect();
163
Mark Renoufc19b4732019-06-26 12:08:33 -0400164 // Listens to user switch so bubbles can be saved and restored.
165 private final NotificationLockscreenUserManager mNotifUserManager;
166
Joshua Tsujif418f9e2019-04-04 17:09:53 -0400167 /** Last known orientation, used to detect orientation changes in {@link #onConfigChanged}. */
168 private int mOrientation = Configuration.ORIENTATION_UNDEFINED;
169
Mady Mellor5549dd22018-11-06 18:07:34 -0800170 /**
Mady Mellord1c78b262018-11-06 18:04:40 -0800171 * Listener to be notified when some states of the bubbles change.
172 */
173 public interface BubbleStateChangeListener {
174 /**
175 * Called when the stack has bubbles or no longer has bubbles.
176 */
177 void onHasBubblesChanged(boolean hasBubbles);
178 }
179
Mady Mellorcd9b1302018-11-06 18:08:04 -0800180 /**
181 * Listener to find out about stack expansion / collapse events.
182 */
183 public interface BubbleExpandListener {
184 /**
185 * Called when the expansion state of the bubble stack changes.
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700186 *
Mady Mellorcd9b1302018-11-06 18:08:04 -0800187 * @param isExpanding whether it's expanding or collapsing
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800188 * @param key the notification key associated with bubble being expanded
Mady Mellorcd9b1302018-11-06 18:08:04 -0800189 */
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800190 void onBubbleExpandChanged(boolean isExpanding, String key);
191 }
192
193 /**
194 * Listens for the current state of the status bar and updates the visibility state
195 * of bubbles as needed.
196 */
197 private class StatusBarStateListener implements StatusBarStateController.StateListener {
198 private int mState;
199 /**
200 * Returns the current status bar state.
201 */
202 public int getCurrentState() {
203 return mState;
204 }
205
206 @Override
207 public void onStateChanged(int newState) {
208 mState = newState;
Mark Renouf71a3af62019-04-08 15:02:54 -0400209 boolean shouldCollapse = (mState != SHADE);
210 if (shouldCollapse) {
211 collapseStack();
212 }
Lyn Han6c40fe72019-05-08 14:06:33 -0700213 updateStack();
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800214 }
Mady Mellorcd9b1302018-11-06 18:08:04 -0800215 }
216
Jason Monk27d01a622018-12-10 15:57:09 -0500217 @Inject
Mady Mellorcfd06c12019-02-13 14:32:12 -0800218 public BubbleController(Context context, StatusBarWindowController statusBarWindowController,
Mady Melloraa8fef22019-04-11 13:36:40 -0700219 BubbleData data, ConfigurationController configurationController,
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400220 NotificationInterruptionStateProvider interruptionStateProvider,
Mark Renoufc19b4732019-06-26 12:08:33 -0400221 ZenModeController zenModeController,
Mady Mellor22f2f072019-04-18 13:26:18 -0700222 NotificationLockscreenUserManager notifUserManager,
223 NotificationGroupManager groupManager) {
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700224 this(context, statusBarWindowController, data, null /* synchronizer */,
Mark Renoufc19b4732019-06-26 12:08:33 -0400225 configurationController, interruptionStateProvider, zenModeController,
Mady Mellor22f2f072019-04-18 13:26:18 -0700226 notifUserManager, groupManager);
Issei Suzukic0387542019-03-08 17:31:14 +0100227 }
228
229 public BubbleController(Context context, StatusBarWindowController statusBarWindowController,
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700230 BubbleData data, @Nullable BubbleStackView.SurfaceSynchronizer synchronizer,
Mady Melloraa8fef22019-04-11 13:36:40 -0700231 ConfigurationController configurationController,
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400232 NotificationInterruptionStateProvider interruptionStateProvider,
Mark Renoufc19b4732019-06-26 12:08:33 -0400233 ZenModeController zenModeController,
Mady Mellor22f2f072019-04-18 13:26:18 -0700234 NotificationLockscreenUserManager notifUserManager,
235 NotificationGroupManager groupManager) {
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800236 mContext = context;
Mady Melloraa8fef22019-04-11 13:36:40 -0700237 mNotificationInterruptionStateProvider = interruptionStateProvider;
Mark Renoufc19b4732019-06-26 12:08:33 -0400238 mNotifUserManager = notifUserManager;
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400239 mZenModeController = zenModeController;
240 mZenModeController.addCallback(new ZenModeController.Callback() {
241 @Override
242 public void onZenChanged(int zen) {
Mady Mellordf48d0a2019-06-25 18:26:46 -0700243 if (mStackView != null) {
244 mStackView.updateDots();
245 }
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400246 }
247
248 @Override
249 public void onConfigChanged(ZenModeConfig config) {
Mady Mellordf48d0a2019-06-25 18:26:46 -0700250 if (mStackView != null) {
251 mStackView.updateDots();
252 }
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400253 }
254 });
Mady Melloraa8fef22019-04-11 13:36:40 -0700255
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700256 configurationController.addCallback(this /* configurationListener */);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800257
Mark Renouf9ba6cea2019-04-17 11:53:50 -0400258 mBubbleData = data;
259 mBubbleData.setListener(mBubbleDataListener);
260
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800261 mNotificationEntryManager = Dependency.get(NotificationEntryManager.class);
Ned Burns01e38212019-01-03 16:32:52 -0500262 mNotificationEntryManager.addNotificationEntryListener(mEntryListener);
Mady Mellorc2ff0112019-03-28 14:18:06 -0700263 mNotificationEntryManager.setNotificationRemoveInterceptor(mRemoveInterceptor);
Mady Mellor22f2f072019-04-18 13:26:18 -0700264 mNotificationGroupManager = groupManager;
Mady Mellor740d85d2019-07-09 15:26:47 -0700265 mNotificationGroupManager.addOnGroupChangeListener(
266 new NotificationGroupManager.OnGroupChangeListener() {
267 @Override
268 public void onGroupSuppressionChanged(
269 NotificationGroupManager.NotificationGroup group,
270 boolean suppressed) {
271 // More notifications could be added causing summary to no longer
272 // be suppressed -- in this case need to remove the key.
273 final String groupKey = group.summary != null
274 ? group.summary.notification.getGroupKey()
275 : null;
276 if (!suppressed && groupKey != null
277 && mBubbleData.isSummarySuppressed(groupKey)) {
278 mBubbleData.removeSuppressedSummary(groupKey);
279 }
280 }
281 });
Mady Mellorb4991e62019-01-10 15:14:51 -0800282
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800283 mStatusBarWindowController = statusBarWindowController;
284 mStatusBarStateListener = new StatusBarStateListener();
285 Dependency.get(StatusBarStateController.class).addCallback(mStatusBarStateListener);
Mark Renoufcecc77b2019-01-30 16:32:24 -0500286
Mark Renoufcecc77b2019-01-30 16:32:24 -0500287 mTaskStackListener = new BubbleTaskStackListener();
288 ActivityManagerWrapper.getInstance().registerTaskStackListener(mTaskStackListener);
Mady Mellor3dff9e62019-02-05 18:12:53 -0800289
Joshua Tsujia19515f2019-02-13 18:02:29 -0500290 try {
291 WindowManagerWrapper.getInstance().addPinnedStackListener(new BubblesImeListener());
292 } catch (RemoteException e) {
293 e.printStackTrace();
294 }
Issei Suzukic0387542019-03-08 17:31:14 +0100295 mSurfaceSynchronizer = synchronizer;
Mady Mellora54e9fa2019-04-18 13:26:18 -0700296
297 mBarService = IStatusBarService.Stub.asInterface(
298 ServiceManager.getService(Context.STATUS_BAR_SERVICE));
Mark Renoufc19b4732019-06-26 12:08:33 -0400299
300 mSavedBubbleKeysPerUser = new SparseSetArray<>();
301 mCurrentUserId = mNotifUserManager.getCurrentUserId();
302 mNotifUserManager.addUserChangedListener(
303 newUserId -> {
304 saveBubbles(mCurrentUserId);
305 mBubbleData.dismissAll(DISMISS_USER_CHANGED);
306 restoreBubbles(newUserId);
307 mCurrentUserId = newUserId;
308 });
Mady Mellor5549dd22018-11-06 18:07:34 -0800309 }
310
Mark Renouf8b6a3c62019-04-09 10:17:40 -0400311 /**
312 * BubbleStackView is lazily created by this method the first time a Bubble is added. This
313 * method initializes the stack view and adds it to the StatusBar just above the scrim.
314 */
315 private void ensureStackViewCreated() {
316 if (mStackView == null) {
317 mStackView = new BubbleStackView(mContext, mBubbleData, mSurfaceSynchronizer);
318 ViewGroup sbv = mStatusBarWindowController.getStatusBarView();
Selim Cinekf97d7f72019-10-25 17:37:00 -0700319 // TODO(b/130237686): When you expand the shade on top of expanded bubble, there is no
320 // scrim between bubble and the shade
321 int bubblePosition = sbv.indexOfChild(sbv.findViewById(R.id.scrim_behind)) + 1;
322 sbv.addView(mStackView, bubblePosition,
Mark Renouf8b6a3c62019-04-09 10:17:40 -0400323 new FrameLayout.LayoutParams(MATCH_PARENT, MATCH_PARENT));
324 if (mExpandListener != null) {
325 mStackView.setExpandListener(mExpandListener);
326 }
Mark Renouf8b6a3c62019-04-09 10:17:40 -0400327 }
328 }
329
Mark Renoufc19b4732019-06-26 12:08:33 -0400330 /**
331 * Records the notification key for any active bubbles. These are used to restore active
332 * bubbles when the user returns to the foreground.
333 *
334 * @param userId the id of the user
335 */
336 private void saveBubbles(@UserIdInt int userId) {
337 // First clear any existing keys that might be stored.
338 mSavedBubbleKeysPerUser.remove(userId);
339 // Add in all active bubbles for the current user.
340 for (Bubble bubble: mBubbleData.getBubbles()) {
341 mSavedBubbleKeysPerUser.add(userId, bubble.getKey());
342 }
343 }
344
345 /**
346 * Promotes existing notifications to Bubbles if they were previously bubbles.
347 *
348 * @param userId the id of the user
349 */
350 private void restoreBubbles(@UserIdInt int userId) {
351 NotificationData notificationData =
352 mNotificationEntryManager.getNotificationData();
353 ArraySet<String> savedBubbleKeys = mSavedBubbleKeysPerUser.get(userId);
354 if (savedBubbleKeys == null) {
355 // There were no bubbles saved for this used.
356 return;
357 }
358 for (NotificationEntry e : notificationData.getNotificationsForCurrentUser()) {
359 if (savedBubbleKeys.contains(e.key)
360 && mNotificationInterruptionStateProvider.shouldBubbleUp(e)
361 && canLaunchInActivityView(mContext, e)) {
362 updateBubble(e, /* suppressFlyout= */ true);
363 }
364 }
365 // Finally, remove the entries for this user now that bubbles are restored.
366 mSavedBubbleKeysPerUser.remove(mCurrentUserId);
367 }
368
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700369 @Override
370 public void onUiModeChanged() {
371 if (mStackView != null) {
Lyn Han02cca812019-04-02 16:27:32 -0700372 mStackView.onThemeChanged();
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700373 }
374 }
375
376 @Override
377 public void onOverlayChanged() {
378 if (mStackView != null) {
Lyn Han02cca812019-04-02 16:27:32 -0700379 mStackView.onThemeChanged();
Lyn Hanf1c9b8b2019-03-14 16:49:48 -0700380 }
381 }
382
Joshua Tsujif418f9e2019-04-04 17:09:53 -0400383 @Override
384 public void onConfigChanged(Configuration newConfig) {
385 if (mStackView != null && newConfig != null && newConfig.orientation != mOrientation) {
Joshua Tsujif418f9e2019-04-04 17:09:53 -0400386 mOrientation = newConfig.orientation;
Lyn Hanf4730312019-06-18 11:18:58 -0700387 mStackView.onOrientationChanged(newConfig.orientation);
Joshua Tsujif418f9e2019-04-04 17:09:53 -0400388 }
389 }
390
Mady Mellor5549dd22018-11-06 18:07:34 -0800391 /**
Mady Mellord1c78b262018-11-06 18:04:40 -0800392 * Set a listener to be notified when some states of the bubbles change.
393 */
394 public void setBubbleStateChangeListener(BubbleStateChangeListener listener) {
395 mStateChangeListener = listener;
396 }
397
398 /**
Mady Mellorcd9b1302018-11-06 18:08:04 -0800399 * Set a listener to be notified of bubble expand events.
400 */
401 public void setExpandListener(BubbleExpandListener listener) {
Issei Suzukiac9fcb72019-02-04 17:45:57 +0100402 mExpandListener = ((isExpanding, key) -> {
403 if (listener != null) {
404 listener.onBubbleExpandChanged(isExpanding, key);
405 }
406 mStatusBarWindowController.setBubbleExpanded(isExpanding);
407 });
Mady Mellorcd9b1302018-11-06 18:08:04 -0800408 if (mStackView != null) {
409 mStackView.setExpandListener(mExpandListener);
410 }
411 }
412
413 /**
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800414 * Whether or not there are bubbles present, regardless of them being visible on the
415 * screen (e.g. if on AOD).
416 */
417 public boolean hasBubbles() {
Mady Mellor3dff9e62019-02-05 18:12:53 -0800418 if (mStackView == null) {
419 return false;
420 }
Mark Renouf71a3af62019-04-08 15:02:54 -0400421 return mBubbleData.hasBubbles();
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800422 }
423
424 /**
425 * Whether the stack of bubbles is expanded or not.
426 */
427 public boolean isStackExpanded() {
Mark Renouf71a3af62019-04-08 15:02:54 -0400428 return mBubbleData.isExpanded();
429 }
430
431 /**
432 * Tell the stack of bubbles to expand.
433 */
434 public void expandStack() {
435 mBubbleData.setExpanded(true);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800436 }
437
438 /**
439 * Tell the stack of bubbles to collapse.
440 */
441 public void collapseStack() {
Mark Renouf71a3af62019-04-08 15:02:54 -0400442 mBubbleData.setExpanded(false /* expanded */);
443 }
444
Mady Mellorce23c462019-06-17 17:30:07 -0700445 /**
Mady Mellore28fe102019-07-09 15:33:32 -0700446 * True if either:
447 * (1) There is a bubble associated with the provided key and if its notification is hidden
448 * from the shade.
449 * (2) There is a group summary associated with the provided key that is hidden from the shade
450 * because it has been dismissed but still has child bubbles active.
Mady Mellorce23c462019-06-17 17:30:07 -0700451 *
Mady Mellore28fe102019-07-09 15:33:32 -0700452 * False otherwise.
Mady Mellorce23c462019-06-17 17:30:07 -0700453 */
454 public boolean isBubbleNotificationSuppressedFromShade(String key) {
Mady Mellore28fe102019-07-09 15:33:32 -0700455 boolean isBubbleAndSuppressed = mBubbleData.hasBubbleWithKey(key)
Mady Mellorce23c462019-06-17 17:30:07 -0700456 && !mBubbleData.getBubbleWithKey(key).showInShadeWhenBubble();
Mady Mellore28fe102019-07-09 15:33:32 -0700457 NotificationEntry entry = mNotificationEntryManager.getNotificationData().get(key);
458 String groupKey = entry != null ? entry.notification.getGroupKey() : null;
459 boolean isSuppressedSummary = mBubbleData.isSummarySuppressed(groupKey);
Mady Mellore4348272019-07-29 17:43:36 -0700460 boolean isSummary = key.equals(mBubbleData.getSummaryKey(groupKey));
461 return (isSummary && isSuppressedSummary) || isBubbleAndSuppressed;
Mady Mellorce23c462019-06-17 17:30:07 -0700462 }
463
Mark Renouf71a3af62019-04-08 15:02:54 -0400464 void selectBubble(Bubble bubble) {
465 mBubbleData.setSelectedBubble(bubble);
466 }
467
468 @VisibleForTesting
469 void selectBubble(String key) {
470 Bubble bubble = mBubbleData.getBubbleWithKey(key);
471 selectBubble(bubble);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800472 }
473
474 /**
Mark Renouffec45da2019-03-13 13:24:27 -0400475 * Request the stack expand if needed, then select the specified Bubble as current.
476 *
477 * @param notificationKey the notification key for the bubble to be selected
478 */
479 public void expandStackAndSelectBubble(String notificationKey) {
Mark Renouf71a3af62019-04-08 15:02:54 -0400480 Bubble bubble = mBubbleData.getBubbleWithKey(notificationKey);
481 if (bubble != null) {
482 mBubbleData.setSelectedBubble(bubble);
483 mBubbleData.setExpanded(true);
Mark Renouffec45da2019-03-13 13:24:27 -0400484 }
485 }
486
487 /**
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800488 * Tell the stack of bubbles to be dismissed, this will remove all of the bubbles in the stack.
489 */
Mark Renouf08bc42a2019-03-07 13:01:59 -0500490 void dismissStack(@DismissReason int reason) {
Mark Renouf71a3af62019-04-08 15:02:54 -0400491 mBubbleData.dismissAll(reason);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800492 }
493
494 /**
Mark Renouf041d7262019-02-06 12:09:41 -0500495 * Directs a back gesture at the bubble stack. When opened, the current expanded bubble
496 * is forwarded a back key down/up pair.
497 */
498 public void performBackPressIfNeeded() {
499 if (mStackView != null) {
500 mStackView.performBackPressIfNeeded();
501 }
502 }
503
504 /**
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800505 * Adds or updates a bubble associated with the provided notification entry.
506 *
Mark Renouf8b6a3c62019-04-09 10:17:40 -0400507 * @param notif the notification associated with this bubble.
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800508 */
Mark Renouff97ed462019-04-05 13:46:24 -0400509 void updateBubble(NotificationEntry notif) {
Mark Renoufc19b4732019-06-26 12:08:33 -0400510 updateBubble(notif, /* supressFlyout */ false);
511 }
512
513 void updateBubble(NotificationEntry notif, boolean suppressFlyout) {
Mady Mellor66efd5e2019-05-15 13:38:11 -0700514 // If this is an interruptive notif, mark that it's interrupted
515 if (notif.importance >= NotificationManager.IMPORTANCE_HIGH) {
516 notif.setInterruption();
517 }
Mark Renoufc19b4732019-06-26 12:08:33 -0400518 mBubbleData.notificationEntryUpdated(notif, suppressFlyout);
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800519 }
520
521 /**
522 * Removes the bubble associated with the {@param uri}.
Mark Renouf658c6bc2019-01-30 10:26:54 -0500523 * <p>
524 * Must be called from the main thread.
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800525 */
Mark Renouf658c6bc2019-01-30 10:26:54 -0500526 @MainThread
Mark Renouf08bc42a2019-03-07 13:01:59 -0500527 void removeBubble(String key, int reason) {
Mark Renouf71a3af62019-04-08 15:02:54 -0400528 // TEMP: refactor to change this to pass entry
529 Bubble bubble = mBubbleData.getBubbleWithKey(key);
530 if (bubble != null) {
Mady Mellored99c272019-06-13 15:58:30 -0700531 mBubbleData.notificationEntryRemoved(bubble.getEntry(), reason);
Mady Mellore8e07712019-01-23 12:45:33 -0800532 }
533 }
534
Ned Burns01e38212019-01-03 16:32:52 -0500535 @SuppressWarnings("FieldCanBeLocal")
Mady Mellorc2ff0112019-03-28 14:18:06 -0700536 private final NotificationRemoveInterceptor mRemoveInterceptor =
537 new NotificationRemoveInterceptor() {
538 @Override
539 public boolean onNotificationRemoveRequested(String key, int reason) {
Mady Mellor22f2f072019-04-18 13:26:18 -0700540 NotificationEntry entry = mNotificationEntryManager.getNotificationData().get(key);
541 String groupKey = entry != null ? entry.notification.getGroupKey() : null;
542 ArrayList<Bubble> bubbleChildren = mBubbleData.getBubblesInGroup(groupKey);
543
544 boolean inBubbleData = mBubbleData.hasBubbleWithKey(key);
Mady Mellore28fe102019-07-09 15:33:32 -0700545 boolean isSuppressedSummary = (mBubbleData.isSummarySuppressed(groupKey)
546 && mBubbleData.getSummaryKey(groupKey).equals(key));
Mady Mellor22f2f072019-04-18 13:26:18 -0700547 boolean isSummary = entry != null
548 && entry.notification.getNotification().isGroupSummary();
Mady Mellore28fe102019-07-09 15:33:32 -0700549 boolean isSummaryOfBubbles = (isSuppressedSummary || isSummary)
550 && bubbleChildren != null && !bubbleChildren.isEmpty();
Mady Mellor22f2f072019-04-18 13:26:18 -0700551
552 if (!inBubbleData && !isSummaryOfBubbles) {
Mady Mellorc2ff0112019-03-28 14:18:06 -0700553 return false;
554 }
Mady Mellorc2ff0112019-03-28 14:18:06 -0700555
556 final boolean isClearAll = reason == REASON_CANCEL_ALL;
Mady Mellor22f2f072019-04-18 13:26:18 -0700557 final boolean isUserDimiss = reason == REASON_CANCEL || reason == REASON_CLICK;
Mady Mellorc2ff0112019-03-28 14:18:06 -0700558 final boolean isAppCancel = reason == REASON_APP_CANCEL
559 || reason == REASON_APP_CANCEL_ALL;
Mady Mellor22f2f072019-04-18 13:26:18 -0700560 final boolean isSummaryCancel = reason == REASON_GROUP_SUMMARY_CANCELED;
Mady Mellorc2ff0112019-03-28 14:18:06 -0700561
562 // Need to check for !appCancel here because the notification may have
563 // previously been dismissed & entry.isRowDismissed would still be true
Mady Mellorca184aae2019-09-17 16:07:12 -0700564 boolean userRemovedNotif = (entry != null && entry.isRowDismissed() && !isAppCancel)
Mady Mellor22f2f072019-04-18 13:26:18 -0700565 || isClearAll || isUserDimiss || isSummaryCancel;
566
567 if (isSummaryOfBubbles) {
568 return handleSummaryRemovalInterception(entry, userRemovedNotif);
569 }
Mady Mellorc2ff0112019-03-28 14:18:06 -0700570
571 // The bubble notification sticks around in the data as long as the bubble is
572 // not dismissed and the app hasn't cancelled the notification.
Mady Mellor22f2f072019-04-18 13:26:18 -0700573 Bubble bubble = mBubbleData.getBubbleWithKey(key);
Mady Mellorca184aae2019-09-17 16:07:12 -0700574 boolean bubbleExtended = entry != null && entry.isBubble() && userRemovedNotif;
Mady Mellorc2ff0112019-03-28 14:18:06 -0700575 if (bubbleExtended) {
Mady Mellorce23c462019-06-17 17:30:07 -0700576 bubble.setShowInShadeWhenBubble(false);
Mady Mellordf48d0a2019-06-25 18:26:46 -0700577 bubble.setShowBubbleDot(false);
Mady Mellorc2ff0112019-03-28 14:18:06 -0700578 if (mStackView != null) {
579 mStackView.updateDotVisibility(entry.key);
580 }
581 mNotificationEntryManager.updateNotifications();
582 return true;
Mady Mellorca184aae2019-09-17 16:07:12 -0700583 } else if (!userRemovedNotif && entry != null) {
Mady Mellorc2ff0112019-03-28 14:18:06 -0700584 // This wasn't a user removal so we should remove the bubble as well
585 mBubbleData.notificationEntryRemoved(entry, DISMISS_NOTIF_CANCEL);
586 return false;
587 }
588 return false;
589 }
590 };
591
Mady Mellor22f2f072019-04-18 13:26:18 -0700592 private boolean handleSummaryRemovalInterception(NotificationEntry summary,
593 boolean userRemovedNotif) {
594 String groupKey = summary.notification.getGroupKey();
595 ArrayList<Bubble> bubbleChildren = mBubbleData.getBubblesInGroup(groupKey);
596
597 if (userRemovedNotif) {
598 // If it's a user dismiss we mark the children to be hidden from the shade.
599 for (int i = 0; i < bubbleChildren.size(); i++) {
600 Bubble bubbleChild = bubbleChildren.get(i);
601 // As far as group manager is concerned, once a child is no longer shown
602 // in the shade, it is essentially removed.
603 mNotificationGroupManager.onEntryRemoved(bubbleChild.getEntry());
604 bubbleChild.setShowInShadeWhenBubble(false);
605 bubbleChild.setShowBubbleDot(false);
606 if (mStackView != null) {
607 mStackView.updateDotVisibility(bubbleChild.getKey());
608 }
609 }
610 // And since all children are removed, remove the summary.
611 mNotificationGroupManager.onEntryRemoved(summary);
612
613 // If the summary was auto-generated we don't need to keep that notification around
614 // because apps can't cancel it; so we only intercept & suppress real summaries.
615 boolean isAutogroupSummary = (summary.notification.getNotification().flags
616 & FLAG_AUTOGROUP_SUMMARY) != 0;
Mady Mellore28fe102019-07-09 15:33:32 -0700617 if (!isAutogroupSummary) {
618 mBubbleData.addSummaryToSuppress(summary.notification.getGroupKey(),
619 summary.key);
620 // Tell shade to update for the suppression
621 mNotificationEntryManager.updateNotifications();
622 }
Mady Mellor22f2f072019-04-18 13:26:18 -0700623 return !isAutogroupSummary;
624 } else {
Mady Mellore28fe102019-07-09 15:33:32 -0700625 // If it's not a user dismiss it's a cancel.
626 mBubbleData.removeSuppressedSummary(groupKey);
627
Mady Mellor22f2f072019-04-18 13:26:18 -0700628 // Remove any associated bubble children.
629 for (int i = 0; i < bubbleChildren.size(); i++) {
630 Bubble bubbleChild = bubbleChildren.get(i);
631 mBubbleData.notificationEntryRemoved(bubbleChild.getEntry(),
632 DISMISS_GROUP_CANCELLED);
633 }
634 return false;
635 }
636 }
637
Mady Mellorc2ff0112019-03-28 14:18:06 -0700638 @SuppressWarnings("FieldCanBeLocal")
Ned Burns01e38212019-01-03 16:32:52 -0500639 private final NotificationEntryListener mEntryListener = new NotificationEntryListener() {
640 @Override
Ned Burnsf81c4c42019-01-07 14:10:43 -0500641 public void onPendingEntryAdded(NotificationEntry entry) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800642 if (!areBubblesEnabled(mContext)) {
643 return;
644 }
Mady Mellorca0c24c2019-05-16 16:14:32 -0700645 if (mNotificationInterruptionStateProvider.shouldBubbleUp(entry)
646 && canLaunchInActivityView(mContext, entry)) {
Mark Renouff97ed462019-04-05 13:46:24 -0400647 updateBubble(entry);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800648 }
649 }
650
651 @Override
652 public void onPreEntryUpdated(NotificationEntry entry) {
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800653 if (!areBubblesEnabled(mContext)) {
654 return;
655 }
Mady Mellorca0c24c2019-05-16 16:14:32 -0700656 boolean shouldBubble = mNotificationInterruptionStateProvider.shouldBubbleUp(entry)
657 && canLaunchInActivityView(mContext, entry);
Mady Melloraa8fef22019-04-11 13:36:40 -0700658 if (!shouldBubble && mBubbleData.hasBubbleWithKey(entry.key)) {
659 // It was previously a bubble but no longer a bubble -- lets remove it
660 removeBubble(entry.key, DISMISS_NO_LONGER_BUBBLE);
Mady Mellorff40e012019-05-03 15:34:41 -0700661 } else if (shouldBubble) {
Mady Mellored99c272019-06-13 15:58:30 -0700662 Bubble b = mBubbleData.getBubbleWithKey(entry.key);
Mark Renouff97ed462019-04-05 13:46:24 -0400663 updateBubble(entry);
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800664 }
665 }
Mark Renoufbbcf07f2019-05-09 10:42:43 -0400666
667 @Override
668 public void onNotificationRankingUpdated(RankingMap rankingMap) {
669 // Forward to BubbleData to block any bubbles which should no longer be shown
670 mBubbleData.notificationRankingUpdated(rankingMap);
671 }
Ned Burns01e38212019-01-03 16:32:52 -0500672 };
673
Mark Renouf71a3af62019-04-08 15:02:54 -0400674 @SuppressWarnings("FieldCanBeLocal")
Mark Renouf3bc5b362019-04-05 14:37:59 -0400675 private final BubbleData.Listener mBubbleDataListener = new BubbleData.Listener() {
Mark Renouf71a3af62019-04-08 15:02:54 -0400676
Mark Renouf3bc5b362019-04-05 14:37:59 -0400677 @Override
Mark Renouf82a40e62019-05-23 16:16:24 -0400678 public void applyUpdate(BubbleData.Update update) {
679 if (mStackView == null && update.addedBubble != null) {
680 // Lazy init stack view when the first bubble is added.
681 ensureStackViewCreated();
Mark Renouf71a3af62019-04-08 15:02:54 -0400682 }
Mark Renouf82a40e62019-05-23 16:16:24 -0400683
684 // If not yet initialized, ignore all other changes.
685 if (mStackView == null) {
686 return;
687 }
688
689 if (update.addedBubble != null) {
690 mStackView.addBubble(update.addedBubble);
691 }
692
693 // Collapsing? Do this first before remaining steps.
694 if (update.expandedChanged && !update.expanded) {
695 mStackView.setExpanded(false);
696 }
697
698 // Do removals, if any.
Mady Mellor22f2f072019-04-18 13:26:18 -0700699 ArrayList<Pair<Bubble, Integer>> removedBubbles =
700 new ArrayList<>(update.removedBubbles);
701 for (Pair<Bubble, Integer> removed : removedBubbles) {
Mark Renouf82a40e62019-05-23 16:16:24 -0400702 final Bubble bubble = removed.first;
703 @DismissReason final int reason = removed.second;
704 mStackView.removeBubble(bubble);
705
Mark Renoufc19b4732019-06-26 12:08:33 -0400706 // If the bubble is removed for user switching, leave the notification in place.
707 if (reason != DISMISS_USER_CHANGED) {
708 if (!mBubbleData.hasBubbleWithKey(bubble.getKey())
709 && !bubble.showInShadeWhenBubble()) {
710 // The bubble is gone & the notification is gone, time to actually remove it
711 mNotificationEntryManager.performRemoveNotification(
712 bubble.getEntry().notification, UNDEFINED_DISMISS_REASON);
713 } else {
714 // Update the flag for SysUI
715 bubble.getEntry().notification.getNotification().flags &= ~FLAG_BUBBLE;
Mady Mellor3a0a1b42019-05-23 06:40:21 -0700716
Mark Renoufc19b4732019-06-26 12:08:33 -0400717 // Make sure NoMan knows it's not a bubble anymore so anyone querying it
718 // will get right result back
719 try {
720 mBarService.onNotificationBubbleChanged(bubble.getKey(),
721 false /* isBubble */);
722 } catch (RemoteException e) {
723 // Bad things have happened
724 }
Mark Renouf82a40e62019-05-23 16:16:24 -0400725 }
Mady Mellor22f2f072019-04-18 13:26:18 -0700726
Mady Mellore28fe102019-07-09 15:33:32 -0700727 // Check if removed bubble has an associated suppressed group summary that needs
728 // to be removed now.
729 final String groupKey = bubble.getEntry().notification.getGroupKey();
730 if (mBubbleData.isSummarySuppressed(groupKey)
731 && mBubbleData.getBubblesInGroup(groupKey).isEmpty()) {
732 // Time to actually remove the summary.
733 String notifKey = mBubbleData.getSummaryKey(groupKey);
734 mBubbleData.removeSuppressedSummary(groupKey);
735 NotificationEntry entry =
736 mNotificationEntryManager.getNotificationData().get(notifKey);
Mady Mellore28fe102019-07-09 15:33:32 -0700737 mNotificationEntryManager.performRemoveNotification(
738 entry.notification, UNDEFINED_DISMISS_REASON);
739 }
740
Mady Mellor22f2f072019-04-18 13:26:18 -0700741 // Check if summary should be removed from NoManGroup
742 NotificationEntry summary = mNotificationGroupManager.getLogicalGroupSummary(
743 bubble.getEntry().notification);
744 if (summary != null) {
745 ArrayList<NotificationEntry> summaryChildren =
746 mNotificationGroupManager.getLogicalChildren(summary.notification);
Mady Mellore4348272019-07-29 17:43:36 -0700747 boolean isSummaryThisNotif = summary.key.equals(bubble.getEntry().key);
748 if (!isSummaryThisNotif
749 && (summaryChildren == null || summaryChildren.isEmpty())) {
Mady Mellor22f2f072019-04-18 13:26:18 -0700750 mNotificationEntryManager.performRemoveNotification(
751 summary.notification, UNDEFINED_DISMISS_REASON);
752 }
753 }
Mady Mellora54e9fa2019-04-18 13:26:18 -0700754 }
755 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400756
Mark Renouf82a40e62019-05-23 16:16:24 -0400757 if (update.updatedBubble != null) {
758 mStackView.updateBubble(update.updatedBubble);
Mark Renouf71a3af62019-04-08 15:02:54 -0400759 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400760
Mark Renouf82a40e62019-05-23 16:16:24 -0400761 if (update.orderChanged) {
762 mStackView.updateBubbleOrder(update.bubbles);
Mark Renoufba5ab512019-05-02 15:21:01 -0400763 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400764
Mark Renouf82a40e62019-05-23 16:16:24 -0400765 if (update.selectionChanged) {
766 mStackView.setSelectedBubble(update.selectedBubble);
Mady Mellor740d85d2019-07-09 15:26:47 -0700767 if (update.selectedBubble != null) {
768 mNotificationGroupManager.updateSuppression(
769 update.selectedBubble.getEntry());
770 }
Mark Renouf71a3af62019-04-08 15:02:54 -0400771 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400772
Mark Renouf82a40e62019-05-23 16:16:24 -0400773 // Expanding? Apply this last.
774 if (update.expandedChanged && update.expanded) {
775 mStackView.setExpanded(true);
Mark Renouf71a3af62019-04-08 15:02:54 -0400776 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400777
Mark Renouf71a3af62019-04-08 15:02:54 -0400778 mNotificationEntryManager.updateNotifications();
Lyn Han6c40fe72019-05-08 14:06:33 -0700779 updateStack();
Mark Renouf9ba6cea2019-04-17 11:53:50 -0400780
Issei Suzukia8d07312019-06-07 12:56:19 +0200781 if (DEBUG_BUBBLE_CONTROLLER) {
Mark Renouf9ba6cea2019-04-17 11:53:50 -0400782 Log.d(TAG, "[BubbleData]");
783 Log.d(TAG, formatBubblesString(mBubbleData.getBubbles(),
784 mBubbleData.getSelectedBubble()));
785
786 if (mStackView != null) {
787 Log.d(TAG, "[BubbleStackView]");
788 Log.d(TAG, formatBubblesString(mStackView.getBubblesOnScreen(),
789 mStackView.getExpandedBubble()));
790 }
791 }
Mark Renouf3bc5b362019-04-05 14:37:59 -0400792 }
793 };
794
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800795 /**
Joshua Tsujidd4d9f92019-05-13 13:57:38 -0400796 * Lets any listeners know if bubble state has changed.
Lyn Han6c40fe72019-05-08 14:06:33 -0700797 * Updates the visibility of the bubbles based on current state.
798 * Does not un-bubble, just hides or un-hides. Notifies any
799 * {@link BubbleStateChangeListener}s of visibility changes.
800 * Updates stack description for TalkBack focus.
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800801 */
Lyn Han6c40fe72019-05-08 14:06:33 -0700802 public void updateStack() {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800803 if (mStackView == null) {
804 return;
Mady Mellord1c78b262018-11-06 18:04:40 -0800805 }
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800806 if (mStatusBarStateListener.getCurrentState() == SHADE && hasBubbles()) {
807 // Bubbles only appear in unlocked shade
808 mStackView.setVisibility(hasBubbles() ? VISIBLE : INVISIBLE);
Mady Mellor698d9e82019-08-01 23:11:53 +0000809 } else if (mStackView != null) {
Mady Mellor3f2efdb2018-11-21 11:30:45 -0800810 mStackView.setVisibility(INVISIBLE);
Mady Mellor5549dd22018-11-06 18:07:34 -0800811 }
Lyn Han6c40fe72019-05-08 14:06:33 -0700812
Mady Mellor698d9e82019-08-01 23:11:53 +0000813 // Let listeners know if bubble state changed.
Lyn Han6c40fe72019-05-08 14:06:33 -0700814 boolean hadBubbles = mStatusBarWindowController.getBubblesShowing();
Mady Mellor698d9e82019-08-01 23:11:53 +0000815 boolean hasBubblesShowing = hasBubbles() && mStackView.getVisibility() == VISIBLE;
Mady Mellor88552b82019-08-05 22:38:59 +0000816 mStatusBarWindowController.setBubblesShowing(hasBubblesShowing);
Lyn Han6c40fe72019-05-08 14:06:33 -0700817 if (mStateChangeListener != null && hadBubbles != hasBubblesShowing) {
818 mStateChangeListener.onHasBubblesChanged(hasBubblesShowing);
819 }
820
821 mStackView.updateContentDescription();
Mady Mellord1c78b262018-11-06 18:04:40 -0800822 }
823
824 /**
825 * Rect indicating the touchable region for the bubble stack / expanded stack.
826 */
827 public Rect getTouchableRegion() {
828 if (mStackView == null || mStackView.getVisibility() != VISIBLE) {
829 return null;
830 }
831 mStackView.getBoundsOnScreen(mTempRect);
832 return mTempRect;
Mady Mellorc3d6f7d2018-11-07 09:36:56 -0800833 }
834
Mady Mellor390bff42019-04-05 15:09:01 -0700835 /**
836 * The display id of the expanded view, if the stack is expanded and not occluded by the
837 * status bar, otherwise returns {@link Display#INVALID_DISPLAY}.
838 */
839 public int getExpandedDisplayId(Context context) {
Issei Suzukicac2a502019-04-16 16:52:50 +0200840 final Bubble bubble = getExpandedBubble(context);
841 return bubble != null ? bubble.getDisplayId() : INVALID_DISPLAY;
842 }
843
844 @Nullable
845 private Bubble getExpandedBubble(Context context) {
Joel Galenson4071ddb2019-04-18 13:30:45 -0700846 if (mStackView == null) {
Issei Suzukicac2a502019-04-16 16:52:50 +0200847 return null;
Joel Galenson4071ddb2019-04-18 13:30:45 -0700848 }
Issei Suzukicac2a502019-04-16 16:52:50 +0200849 final boolean defaultDisplay = context.getDisplay() != null
Mady Mellor390bff42019-04-05 15:09:01 -0700850 && context.getDisplay().getDisplayId() == DEFAULT_DISPLAY;
Issei Suzukicac2a502019-04-16 16:52:50 +0200851 final Bubble expandedBubble = mStackView.getExpandedBubble();
852 if (defaultDisplay && expandedBubble != null && isStackExpanded()
Mady Mellor390bff42019-04-05 15:09:01 -0700853 && !mStatusBarWindowController.getPanelExpanded()) {
Issei Suzukicac2a502019-04-16 16:52:50 +0200854 return expandedBubble;
Mady Mellor390bff42019-04-05 15:09:01 -0700855 }
Issei Suzukicac2a502019-04-16 16:52:50 +0200856 return null;
Mady Mellor390bff42019-04-05 15:09:01 -0700857 }
858
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800859 @VisibleForTesting
860 BubbleStackView getStackView() {
861 return mStackView;
862 }
863
Mady Mellor70cba7bb2019-07-02 15:06:07 -0700864 /**
865 * Description of current bubble state.
866 */
867 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
868 pw.println("BubbleController state:");
869 mBubbleData.dump(fd, pw, args);
870 pw.println();
Joshua Tsuji395bcfe2019-07-02 19:23:23 -0400871 if (mStackView != null) {
872 mStackView.dump(fd, pw, args);
873 }
874 pw.println();
Mady Mellor70cba7bb2019-07-02 15:06:07 -0700875 }
876
Mark Renouf9ba6cea2019-04-17 11:53:50 -0400877 static String formatBubblesString(List<Bubble> bubbles, Bubble selected) {
878 StringBuilder sb = new StringBuilder();
879 for (Bubble bubble : bubbles) {
880 if (bubble == null) {
881 sb.append(" <null> !!!!!\n");
882 } else {
883 boolean isSelected = (bubble == selected);
884 sb.append(String.format("%s Bubble{act=%12d, ongoing=%d, key=%s}\n",
885 ((isSelected) ? "->" : " "),
886 bubble.getLastActivity(),
887 (bubble.isOngoing() ? 1 : 0),
888 bubble.getKey()));
889 }
890 }
891 return sb.toString();
892 }
893
Mady Mellore80930e2019-03-21 16:00:45 -0700894 /**
Mark Renoufcecc77b2019-01-30 16:32:24 -0500895 * This task stack listener is responsible for responding to tasks moved to the front
896 * which are on the default (main) display. When this happens, expanded bubbles must be
897 * collapsed so the user may interact with the app which was just moved to the front.
898 * <p>
899 * This listener is registered with SystemUI's ActivityManagerWrapper which dispatches
900 * these calls via a main thread Handler.
901 */
902 @MainThread
903 private class BubbleTaskStackListener extends TaskStackChangeListener {
904
Mark Renoufcecc77b2019-01-30 16:32:24 -0500905 @Override
Mark Renoufc808f062019-02-07 15:20:37 -0500906 public void onTaskMovedToFront(RunningTaskInfo taskInfo) {
907 if (mStackView != null && taskInfo.displayId == Display.DEFAULT_DISPLAY) {
Mady Mellor047e24e2019-08-05 11:35:40 -0700908 if (!mStackView.isExpansionAnimating()) {
909 mBubbleData.setExpanded(false);
910 }
Mark Renoufcecc77b2019-01-30 16:32:24 -0500911 }
912 }
913
Mark Renoufcecc77b2019-01-30 16:32:24 -0500914 @Override
Mark Renoufa12e8762019-03-07 15:43:01 -0500915 public void onActivityLaunchOnSecondaryDisplayRerouted() {
Mark Renoufcecc77b2019-01-30 16:32:24 -0500916 if (mStackView != null) {
Mark Renouf71a3af62019-04-08 15:02:54 -0400917 mBubbleData.setExpanded(false);
Mark Renoufcecc77b2019-01-30 16:32:24 -0500918 }
919 }
Mark Renouf446251d2019-04-26 10:22:41 -0400920
921 @Override
922 public void onBackPressedOnTaskRoot(RunningTaskInfo taskInfo) {
923 if (mStackView != null && taskInfo.displayId == getExpandedDisplayId(mContext)) {
924 mBubbleData.setExpanded(false);
925 }
926 }
Issei Suzukicac2a502019-04-16 16:52:50 +0200927
928 @Override
929 public void onSingleTaskDisplayDrawn(int displayId) {
Mady Mellor5186b132019-09-16 17:55:48 -0700930 final Bubble expandedBubble = mStackView != null
931 ? mStackView.getExpandedBubble()
932 : null;
Issei Suzukicac2a502019-04-16 16:52:50 +0200933 if (expandedBubble != null && expandedBubble.getDisplayId() == displayId) {
934 expandedBubble.setContentVisibility(true);
935 }
936 }
Issei Suzuki734bc942019-06-05 13:59:52 +0200937
938 @Override
939 public void onSingleTaskDisplayEmpty(int displayId) {
Mady Mellor5186b132019-09-16 17:55:48 -0700940 final Bubble expandedBubble = mStackView != null
941 ? mStackView.getExpandedBubble()
942 : null;
Mady Mellorca184aae2019-09-17 16:07:12 -0700943 int expandedId = expandedBubble != null ? expandedBubble.getDisplayId() : -1;
944 if (mStackView != null && mStackView.isExpanded() && expandedId == displayId) {
Issei Suzuki734bc942019-06-05 13:59:52 +0200945 mBubbleData.setExpanded(false);
Issei Suzuki734bc942019-06-05 13:59:52 +0200946 }
Mady Mellorca184aae2019-09-17 16:07:12 -0700947 mBubbleData.notifyDisplayEmpty(displayId);
Issei Suzuki734bc942019-06-05 13:59:52 +0200948 }
Mark Renoufcecc77b2019-01-30 16:32:24 -0500949 }
950
Mady Mellorf6e3ac02019-01-29 10:37:52 -0800951 private static boolean areBubblesEnabled(Context context) {
952 return Settings.Secure.getInt(context.getContentResolver(),
953 ENABLE_BUBBLES, 1) != 0;
954 }
Joshua Tsujia19515f2019-02-13 18:02:29 -0500955
Mady Mellorca0c24c2019-05-16 16:14:32 -0700956 /**
957 * Whether an intent is properly configured to display in an {@link android.app.ActivityView}.
958 *
959 * Keep checks in sync with NotificationManagerService#canLaunchInActivityView. Typically
960 * that should filter out any invalid bubbles, but should protect SysUI side just in case.
961 *
962 * @param context the context to use.
963 * @param entry the entry to bubble.
964 */
965 static boolean canLaunchInActivityView(Context context, NotificationEntry entry) {
966 PendingIntent intent = entry.getBubbleMetadata() != null
967 ? entry.getBubbleMetadata().getIntent()
968 : null;
969 if (intent == null) {
970 Log.w(TAG, "Unable to create bubble -- no intent");
971 return false;
972 }
973 ActivityInfo info =
974 intent.getIntent().resolveActivityInfo(context.getPackageManager(), 0);
975 if (info == null) {
976 Log.w(TAG, "Unable to send as bubble -- couldn't find activity info for intent: "
977 + intent);
978 return false;
979 }
980 if (!ActivityInfo.isResizeableMode(info.resizeMode)) {
981 Log.w(TAG, "Unable to send as bubble -- activity is not resizable for intent: "
982 + intent);
983 return false;
984 }
985 if (info.documentLaunchMode != DOCUMENT_LAUNCH_ALWAYS) {
986 Log.w(TAG, "Unable to send as bubble -- activity is not documentLaunchMode=always "
987 + "for intent: " + intent);
988 return false;
989 }
990 if ((info.flags & ActivityInfo.FLAG_ALLOW_EMBEDDED) == 0) {
991 Log.w(TAG, "Unable to send as bubble -- activity is not embeddable for intent: "
992 + intent);
993 return false;
994 }
995 return true;
996 }
997
Joshua Tsujia19515f2019-02-13 18:02:29 -0500998 /** PinnedStackListener that dispatches IME visibility updates to the stack. */
999 private class BubblesImeListener extends IPinnedStackListener.Stub {
1000
1001 @Override
1002 public void onListenerRegistered(IPinnedStackController controller) throws RemoteException {
1003 }
1004
1005 @Override
1006 public void onMovementBoundsChanged(Rect insetBounds, Rect normalBounds,
1007 Rect animatingBounds, boolean fromImeAdjustment, boolean fromShelfAdjustment,
1008 int displayRotation) throws RemoteException {}
1009
1010 @Override
Joshua Tsujid9422832019-03-05 13:32:37 -05001011 public void onImeVisibilityChanged(boolean imeVisible, int imeHeight) {
1012 if (mStackView != null && mStackView.getBubbleCount() > 0) {
1013 mStackView.post(() -> mStackView.onImeVisibilityChanged(imeVisible, imeHeight));
Joshua Tsujia19515f2019-02-13 18:02:29 -05001014 }
1015 }
1016
1017 @Override
1018 public void onShelfVisibilityChanged(boolean shelfVisible, int shelfHeight)
1019 throws RemoteException {}
1020
1021 @Override
1022 public void onMinimizedStateChanged(boolean isMinimized) throws RemoteException {}
1023
1024 @Override
1025 public void onActionsChanged(ParceledListSlice actions) throws RemoteException {}
1026 }
Mady Mellorc3d6f7d2018-11-07 09:36:56 -08001027}