Home | History | Annotate | Download | only in phone
      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 
     48     public void setOnGroupChangeListener(OnGroupChangeListener listener) {
     49         mListener = listener;
     50     }
     51 
     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 
     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 
     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 
     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      */
     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 
    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 
    138     private void onEntryBecomingChild(NotificationData.Entry entry) {
    139         if (entry.row.isHeadsUp()) {
    140             onHeadsUpStateChanged(entry, true);
    141         }
    142     }
    143 
    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 
    164     private boolean hasIsolatedChildren(NotificationGroup group) {
    165         return getNumberOfIsolatedChildren(group.summary.notification.getGroupKey()) != 0;
    166     }
    167 
    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 
    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 
    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 
    211     public boolean isSummaryOfSuppressedGroup(StatusBarNotification sbn) {
    212         return isGroupSuppressed(getGroupKey(sbn)) && sbn.getNotification().isGroupSummary();
    213     }
    214 
    215     private boolean isOnlyChild(StatusBarNotification sbn) {
    216         return !sbn.getNotification().isGroupSummary()
    217                 && getTotalNumberOfChildren(sbn) == 1;
    218     }
    219 
    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 
    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 
    236     private boolean isGroupSuppressed(String groupKey) {
    237         NotificationGroup group = mGroupMap.get(groupKey);
    238         return group != null && group.suppressed;
    239     }
    240 
    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 
    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      */
    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      */
    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      */
    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      */
    312     public ExpandableNotificationRow getLogicalGroupSummary(
    313             StatusBarNotification sbn) {
    314         return getGroupSummary(sbn.getGroupKey());
    315     }
    316 
    317     @Nullable
    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. */
    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 
    335     private boolean isIsolated(StatusBarNotification sbn) {
    336         return mIsolatedEntries.containsKey(sbn.getKey());
    337     }
    338 
    339     private boolean isGroupSummary(StatusBarNotification sbn) {
    340         if (isIsolated(sbn)) {
    341             return true;
    342         }
    343         return sbn.getNotification().isGroupSummary();
    344     }
    345 
    346     private boolean isGroupChild(StatusBarNotification sbn) {
    347         if (isIsolated(sbn)) {
    348             return false;
    349         }
    350         return sbn.isGroup() && !sbn.getNotification().isGroupSummary();
    351     }
    352 
    353     private String getGroupKey(StatusBarNotification sbn) {
    354         if (isIsolated(sbn)) {
    355             return sbn.getKey();
    356         }
    357         return sbn.getGroupKey();
    358     }
    359 
    360     @Override
    361     public void onHeadsUpPinnedModeChanged(boolean inPinnedMode) {
    362     }
    363 
    364     @Override
    365     public void onHeadsUpPinned(ExpandableNotificationRow headsUp) {
    366     }
    367 
    368     @Override
    369     public void onHeadsUpUnPinned(ExpandableNotificationRow headsUp) {
    370     }
    371 
    372     @Override
    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 
    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 
    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 
    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 
    448     public void setHeadsUpManager(HeadsUpManager headsUpManager) {
    449         mHeadsUpManager = headsUpManager;
    450     }
    451 
    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
    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          */
    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          */
    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          */
    507         void onGroupsChanged();
    508     }
    509 }
    510