1 /* 2 * Copyright (C) 2015 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.phone; 18 19 import android.service.notification.StatusBarNotification; 20 import android.support.annotation.Nullable; 21 22 import com.android.systemui.statusbar.ExpandableNotificationRow; 23 import com.android.systemui.statusbar.NotificationData; 24 import com.android.systemui.statusbar.StatusBarState; 25 import com.android.systemui.statusbar.policy.HeadsUpManager; 26 import com.android.systemui.statusbar.policy.OnHeadsUpChangedListener; 27 28 import java.io.FileDescriptor; 29 import java.io.PrintWriter; 30 import java.util.ArrayList; 31 import java.util.HashMap; 32 import java.util.HashSet; 33 import java.util.Iterator; 34 import java.util.Map; 35 36 /** 37 * A class to handle notifications and their corresponding groups. 38 */ 39 public class NotificationGroupManager implements OnHeadsUpChangedListener { 40 41 private final HashMap<String, NotificationGroup> mGroupMap = new HashMap<>(); 42 private OnGroupChangeListener mListener; 43 private int mBarState = -1; 44 private HashMap<String, StatusBarNotification> mIsolatedEntries = new HashMap<>(); 45 private HeadsUpManager mHeadsUpManager; 46 private boolean mIsUpdatingUnchangedGroup; 47 setOnGroupChangeListener(OnGroupChangeListener listener)48 public void setOnGroupChangeListener(OnGroupChangeListener listener) { 49 mListener = listener; 50 } 51 isGroupExpanded(StatusBarNotification sbn)52 public boolean isGroupExpanded(StatusBarNotification sbn) { 53 NotificationGroup group = mGroupMap.get(getGroupKey(sbn)); 54 if (group == null) { 55 return false; 56 } 57 return group.expanded; 58 } 59 setGroupExpanded(StatusBarNotification sbn, boolean expanded)60 public void setGroupExpanded(StatusBarNotification sbn, boolean expanded) { 61 NotificationGroup group = mGroupMap.get(getGroupKey(sbn)); 62 if (group == null) { 63 return; 64 } 65 setGroupExpanded(group, expanded); 66 } 67 setGroupExpanded(NotificationGroup group, boolean expanded)68 private void setGroupExpanded(NotificationGroup group, boolean expanded) { 69 group.expanded = expanded; 70 if (group.summary != null) { 71 mListener.onGroupExpansionChanged(group.summary.row, expanded); 72 } 73 } 74 onEntryRemoved(NotificationData.Entry removed)75 public void onEntryRemoved(NotificationData.Entry removed) { 76 onEntryRemovedInternal(removed, removed.notification); 77 mIsolatedEntries.remove(removed.key); 78 } 79 80 /** 81 * An entry was removed. 82 * 83 * @param removed the removed entry 84 * @param sbn the notification the entry has, which doesn't need to be the same as it's internal 85 * notification 86 */ onEntryRemovedInternal(NotificationData.Entry removed, final StatusBarNotification sbn)87 private void onEntryRemovedInternal(NotificationData.Entry removed, 88 final StatusBarNotification sbn) { 89 String groupKey = getGroupKey(sbn); 90 final NotificationGroup group = mGroupMap.get(groupKey); 91 if (group == null) { 92 // When an app posts 2 different notifications as summary of the same group, then a 93 // cancellation of the first notification removes this group. 94 // This situation is not supported and we will not allow such notifications anymore in 95 // the close future. See b/23676310 for reference. 96 return; 97 } 98 if (isGroupChild(sbn)) { 99 group.children.remove(removed); 100 } else { 101 group.summary = null; 102 } 103 updateSuppression(group); 104 if (group.children.isEmpty()) { 105 if (group.summary == null) { 106 mGroupMap.remove(groupKey); 107 } 108 } 109 } 110 onEntryAdded(final NotificationData.Entry added)111 public void onEntryAdded(final NotificationData.Entry added) { 112 final StatusBarNotification sbn = added.notification; 113 boolean isGroupChild = isGroupChild(sbn); 114 String groupKey = getGroupKey(sbn); 115 NotificationGroup group = mGroupMap.get(groupKey); 116 if (group == null) { 117 group = new NotificationGroup(); 118 mGroupMap.put(groupKey, group); 119 } 120 if (isGroupChild) { 121 group.children.add(added); 122 updateSuppression(group); 123 } else { 124 group.summary = added; 125 group.expanded = added.row.areChildrenExpanded(); 126 updateSuppression(group); 127 if (!group.children.isEmpty()) { 128 HashSet<NotificationData.Entry> childrenCopy = 129 (HashSet<NotificationData.Entry>) group.children.clone(); 130 for (NotificationData.Entry child : childrenCopy) { 131 onEntryBecomingChild(child); 132 } 133 mListener.onGroupCreatedFromChildren(group); 134 } 135 } 136 } 137 onEntryBecomingChild(NotificationData.Entry entry)138 private void onEntryBecomingChild(NotificationData.Entry entry) { 139 if (entry.row.isHeadsUp()) { 140 onHeadsUpStateChanged(entry, true); 141 } 142 } 143 updateSuppression(NotificationGroup group)144 private void updateSuppression(NotificationGroup group) { 145 if (group == null) { 146 return; 147 } 148 boolean prevSuppressed = group.suppressed; 149 group.suppressed = group.summary != null && !group.expanded 150 && (group.children.size() == 1 151 || (group.children.size() == 0 152 && group.summary.notification.getNotification().isGroupSummary() 153 && hasIsolatedChildren(group))); 154 if (prevSuppressed != group.suppressed) { 155 if (group.suppressed) { 156 handleSuppressedSummaryHeadsUpped(group.summary); 157 } 158 if (!mIsUpdatingUnchangedGroup) { 159 mListener.onGroupsChanged(); 160 } 161 } 162 } 163 hasIsolatedChildren(NotificationGroup group)164 private boolean hasIsolatedChildren(NotificationGroup group) { 165 return getNumberOfIsolatedChildren(group.summary.notification.getGroupKey()) != 0; 166 } 167 getNumberOfIsolatedChildren(String groupKey)168 private int getNumberOfIsolatedChildren(String groupKey) { 169 int count = 0; 170 for (StatusBarNotification sbn : mIsolatedEntries.values()) { 171 if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) { 172 count++; 173 } 174 } 175 return count; 176 } 177 getIsolatedChild(String groupKey)178 private NotificationData.Entry getIsolatedChild(String groupKey) { 179 for (StatusBarNotification sbn : mIsolatedEntries.values()) { 180 if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) { 181 return mGroupMap.get(sbn.getKey()).summary; 182 } 183 } 184 return null; 185 } 186 onEntryUpdated(NotificationData.Entry entry, StatusBarNotification oldNotification)187 public void onEntryUpdated(NotificationData.Entry entry, 188 StatusBarNotification oldNotification) { 189 String oldKey = oldNotification.getGroupKey(); 190 String newKey = entry.notification.getGroupKey(); 191 boolean groupKeysChanged = !oldKey.equals(newKey); 192 boolean wasGroupChild = isGroupChild(oldNotification); 193 boolean isGroupChild = isGroupChild(entry.notification); 194 mIsUpdatingUnchangedGroup = !groupKeysChanged && wasGroupChild == isGroupChild; 195 if (mGroupMap.get(getGroupKey(oldNotification)) != null) { 196 onEntryRemovedInternal(entry, oldNotification); 197 } 198 onEntryAdded(entry); 199 mIsUpdatingUnchangedGroup = false; 200 if (isIsolated(entry.notification)) { 201 mIsolatedEntries.put(entry.key, entry.notification); 202 if (groupKeysChanged) { 203 updateSuppression(mGroupMap.get(oldKey)); 204 updateSuppression(mGroupMap.get(newKey)); 205 } 206 } else if (!wasGroupChild && isGroupChild) { 207 onEntryBecomingChild(entry); 208 } 209 } 210 isSummaryOfSuppressedGroup(StatusBarNotification sbn)211 public boolean isSummaryOfSuppressedGroup(StatusBarNotification sbn) { 212 return isGroupSuppressed(getGroupKey(sbn)) && sbn.getNotification().isGroupSummary(); 213 } 214 isOnlyChild(StatusBarNotification sbn)215 private boolean isOnlyChild(StatusBarNotification sbn) { 216 return !sbn.getNotification().isGroupSummary() 217 && getTotalNumberOfChildren(sbn) == 1; 218 } 219 isOnlyChildInGroup(StatusBarNotification sbn)220 public boolean isOnlyChildInGroup(StatusBarNotification sbn) { 221 if (!isOnlyChild(sbn)) { 222 return false; 223 } 224 ExpandableNotificationRow logicalGroupSummary = getLogicalGroupSummary(sbn); 225 return logicalGroupSummary != null 226 && !logicalGroupSummary.getStatusBarNotification().equals(sbn); 227 } 228 getTotalNumberOfChildren(StatusBarNotification sbn)229 private int getTotalNumberOfChildren(StatusBarNotification sbn) { 230 int isolatedChildren = getNumberOfIsolatedChildren(sbn.getGroupKey()); 231 NotificationGroup group = mGroupMap.get(sbn.getGroupKey()); 232 int realChildren = group != null ? group.children.size() : 0; 233 return isolatedChildren + realChildren; 234 } 235 isGroupSuppressed(String groupKey)236 private boolean isGroupSuppressed(String groupKey) { 237 NotificationGroup group = mGroupMap.get(groupKey); 238 return group != null && group.suppressed; 239 } 240 setStatusBarState(int newState)241 public void setStatusBarState(int newState) { 242 if (mBarState == newState) { 243 return; 244 } 245 mBarState = newState; 246 if (mBarState == StatusBarState.KEYGUARD) { 247 collapseAllGroups(); 248 } 249 } 250 collapseAllGroups()251 public void collapseAllGroups() { 252 // Because notifications can become isolated when the group becomes suppressed it can 253 // lead to concurrent modifications while looping. We need to make a copy. 254 ArrayList<NotificationGroup> groupCopy = new ArrayList<>(mGroupMap.values()); 255 int size = groupCopy.size(); 256 for (int i = 0; i < size; i++) { 257 NotificationGroup group = groupCopy.get(i); 258 if (group.expanded) { 259 setGroupExpanded(group, false); 260 } 261 updateSuppression(group); 262 } 263 } 264 265 /** 266 * @return whether a given notification is a child in a group which has a summary 267 */ isChildInGroupWithSummary(StatusBarNotification sbn)268 public boolean isChildInGroupWithSummary(StatusBarNotification sbn) { 269 if (!isGroupChild(sbn)) { 270 return false; 271 } 272 NotificationGroup group = mGroupMap.get(getGroupKey(sbn)); 273 if (group == null || group.summary == null || group.suppressed) { 274 return false; 275 } 276 if (group.children.isEmpty()) { 277 // If the suppression of a group changes because the last child was removed, this can 278 // still be called temporarily because the child hasn't been fully removed yet. Let's 279 // make sure we still return false in that case. 280 return false; 281 } 282 return true; 283 } 284 285 /** 286 * @return whether a given notification is a summary in a group which has children 287 */ isSummaryOfGroup(StatusBarNotification sbn)288 public boolean isSummaryOfGroup(StatusBarNotification sbn) { 289 if (!isGroupSummary(sbn)) { 290 return false; 291 } 292 NotificationGroup group = mGroupMap.get(getGroupKey(sbn)); 293 if (group == null) { 294 return false; 295 } 296 return !group.children.isEmpty(); 297 } 298 299 /** 300 * Get the summary of a specified status bar notification. For isolated notification this return 301 * itself. 302 */ getGroupSummary(StatusBarNotification sbn)303 public ExpandableNotificationRow getGroupSummary(StatusBarNotification sbn) { 304 return getGroupSummary(getGroupKey(sbn)); 305 } 306 307 /** 308 * Similar to {@link #getGroupSummary(StatusBarNotification)} but doesn't get the visual summary 309 * but the logical summary, i.e when a child is isolated, it still returns the summary as if 310 * it wasn't isolated. 311 */ getLogicalGroupSummary( StatusBarNotification sbn)312 public ExpandableNotificationRow getLogicalGroupSummary( 313 StatusBarNotification sbn) { 314 return getGroupSummary(sbn.getGroupKey()); 315 } 316 317 @Nullable getGroupSummary(String groupKey)318 private ExpandableNotificationRow getGroupSummary(String groupKey) { 319 NotificationGroup group = mGroupMap.get(groupKey); 320 return group == null ? null 321 : group.summary == null ? null 322 : group.summary.row; 323 } 324 325 /** @return group expansion state after toggling. */ toggleGroupExpansion(StatusBarNotification sbn)326 public boolean toggleGroupExpansion(StatusBarNotification sbn) { 327 NotificationGroup group = mGroupMap.get(getGroupKey(sbn)); 328 if (group == null) { 329 return false; 330 } 331 setGroupExpanded(group, !group.expanded); 332 return group.expanded; 333 } 334 isIsolated(StatusBarNotification sbn)335 private boolean isIsolated(StatusBarNotification sbn) { 336 return mIsolatedEntries.containsKey(sbn.getKey()); 337 } 338 isGroupSummary(StatusBarNotification sbn)339 private boolean isGroupSummary(StatusBarNotification sbn) { 340 if (isIsolated(sbn)) { 341 return true; 342 } 343 return sbn.getNotification().isGroupSummary(); 344 } 345 isGroupChild(StatusBarNotification sbn)346 private boolean isGroupChild(StatusBarNotification sbn) { 347 if (isIsolated(sbn)) { 348 return false; 349 } 350 return sbn.isGroup() && !sbn.getNotification().isGroupSummary(); 351 } 352 getGroupKey(StatusBarNotification sbn)353 private String getGroupKey(StatusBarNotification sbn) { 354 if (isIsolated(sbn)) { 355 return sbn.getKey(); 356 } 357 return sbn.getGroupKey(); 358 } 359 360 @Override onHeadsUpPinnedModeChanged(boolean inPinnedMode)361 public void onHeadsUpPinnedModeChanged(boolean inPinnedMode) { 362 } 363 364 @Override onHeadsUpPinned(ExpandableNotificationRow headsUp)365 public void onHeadsUpPinned(ExpandableNotificationRow headsUp) { 366 } 367 368 @Override onHeadsUpUnPinned(ExpandableNotificationRow headsUp)369 public void onHeadsUpUnPinned(ExpandableNotificationRow headsUp) { 370 } 371 372 @Override onHeadsUpStateChanged(NotificationData.Entry entry, boolean isHeadsUp)373 public void onHeadsUpStateChanged(NotificationData.Entry entry, boolean isHeadsUp) { 374 final StatusBarNotification sbn = entry.notification; 375 if (entry.row.isHeadsUp()) { 376 if (shouldIsolate(sbn)) { 377 // We will be isolated now, so lets update the groups 378 onEntryRemovedInternal(entry, entry.notification); 379 380 mIsolatedEntries.put(sbn.getKey(), sbn); 381 382 onEntryAdded(entry); 383 // We also need to update the suppression of the old group, because this call comes 384 // even before the groupManager knows about the notification at all. 385 // When the notification gets added afterwards it is already isolated and therefore 386 // it doesn't lead to an update. 387 updateSuppression(mGroupMap.get(entry.notification.getGroupKey())); 388 mListener.onGroupsChanged(); 389 } else { 390 handleSuppressedSummaryHeadsUpped(entry); 391 } 392 } else { 393 if (mIsolatedEntries.containsKey(sbn.getKey())) { 394 // not isolated anymore, we need to update the groups 395 onEntryRemovedInternal(entry, entry.notification); 396 mIsolatedEntries.remove(sbn.getKey()); 397 onEntryAdded(entry); 398 mListener.onGroupsChanged(); 399 } 400 } 401 } 402 handleSuppressedSummaryHeadsUpped(NotificationData.Entry entry)403 private void handleSuppressedSummaryHeadsUpped(NotificationData.Entry entry) { 404 StatusBarNotification sbn = entry.notification; 405 if (!isGroupSuppressed(sbn.getGroupKey()) 406 || !sbn.getNotification().isGroupSummary() 407 || !entry.row.isHeadsUp()) { 408 return; 409 } 410 // The parent of a suppressed group got huned, lets hun the child! 411 NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey()); 412 if (notificationGroup != null) { 413 Iterator<NotificationData.Entry> iterator = notificationGroup.children.iterator(); 414 NotificationData.Entry child = iterator.hasNext() ? iterator.next() : null; 415 if (child == null) { 416 child = getIsolatedChild(sbn.getGroupKey()); 417 } 418 if (child != null) { 419 if (child.row.keepInParent() || child.row.isRemoved() || child.row.isDismissed()) { 420 // the notification is actually already removed, no need to do heads-up on it. 421 return; 422 } 423 if (mHeadsUpManager.isHeadsUp(child.key)) { 424 mHeadsUpManager.updateNotification(child, true); 425 } else { 426 mHeadsUpManager.showNotification(child); 427 } 428 } 429 } 430 mHeadsUpManager.releaseImmediately(entry.key); 431 } 432 shouldIsolate(StatusBarNotification sbn)433 private boolean shouldIsolate(StatusBarNotification sbn) { 434 NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey()); 435 return (sbn.isGroup() && !sbn.getNotification().isGroupSummary()) 436 && (sbn.getNotification().fullScreenIntent != null 437 || notificationGroup == null 438 || !notificationGroup.expanded 439 || isGroupNotFullyVisible(notificationGroup)); 440 } 441 isGroupNotFullyVisible(NotificationGroup notificationGroup)442 private boolean isGroupNotFullyVisible(NotificationGroup notificationGroup) { 443 return notificationGroup.summary == null 444 || notificationGroup.summary.row.getClipTopAmount() > 0 445 || notificationGroup.summary.row.getTranslationY() < 0; 446 } 447 setHeadsUpManager(HeadsUpManager headsUpManager)448 public void setHeadsUpManager(HeadsUpManager headsUpManager) { 449 mHeadsUpManager = headsUpManager; 450 } 451 dump(FileDescriptor fd, PrintWriter pw, String[] args)452 public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { 453 pw.println("GroupManager state:"); 454 pw.println(" number of groups: " + mGroupMap.size()); 455 for (Map.Entry<String, NotificationGroup> entry : mGroupMap.entrySet()) { 456 pw.println("\n key: " + entry.getKey()); pw.println(entry.getValue()); 457 } 458 pw.println("\n isolated entries: " + mIsolatedEntries.size()); 459 for (Map.Entry<String, StatusBarNotification> entry : mIsolatedEntries.entrySet()) { 460 pw.print(" "); pw.print(entry.getKey()); 461 pw.print(", "); pw.println(entry.getValue()); 462 } 463 } 464 465 public static class NotificationGroup { 466 public final HashSet<NotificationData.Entry> children = new HashSet<>(); 467 public NotificationData.Entry summary; 468 public boolean expanded; 469 /** 470 * Is this notification group suppressed, i.e its summary is hidden 471 */ 472 public boolean suppressed; 473 474 @Override toString()475 public String toString() { 476 String result = " summary:\n " 477 + (summary != null ? summary.notification : "null"); 478 result += "\n children size: " + children.size(); 479 for (NotificationData.Entry child : children) { 480 result += "\n " + child.notification; 481 } 482 return result; 483 } 484 } 485 486 public interface OnGroupChangeListener { 487 /** 488 * The expansion of a group has changed. 489 * 490 * @param changedRow the row for which the expansion has changed, which is also the summary 491 * @param expanded a boolean indicating the new expanded state 492 */ onGroupExpansionChanged(ExpandableNotificationRow changedRow, boolean expanded)493 void onGroupExpansionChanged(ExpandableNotificationRow changedRow, boolean expanded); 494 495 /** 496 * A group of children just received a summary notification and should therefore become 497 * children of it. 498 * 499 * @param group the group created 500 */ onGroupCreatedFromChildren(NotificationGroup group)501 void onGroupCreatedFromChildren(NotificationGroup group); 502 503 /** 504 * The groups have changed. This can happen if the isolation of a child has changes or if a 505 * group became suppressed / unsuppressed 506 */ onGroupsChanged()507 void onGroupsChanged(); 508 } 509 } 510