125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek/*
225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * Copyright (C) 2015 The Android Open Source Project
325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek *
425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * Licensed under the Apache License, Version 2.0 (the "License");
525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * you may not use this file except in compliance with the License.
625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * You may obtain a copy of the License at
725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek *
825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek *      http://www.apache.org/licenses/LICENSE-2.0
925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek *
1025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * Unless required by applicable law or agreed to in writing, software
1125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * distributed under the License is distributed on an "AS IS" BASIS,
1225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * See the License for the specific language governing permissions and
1425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * limitations under the License
1525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek */
1625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
1725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekpackage com.android.systemui.statusbar.phone;
1825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
1925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport android.service.notification.StatusBarNotification;
2023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinekimport android.support.annotation.Nullable;
2125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport com.android.systemui.statusbar.ExpandableNotificationRow;
2325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport com.android.systemui.statusbar.NotificationData;
249c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinekimport com.android.systemui.statusbar.StatusBarState;
25ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinekimport com.android.systemui.statusbar.policy.HeadsUpManager;
26add9526b1dd91360d0413296f2dbcb2053480162Selim Cinekimport com.android.systemui.statusbar.policy.OnHeadsUpChangedListener;
2725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.io.FileDescriptor;
2952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.io.PrintWriter;
30c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinekimport java.util.ArrayList;
3125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport java.util.HashMap;
3225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport java.util.HashSet;
33967ed2a151c9b349734892a2274304acd1bd373cSelim Cinekimport java.util.Iterator;
3452941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.util.Map;
3525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
3625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek/**
3725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * A class to handle notifications and their corresponding groups.
3825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek */
39add9526b1dd91360d0413296f2dbcb2053480162Selim Cinekpublic class NotificationGroupManager implements OnHeadsUpChangedListener {
4025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
4125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private final HashMap<String, NotificationGroup> mGroupMap = new HashMap<>();
4225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private OnGroupChangeListener mListener;
4325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private int mBarState = -1;
44a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    private HashMap<String, StatusBarNotification> mIsolatedEntries = new HashMap<>();
45967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    private HeadsUpManager mHeadsUpManager;
4680c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek    private boolean mIsUpdatingUnchangedGroup;
4725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
4825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void setOnGroupChangeListener(OnGroupChangeListener listener) {
4925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        mListener = listener;
5025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
5125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
5225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public boolean isGroupExpanded(StatusBarNotification sbn) {
53ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
5425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
5525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
5625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
5725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return group.expanded;
5825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
5925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
6025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void setGroupExpanded(StatusBarNotification sbn, boolean expanded) {
61ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
6225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
6325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return;
6425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
6525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        setGroupExpanded(group, expanded);
6625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
6725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
6825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private void setGroupExpanded(NotificationGroup group, boolean expanded) {
6925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        group.expanded = expanded;
7025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group.summary != null) {
7125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            mListener.onGroupExpansionChanged(group.summary.row, expanded);
7225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
7325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
7425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
7525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void onEntryRemoved(NotificationData.Entry removed) {
7625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        onEntryRemovedInternal(removed, removed.notification);
7752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        mIsolatedEntries.remove(removed.key);
7825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
7925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
8025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    /**
8125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * An entry was removed.
8225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     *
8325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @param removed the removed entry
8425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @param sbn the notification the entry has, which doesn't need to be the same as it's internal
8525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     *            notification
8625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     */
8725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private void onEntryRemovedInternal(NotificationData.Entry removed,
8825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            final StatusBarNotification sbn) {
89ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        String groupKey = getGroupKey(sbn);
9025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        final NotificationGroup group = mGroupMap.get(groupKey);
910b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek        if (group == null) {
920b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // When an app posts 2 different notifications as summary of the same group, then a
930b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // cancellation of the first notification removes this group.
940b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // This situation is not supported and we will not allow such notifications anymore in
950b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // the close future. See b/23676310 for reference.
960b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            return;
970b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek        }
98ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isGroupChild(sbn)) {
9925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group.children.remove(removed);
100e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek        } else {
101e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek            group.summary = null;
10225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
1032a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        updateSuppression(group);
10425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group.children.isEmpty()) {
10525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            if (group.summary == null) {
10625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                mGroupMap.remove(groupKey);
10725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            }
10825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
10925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
11025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
111ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onEntryAdded(final NotificationData.Entry added) {
112ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        final StatusBarNotification sbn = added.notification;
113ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        boolean isGroupChild = isGroupChild(sbn);
114ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        String groupKey = getGroupKey(sbn);
11525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
11625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
11725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group = new NotificationGroup();
11825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            mGroupMap.put(groupKey, group);
11925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
120ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isGroupChild) {
121e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek            group.children.add(added);
1222a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
123e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek        } else {
12425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group.summary = added;
125b5605e58cb8080c8c887b1885336b707596c8094Selim Cinek            group.expanded = added.row.areChildrenExpanded();
1262a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
12725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            if (!group.children.isEmpty()) {
12850e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                HashSet<NotificationData.Entry> childrenCopy =
12950e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                        (HashSet<NotificationData.Entry>) group.children.clone();
13050e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                for (NotificationData.Entry child : childrenCopy) {
13150e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                    onEntryBecomingChild(child);
13250e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                }
13325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                mListener.onGroupCreatedFromChildren(group);
13425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            }
13525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
13625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
13725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
13850e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek    private void onEntryBecomingChild(NotificationData.Entry entry) {
13950e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek        if (entry.row.isHeadsUp()) {
14050e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek            onHeadsUpStateChanged(entry, true);
14150e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek        }
14250e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek    }
14350e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek
1442a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    private void updateSuppression(NotificationGroup group) {
14580c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek        if (group == null) {
14623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            return;
14723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
1482a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        boolean prevSuppressed = group.suppressed;
14923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        group.suppressed = group.summary != null && !group.expanded
15023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                && (group.children.size() == 1
15123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                || (group.children.size() == 0
152e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds                        && group.summary.notification.getNotification().isGroupSummary()
15323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                        && hasIsolatedChildren(group)));
1542a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        if (prevSuppressed != group.suppressed) {
155967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (group.suppressed) {
156967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                handleSuppressedSummaryHeadsUpped(group.summary);
157967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
15880c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek            if (!mIsUpdatingUnchangedGroup) {
15980c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek                mListener.onGroupsChanged();
16080c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek            }
1612a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        }
1622a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    }
1632a7393410b6390831143dca198438a4e58bdf88aSelim Cinek
16423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private boolean hasIsolatedChildren(NotificationGroup group) {
16523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getNumberOfIsolatedChildren(group.summary.notification.getGroupKey()) != 0;
16623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
16723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
16823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private int getNumberOfIsolatedChildren(String groupKey) {
16923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        int count = 0;
170a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        for (StatusBarNotification sbn : mIsolatedEntries.values()) {
17123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) {
17223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                count++;
17323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            }
17423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
17523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return count;
17623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
17723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
178967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    private NotificationData.Entry getIsolatedChild(String groupKey) {
179967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        for (StatusBarNotification sbn : mIsolatedEntries.values()) {
180967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) {
181967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                return mGroupMap.get(sbn.getKey()).summary;
182967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
183967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
184967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        return null;
185967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
186967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
18725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void onEntryUpdated(NotificationData.Entry entry,
18825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            StatusBarNotification oldNotification) {
18968bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        String oldKey = oldNotification.getGroupKey();
19068bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        String newKey = entry.notification.getGroupKey();
19168bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        boolean groupKeysChanged = !oldKey.equals(newKey);
19268bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        boolean wasGroupChild = isGroupChild(oldNotification);
19368bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        boolean isGroupChild = isGroupChild(entry.notification);
19480c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek        mIsUpdatingUnchangedGroup = !groupKeysChanged && wasGroupChild == isGroupChild;
195ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (mGroupMap.get(getGroupKey(oldNotification)) != null) {
19625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            onEntryRemovedInternal(entry, oldNotification);
19725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
19825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        onEntryAdded(entry);
19980c44dd800d2ef03446152495b7ef3c486f1be90Selim Cinek        mIsUpdatingUnchangedGroup = false;
200a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        if (isIsolated(entry.notification)) {
201a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            mIsolatedEntries.put(entry.key, entry.notification);
20268bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek            if (groupKeysChanged) {
20323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(oldKey));
20423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(newKey));
20523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            }
20668bdff16a8dc3bb049c2330025395a0b981d1d18Selim Cinek        } else if (!wasGroupChild && isGroupChild) {
20750e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek            onEntryBecomingChild(entry);
20823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
20925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
21025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2112a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    public boolean isSummaryOfSuppressedGroup(StatusBarNotification sbn) {
21223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return isGroupSuppressed(getGroupKey(sbn)) && sbn.getNotification().isGroupSummary();
21325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
21425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
21553c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    private boolean isOnlyChild(StatusBarNotification sbn) {
21636b02233fa8a5121e9fabaf5ac447c7ff4bb20a8Selim Cinek        return !sbn.getNotification().isGroupSummary()
21723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                && getTotalNumberOfChildren(sbn) == 1;
2182a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    }
2192a7393410b6390831143dca198438a4e58bdf88aSelim Cinek
22053c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    public boolean isOnlyChildInGroup(StatusBarNotification sbn) {
22188086e718340fdb869cea40b20ae1d747074bc43Selim Cinek        if (!isOnlyChild(sbn)) {
22288086e718340fdb869cea40b20ae1d747074bc43Selim Cinek            return false;
22388086e718340fdb869cea40b20ae1d747074bc43Selim Cinek        }
22488086e718340fdb869cea40b20ae1d747074bc43Selim Cinek        ExpandableNotificationRow logicalGroupSummary = getLogicalGroupSummary(sbn);
22588086e718340fdb869cea40b20ae1d747074bc43Selim Cinek        return logicalGroupSummary != null
22688086e718340fdb869cea40b20ae1d747074bc43Selim Cinek                && !logicalGroupSummary.getStatusBarNotification().equals(sbn);
22753c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    }
22853c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek
22923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private int getTotalNumberOfChildren(StatusBarNotification sbn) {
23053c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        int isolatedChildren = getNumberOfIsolatedChildren(sbn.getGroupKey());
23153c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        NotificationGroup group = mGroupMap.get(sbn.getGroupKey());
23253c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        int realChildren = group != null ? group.children.size() : 0;
23353c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        return isolatedChildren + realChildren;
23423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
23523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
23623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private boolean isGroupSuppressed(String groupKey) {
23723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
2382a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        return group != null && group.suppressed;
23925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
24025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2419c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek    public void setStatusBarState(int newState) {
2429c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        if (mBarState == newState) {
2439c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek            return;
2449c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        }
2459c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        mBarState = newState;
2469c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        if (mBarState == StatusBarState.KEYGUARD) {
2479184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek            collapseAllGroups();
2489184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek        }
2499184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek    }
2509184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek
2519184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek    public void collapseAllGroups() {
252c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        // Because notifications can become isolated when the group becomes suppressed it can
253c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        // lead to concurrent modifications while looping. We need to make a copy.
254c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        ArrayList<NotificationGroup> groupCopy = new ArrayList<>(mGroupMap.values());
255c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        int size = groupCopy.size();
256c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        for (int i = 0; i < size; i++) {
257c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek            NotificationGroup group =  groupCopy.get(i);
2589184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek            if (group.expanded) {
2599184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek                setGroupExpanded(group, false);
2609c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek            }
2612a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
2629c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        }
2639c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek    }
2649c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek
26525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    /**
26625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @return whether a given notification is a child in a group which has a summary
26725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     */
26825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public boolean isChildInGroupWithSummary(StatusBarNotification sbn) {
269ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (!isGroupChild(sbn)) {
27025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
27125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
272ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
2732a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        if (group == null || group.summary == null || group.suppressed) {
27425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
27525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
2763f19f60d654421eee5b35a92129081f08c977629Selim Cinek        if (group.children.isEmpty()) {
2773f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // If the suppression of a group changes because the last child was removed, this can
2783f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // still be called temporarily because the child hasn't been fully removed yet. Let's
2793f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // make sure we still return false in that case.
2803f19f60d654421eee5b35a92129081f08c977629Selim Cinek            return false;
2813f19f60d654421eee5b35a92129081f08c977629Selim Cinek        }
28225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return true;
28325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
28425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
285263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    /**
286263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek     * @return whether a given notification is a summary in a group which has children
287263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek     */
288263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    public boolean isSummaryOfGroup(StatusBarNotification sbn) {
289ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (!isGroupSummary(sbn)) {
290263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek            return false;
291263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        }
292ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
293263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        if (group == null) {
294263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek            return false;
295263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        }
296263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        return !group.children.isEmpty();
297263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    }
298263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek
29923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    /**
30023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * Get the summary of a specified status bar notification. For isolated notification this return
30123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * itself.
30223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     */
30325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public ExpandableNotificationRow getGroupSummary(StatusBarNotification sbn) {
30423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getGroupSummary(getGroupKey(sbn));
30523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
30623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
30723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    /**
30823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * Similar to {@link #getGroupSummary(StatusBarNotification)} but doesn't get the visual summary
30923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * but the logical summary, i.e when a child is isolated, it still returns the summary as if
31023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * it wasn't isolated.
31123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     */
31223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    public ExpandableNotificationRow getLogicalGroupSummary(
31323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            StatusBarNotification sbn) {
31423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getGroupSummary(sbn.getGroupKey());
31523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
31623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
31723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    @Nullable
31823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private ExpandableNotificationRow getGroupSummary(String groupKey) {
31923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
32025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return group == null ? null
32125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                : group.summary == null ? null
32223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                        : group.summary.row;
32325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
32425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
325698b1706be196de41fc6bf03cf15971c82a11949Chris Wren    /** @return group expansion state after toggling. */
326698b1706be196de41fc6bf03cf15971c82a11949Chris Wren    public boolean toggleGroupExpansion(StatusBarNotification sbn) {
327ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
32883bc78338437a460076a4b5778ded38440ac3501Selim Cinek        if (group == null) {
329698b1706be196de41fc6bf03cf15971c82a11949Chris Wren            return false;
33083bc78338437a460076a4b5778ded38440ac3501Selim Cinek        }
33183bc78338437a460076a4b5778ded38440ac3501Selim Cinek        setGroupExpanded(group, !group.expanded);
332698b1706be196de41fc6bf03cf15971c82a11949Chris Wren        return group.expanded;
33383bc78338437a460076a4b5778ded38440ac3501Selim Cinek    }
33483bc78338437a460076a4b5778ded38440ac3501Selim Cinek
335ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isIsolated(StatusBarNotification sbn) {
336a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        return mIsolatedEntries.containsKey(sbn.getKey());
337ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
338ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
339ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isGroupSummary(StatusBarNotification sbn) {
340ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
341ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return true;
342ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
343ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        return sbn.getNotification().isGroupSummary();
344ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
345e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds
346ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isGroupChild(StatusBarNotification sbn) {
347ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
348ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return false;
349ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
350e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        return sbn.isGroup() && !sbn.getNotification().isGroupSummary();
351ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
352ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
353ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private String getGroupKey(StatusBarNotification sbn) {
354ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
355ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return sbn.getKey();
356ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
357ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        return sbn.getGroupKey();
358ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
359ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
360ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
361ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpPinnedModeChanged(boolean inPinnedMode) {
362ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
363ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
364ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
365ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpPinned(ExpandableNotificationRow headsUp) {
366ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
367ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
368ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
369ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpUnPinned(ExpandableNotificationRow headsUp) {
370ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
371ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
372ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
373ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpStateChanged(NotificationData.Entry entry, boolean isHeadsUp) {
374ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        final StatusBarNotification sbn = entry.notification;
375ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (entry.row.isHeadsUp()) {
376a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            if (shouldIsolate(sbn)) {
37723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // We will be isolated now, so lets update the groups
37823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                onEntryRemovedInternal(entry, entry.notification);
379a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
380a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mIsolatedEntries.put(sbn.getKey(), sbn);
381a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
38223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                onEntryAdded(entry);
38323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // We also need to update the suppression of the old group, because this call comes
38423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // even before the groupManager knows about the notification at all.
38523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // When the notification gets added afterwards it is already isolated and therefore
38623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // it doesn't lead to an update.
38723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(entry.notification.getGroupKey()));
38823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                mListener.onGroupsChanged();
389967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            } else {
390967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                handleSuppressedSummaryHeadsUpped(entry);
391ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            }
392ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        } else {
393a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            if (mIsolatedEntries.containsKey(sbn.getKey())) {
394a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                // not isolated anymore, we need to update the groups
395a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                onEntryRemovedInternal(entry, entry.notification);
396a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mIsolatedEntries.remove(sbn.getKey());
397a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                onEntryAdded(entry);
398a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mListener.onGroupsChanged();
399ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            }
400ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
401ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
402ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
403967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    private void handleSuppressedSummaryHeadsUpped(NotificationData.Entry entry) {
404967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        StatusBarNotification sbn = entry.notification;
405967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        if (!isGroupSuppressed(sbn.getGroupKey())
406967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                || !sbn.getNotification().isGroupSummary()
407967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                || !entry.row.isHeadsUp()) {
408967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            return;
409967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
410967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        // The parent of a suppressed group got huned, lets hun the child!
411967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey());
412967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        if (notificationGroup != null) {
413967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            Iterator<NotificationData.Entry> iterator = notificationGroup.children.iterator();
414967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            NotificationData.Entry child = iterator.hasNext() ? iterator.next() : null;
415967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (child == null) {
416967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                child = getIsolatedChild(sbn.getGroupKey());
417967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
418967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (child != null) {
419967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                if (mHeadsUpManager.isHeadsUp(child.key)) {
420967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                    mHeadsUpManager.updateNotification(child, true);
421967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                } else {
422967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                    mHeadsUpManager.showNotification(child);
423967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                }
424967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
425967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
426967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        mHeadsUpManager.releaseImmediately(entry.key);
427967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
428967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
429a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    private boolean shouldIsolate(StatusBarNotification sbn) {
430a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey());
431e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        return (sbn.isGroup() && !sbn.getNotification().isGroupSummary())
432a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                && (sbn.getNotification().fullScreenIntent != null
433a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || notificationGroup == null
434a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || !notificationGroup.expanded
435a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || isGroupNotFullyVisible(notificationGroup));
436a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    }
437a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
438a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    private boolean isGroupNotFullyVisible(NotificationGroup notificationGroup) {
439a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        return notificationGroup.summary == null
440a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                || notificationGroup.summary.row.getClipTopAmount() > 0
441a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                || notificationGroup.summary.row.getTranslationY() < 0;
442a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    }
443a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
444967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    public void setHeadsUpManager(HeadsUpManager headsUpManager) {
445967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        mHeadsUpManager = headsUpManager;
446967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
447967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
44852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek    public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
44952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("GroupManager state:");
45052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("  number of groups: " +  mGroupMap.size());
45152941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        for (Map.Entry<String, NotificationGroup>  entry : mGroupMap.entrySet()) {
45252941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.println("\n    key: " + entry.getKey()); pw.println(entry.getValue());
45352941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
45452941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("\n    isolated entries: " +  mIsolatedEntries.size());
45552941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        for (Map.Entry<String, StatusBarNotification> entry : mIsolatedEntries.entrySet()) {
45652941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.print("      "); pw.print(entry.getKey());
45752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.print(", "); pw.println(entry.getValue());
45852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
45952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek    }
46052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek
46125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public static class NotificationGroup {
46225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public final HashSet<NotificationData.Entry> children = new HashSet<>();
46325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public NotificationData.Entry summary;
46425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public boolean expanded;
4652a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        /**
4662a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * Is this notification group suppressed, i.e its summary is hidden
4672a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         */
4682a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        public boolean suppressed;
46952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek
47052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        @Override
47152941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        public String toString() {
47260ca7879ec1fa7164c6fed95413d40313423fabbSelim Cinek            String result = "    summary:\n      "
47360ca7879ec1fa7164c6fed95413d40313423fabbSelim Cinek                    + (summary != null ? summary.notification : "null");
47452941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            result += "\n    children size: " + children.size();
47552941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            for (NotificationData.Entry child : children) {
47652941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek                result += "\n      " + child.notification;
47752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            }
47852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            return result;
47952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
48025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
48125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
48225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public interface OnGroupChangeListener {
48325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        /**
48425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * The expansion of a group has changed.
48525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         *
48625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param changedRow the row for which the expansion has changed, which is also the summary
48725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param expanded a boolean indicating the new expanded state
48825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         */
48925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        void onGroupExpansionChanged(ExpandableNotificationRow changedRow, boolean expanded);
49025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
49125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        /**
49225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * A group of children just received a summary notification and should therefore become
49325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * children of it.
49425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         *
49525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param group the group created
49625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         */
49725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        void onGroupCreatedFromChildren(NotificationGroup group);
498ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
499ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        /**
5002a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * The groups have changed. This can happen if the isolation of a child has changes or if a
5012a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * group became suppressed / unsuppressed
502ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek         */
5032a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        void onGroupsChanged();
50425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
50525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek}
506