NotificationGroupManager.java revision 698b1706be196de41fc6bf03cf15971c82a11949
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;
2625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.io.FileDescriptor;
2852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.io.PrintWriter;
29c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinekimport java.util.ArrayList;
3025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport java.util.HashMap;
3125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinekimport java.util.HashSet;
32967ed2a151c9b349734892a2274304acd1bd373cSelim Cinekimport java.util.Iterator;
3352941c5618d79c0fb7af655f47f558d956af28c8Selim Cinekimport java.util.Map;
34e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynoldsimport java.util.Objects;
3525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
3625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek/**
3725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek * A class to handle notifications and their corresponding groups.
3825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek */
39ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinekpublic class NotificationGroupManager implements HeadsUpManager.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;
4625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
4725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void setOnGroupChangeListener(OnGroupChangeListener listener) {
4825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        mListener = listener;
4925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
5025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
5125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public boolean isGroupExpanded(StatusBarNotification sbn) {
52ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
5325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
5425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
5525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
5625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return group.expanded;
5725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
5825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
5925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void setGroupExpanded(StatusBarNotification sbn, boolean expanded) {
60ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
6125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
6225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return;
6325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
6425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        setGroupExpanded(group, expanded);
6525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
6625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
6725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private void setGroupExpanded(NotificationGroup group, boolean expanded) {
6825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        group.expanded = expanded;
6925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group.summary != null) {
7025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            mListener.onGroupExpansionChanged(group.summary.row, expanded);
7125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
7225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
7325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
7425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void onEntryRemoved(NotificationData.Entry removed) {
7525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        onEntryRemovedInternal(removed, removed.notification);
7652941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        mIsolatedEntries.remove(removed.key);
7725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
7825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
7925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    /**
8025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * An entry was removed.
8125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     *
8225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @param removed the removed entry
8325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @param sbn the notification the entry has, which doesn't need to be the same as it's internal
8425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     *            notification
8525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     */
8625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    private void onEntryRemovedInternal(NotificationData.Entry removed,
8725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            final StatusBarNotification sbn) {
88ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        String groupKey = getGroupKey(sbn);
8925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        final NotificationGroup group = mGroupMap.get(groupKey);
900b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek        if (group == null) {
910b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // When an app posts 2 different notifications as summary of the same group, then a
920b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // cancellation of the first notification removes this group.
930b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // This situation is not supported and we will not allow such notifications anymore in
940b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            // the close future. See b/23676310 for reference.
950b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek            return;
960b4aeab281d0bd18e67f245eeccbbc468f3065f1Selim Cinek        }
97ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isGroupChild(sbn)) {
9825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group.children.remove(removed);
99e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek        } else {
100e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek            group.summary = null;
10125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
1022a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        updateSuppression(group);
10325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group.children.isEmpty()) {
10425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            if (group.summary == null) {
10525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                mGroupMap.remove(groupKey);
10625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            }
10725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
10825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
10925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
110ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onEntryAdded(final NotificationData.Entry added) {
111ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        final StatusBarNotification sbn = added.notification;
112ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        boolean isGroupChild = isGroupChild(sbn);
113ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        String groupKey = getGroupKey(sbn);
11425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
11525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        if (group == null) {
11625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group = new NotificationGroup();
11725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            mGroupMap.put(groupKey, group);
11825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
119ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isGroupChild) {
120e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek            group.children.add(added);
1212a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
122e73ad216d322d0e7002d1ce2e59caf23030dbf5bSelim Cinek        } else {
12325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            group.summary = added;
124b5605e58cb8080c8c887b1885336b707596c8094Selim Cinek            group.expanded = added.row.areChildrenExpanded();
1252a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
12625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            if (!group.children.isEmpty()) {
12750e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                HashSet<NotificationData.Entry> childrenCopy =
12850e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                        (HashSet<NotificationData.Entry>) group.children.clone();
12950e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                for (NotificationData.Entry child : childrenCopy) {
13050e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                    onEntryBecomingChild(child);
13150e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek                }
13225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                mListener.onGroupCreatedFromChildren(group);
13325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            }
13425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
13525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
13625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
13750e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek    private void onEntryBecomingChild(NotificationData.Entry entry) {
13850e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek        if (entry.row.isHeadsUp()) {
13950e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek            onHeadsUpStateChanged(entry, true);
14050e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek        }
14150e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek    }
14250e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek
143e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds    public void onEntryBundlingUpdated(final NotificationData.Entry updated,
144e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds            final String overrideGroupKey) {
145e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        final StatusBarNotification oldSbn = updated.notification.clone();
146e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        if (!Objects.equals(oldSbn.getOverrideGroupKey(), overrideGroupKey)) {
147e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds            updated.notification.setOverrideGroupKey(overrideGroupKey);
148e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds            onEntryUpdated(updated, oldSbn);
149e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        }
150e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds    }
151e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds
1522a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    private void updateSuppression(NotificationGroup group) {
15323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        if (group == null) {
15423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            return;
15523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
1562a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        boolean prevSuppressed = group.suppressed;
15723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        group.suppressed = group.summary != null && !group.expanded
15823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                && (group.children.size() == 1
15923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                || (group.children.size() == 0
160e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds                        && group.summary.notification.getNotification().isGroupSummary()
16123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                        && hasIsolatedChildren(group)));
1622a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        if (prevSuppressed != group.suppressed) {
163967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (group.suppressed) {
164967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                handleSuppressedSummaryHeadsUpped(group.summary);
165967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
1662a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            mListener.onGroupsChanged();
1672a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        }
1682a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    }
1692a7393410b6390831143dca198438a4e58bdf88aSelim Cinek
17023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private boolean hasIsolatedChildren(NotificationGroup group) {
17123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getNumberOfIsolatedChildren(group.summary.notification.getGroupKey()) != 0;
17223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
17323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
17423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private int getNumberOfIsolatedChildren(String groupKey) {
17523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        int count = 0;
176a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        for (StatusBarNotification sbn : mIsolatedEntries.values()) {
17723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) {
17823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                count++;
17923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            }
18023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
18123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return count;
18223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
18323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
184967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    private NotificationData.Entry getIsolatedChild(String groupKey) {
185967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        for (StatusBarNotification sbn : mIsolatedEntries.values()) {
186967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (sbn.getGroupKey().equals(groupKey) && isIsolated(sbn)) {
187967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                return mGroupMap.get(sbn.getKey()).summary;
188967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
189967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
190967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        return null;
191967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
192967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
19325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public void onEntryUpdated(NotificationData.Entry entry,
19425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            StatusBarNotification oldNotification) {
195ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (mGroupMap.get(getGroupKey(oldNotification)) != null) {
19625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            onEntryRemovedInternal(entry, oldNotification);
19725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
19825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        onEntryAdded(entry);
199a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        if (isIsolated(entry.notification)) {
200a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            mIsolatedEntries.put(entry.key, entry.notification);
20123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            String oldKey = oldNotification.getGroupKey();
20223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            String newKey = entry.notification.getGroupKey();
20323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            if (!oldKey.equals(newKey)) {
20423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(oldKey));
20523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(newKey));
20623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            }
20750e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek        } else if (!isGroupChild(oldNotification) && isGroupChild(entry.notification)) {
20850e7467eb6688a37d7877aaf2c9389a7b4a62abcSelim Cinek            onEntryBecomingChild(entry);
20923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        }
21025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
21125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2122a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    public boolean isSummaryOfSuppressedGroup(StatusBarNotification sbn) {
21323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return isGroupSuppressed(getGroupKey(sbn)) && sbn.getNotification().isGroupSummary();
21425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
21525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
21623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    public boolean isOnlyChildInSuppressedGroup(StatusBarNotification sbn) {
21723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return isGroupSuppressed(sbn.getGroupKey())
21853c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek                && isOnlyChild(sbn);
21936b02233fa8a5121e9fabaf5ac447c7ff4bb20a8Selim Cinek    }
22036b02233fa8a5121e9fabaf5ac447c7ff4bb20a8Selim Cinek
22153c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    private boolean isOnlyChild(StatusBarNotification sbn) {
22236b02233fa8a5121e9fabaf5ac447c7ff4bb20a8Selim Cinek        return !sbn.getNotification().isGroupSummary()
22323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                && getTotalNumberOfChildren(sbn) == 1;
2242a7393410b6390831143dca198438a4e58bdf88aSelim Cinek    }
2252a7393410b6390831143dca198438a4e58bdf88aSelim Cinek
22653c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    public boolean isOnlyChildInGroup(StatusBarNotification sbn) {
22753c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        return isOnlyChild(sbn) && getLogicalGroupSummary(sbn) != null;
22853c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek    }
22953c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek
23023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private int getTotalNumberOfChildren(StatusBarNotification sbn) {
23153c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        int isolatedChildren = getNumberOfIsolatedChildren(sbn.getGroupKey());
23253c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        NotificationGroup group = mGroupMap.get(sbn.getGroupKey());
23353c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        int realChildren = group != null ? group.children.size() : 0;
23453c4d3d23e70496c377139ffa9397aae463d2a5eSelim Cinek        return isolatedChildren + realChildren;
23523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
23623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
23723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private boolean isGroupSuppressed(String groupKey) {
23823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
2392a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        return group != null && group.suppressed;
24025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
24125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
2429c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek    public void setStatusBarState(int newState) {
2439c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        if (mBarState == newState) {
2449c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek            return;
2459c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        }
2469c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        mBarState = newState;
2479c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        if (mBarState == StatusBarState.KEYGUARD) {
2489184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek            collapseAllGroups();
2499184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek        }
2509184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek    }
2519184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek
2529184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek    public void collapseAllGroups() {
253c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        // Because notifications can become isolated when the group becomes suppressed it can
254c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        // lead to concurrent modifications while looping. We need to make a copy.
255c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        ArrayList<NotificationGroup> groupCopy = new ArrayList<>(mGroupMap.values());
256c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        int size = groupCopy.size();
257c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek        for (int i = 0; i < size; i++) {
258c0b14b0e895d65ab428d5c05778aae37ee946e19Selim Cinek            NotificationGroup group =  groupCopy.get(i);
2599184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek            if (group.expanded) {
2609184f9c4cdff27c6eea47c885d95bad2859b5eb3Selim Cinek                setGroupExpanded(group, false);
2619c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek            }
2622a7393410b6390831143dca198438a4e58bdf88aSelim Cinek            updateSuppression(group);
2639c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek        }
2649c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek    }
2659c4c41461d98edf622f79c0bd4b2a8af8cfa2de8Selim Cinek
26625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    /**
26725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     * @return whether a given notification is a child in a group which has a summary
26825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek     */
26925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public boolean isChildInGroupWithSummary(StatusBarNotification sbn) {
270ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (!isGroupChild(sbn)) {
27125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
27225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
273ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
2742a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        if (group == null || group.summary == null || group.suppressed) {
27525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek            return false;
27625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        }
2773f19f60d654421eee5b35a92129081f08c977629Selim Cinek        if (group.children.isEmpty()) {
2783f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // If the suppression of a group changes because the last child was removed, this can
2793f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // still be called temporarily because the child hasn't been fully removed yet. Let's
2803f19f60d654421eee5b35a92129081f08c977629Selim Cinek            // make sure we still return false in that case.
2813f19f60d654421eee5b35a92129081f08c977629Selim Cinek            return false;
2823f19f60d654421eee5b35a92129081f08c977629Selim Cinek        }
28325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return true;
28425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
28525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
286263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    /**
287263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek     * @return whether a given notification is a summary in a group which has children
288263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek     */
289263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    public boolean isSummaryOfGroup(StatusBarNotification sbn) {
290ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (!isGroupSummary(sbn)) {
291263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek            return false;
292263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        }
293ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
294263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        if (group == null) {
295263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek            return false;
296263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        }
297263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek        return !group.children.isEmpty();
298263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek    }
299263398f0175efc8bc8c965473f9565a693a0a0e0Selim Cinek
30023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    /**
30123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * Get the summary of a specified status bar notification. For isolated notification this return
30223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * itself.
30323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     */
30425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public ExpandableNotificationRow getGroupSummary(StatusBarNotification sbn) {
30523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getGroupSummary(getGroupKey(sbn));
30623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
30723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
30823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    /**
30923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * Similar to {@link #getGroupSummary(StatusBarNotification)} but doesn't get the visual summary
31023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * but the logical summary, i.e when a child is isolated, it still returns the summary as if
31123c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     * it wasn't isolated.
31223c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek     */
31323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    public ExpandableNotificationRow getLogicalGroupSummary(
31423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek            StatusBarNotification sbn) {
31523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        return getGroupSummary(sbn.getGroupKey());
31623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    }
31723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek
31823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    @Nullable
31923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek    private ExpandableNotificationRow getGroupSummary(String groupKey) {
32023c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek        NotificationGroup group = mGroupMap.get(groupKey);
32125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        return group == null ? null
32225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek                : group.summary == null ? null
32323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                        : group.summary.row;
32425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
32525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
326698b1706be196de41fc6bf03cf15971c82a11949Chris Wren    /** @return group expansion state after toggling. */
327698b1706be196de41fc6bf03cf15971c82a11949Chris Wren    public boolean toggleGroupExpansion(StatusBarNotification sbn) {
328ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        NotificationGroup group = mGroupMap.get(getGroupKey(sbn));
32983bc78338437a460076a4b5778ded38440ac3501Selim Cinek        if (group == null) {
330698b1706be196de41fc6bf03cf15971c82a11949Chris Wren            return false;
33183bc78338437a460076a4b5778ded38440ac3501Selim Cinek        }
33283bc78338437a460076a4b5778ded38440ac3501Selim Cinek        setGroupExpanded(group, !group.expanded);
333698b1706be196de41fc6bf03cf15971c82a11949Chris Wren        return group.expanded;
33483bc78338437a460076a4b5778ded38440ac3501Selim Cinek    }
33583bc78338437a460076a4b5778ded38440ac3501Selim Cinek
336ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isIsolated(StatusBarNotification sbn) {
337a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        return mIsolatedEntries.containsKey(sbn.getKey());
338ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
339ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
340ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isGroupSummary(StatusBarNotification sbn) {
341ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
342ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return true;
343ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
344ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        return sbn.getNotification().isGroupSummary();
345ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
346e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds
347ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private boolean isGroupChild(StatusBarNotification sbn) {
348ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
349ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return false;
350ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
351e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        return sbn.isGroup() && !sbn.getNotification().isGroupSummary();
352ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
353ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
354ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    private String getGroupKey(StatusBarNotification sbn) {
355ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (isIsolated(sbn)) {
356ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            return sbn.getKey();
357ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
358ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        return sbn.getGroupKey();
359ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
360ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
361ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
362ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpPinnedModeChanged(boolean inPinnedMode) {
363ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
364ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
365ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
366ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpPinned(ExpandableNotificationRow headsUp) {
367ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
368ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
369ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
370ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpUnPinned(ExpandableNotificationRow headsUp) {
371ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
372ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
373ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    @Override
374ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    public void onHeadsUpStateChanged(NotificationData.Entry entry, boolean isHeadsUp) {
375ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        final StatusBarNotification sbn = entry.notification;
376ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        if (entry.row.isHeadsUp()) {
377a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            if (shouldIsolate(sbn)) {
37823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // We will be isolated now, so lets update the groups
37923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                onEntryRemovedInternal(entry, entry.notification);
380a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
381a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mIsolatedEntries.put(sbn.getKey(), sbn);
382a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
38323c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                onEntryAdded(entry);
38423c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // We also need to update the suppression of the old group, because this call comes
38523c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // even before the groupManager knows about the notification at all.
38623c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // When the notification gets added afterwards it is already isolated and therefore
38723c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                // it doesn't lead to an update.
38823c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                updateSuppression(mGroupMap.get(entry.notification.getGroupKey()));
38923c80348d29e4a28968cc8800181d088c6ca0436Selim Cinek                mListener.onGroupsChanged();
390967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            } else {
391967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                handleSuppressedSummaryHeadsUpped(entry);
392ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            }
393ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        } else {
394a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek            if (mIsolatedEntries.containsKey(sbn.getKey())) {
395a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                // not isolated anymore, we need to update the groups
396a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                onEntryRemovedInternal(entry, entry.notification);
397a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mIsolatedEntries.remove(sbn.getKey());
398a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                onEntryAdded(entry);
399a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                mListener.onGroupsChanged();
400ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek            }
401ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        }
402ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek    }
403ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
404967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    private void handleSuppressedSummaryHeadsUpped(NotificationData.Entry entry) {
405967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        StatusBarNotification sbn = entry.notification;
406967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        if (!isGroupSuppressed(sbn.getGroupKey())
407967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                || !sbn.getNotification().isGroupSummary()
408967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                || !entry.row.isHeadsUp()) {
409967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            return;
410967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
411967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        // The parent of a suppressed group got huned, lets hun the child!
412967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey());
413967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        if (notificationGroup != null) {
414967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            Iterator<NotificationData.Entry> iterator = notificationGroup.children.iterator();
415967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            NotificationData.Entry child = iterator.hasNext() ? iterator.next() : null;
416967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (child == null) {
417967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                child = getIsolatedChild(sbn.getGroupKey());
418967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
419967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            if (child != null) {
420967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                if (mHeadsUpManager.isHeadsUp(child.key)) {
421967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                    mHeadsUpManager.updateNotification(child, true);
422967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                } else {
423967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                    mHeadsUpManager.showNotification(child);
424967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek                }
425967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek            }
426967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        }
427967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        mHeadsUpManager.releaseImmediately(entry.key);
428967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
429967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
430a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    private boolean shouldIsolate(StatusBarNotification sbn) {
431a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        NotificationGroup notificationGroup = mGroupMap.get(sbn.getGroupKey());
432e46bb37acf6d3cfb9974672ace93f5381f70ad99Julia Reynolds        return (sbn.isGroup() && !sbn.getNotification().isGroupSummary())
433a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                && (sbn.getNotification().fullScreenIntent != null
434a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || notificationGroup == null
435a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || !notificationGroup.expanded
436a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                        || isGroupNotFullyVisible(notificationGroup));
437a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    }
438a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
439a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    private boolean isGroupNotFullyVisible(NotificationGroup notificationGroup) {
440a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek        return notificationGroup.summary == null
441a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                || notificationGroup.summary.row.getClipTopAmount() > 0
442a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek                || notificationGroup.summary.row.getTranslationY() < 0;
443a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek    }
444a6c0cef0a0cf9895d9241cb3293a7355c3e8af4aSelim Cinek
445967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    public void setHeadsUpManager(HeadsUpManager headsUpManager) {
446967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek        mHeadsUpManager = headsUpManager;
447967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek    }
448967ed2a151c9b349734892a2274304acd1bd373cSelim Cinek
44952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek    public void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
45052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("GroupManager state:");
45152941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("  number of groups: " +  mGroupMap.size());
45252941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        for (Map.Entry<String, NotificationGroup>  entry : mGroupMap.entrySet()) {
45352941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.println("\n    key: " + entry.getKey()); pw.println(entry.getValue());
45452941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
45552941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        pw.println("\n    isolated entries: " +  mIsolatedEntries.size());
45652941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        for (Map.Entry<String, StatusBarNotification> entry : mIsolatedEntries.entrySet()) {
45752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.print("      "); pw.print(entry.getKey());
45852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            pw.print(", "); pw.println(entry.getValue());
45952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
46052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek    }
46152941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek
46225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public static class NotificationGroup {
46325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public final HashSet<NotificationData.Entry> children = new HashSet<>();
46425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public NotificationData.Entry summary;
46525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        public boolean expanded;
4662a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        /**
4672a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * Is this notification group suppressed, i.e its summary is hidden
4682a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         */
4692a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        public boolean suppressed;
47052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek
47152941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        @Override
47252941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        public String toString() {
47360ca7879ec1fa7164c6fed95413d40313423fabbSelim Cinek            String result = "    summary:\n      "
47460ca7879ec1fa7164c6fed95413d40313423fabbSelim Cinek                    + (summary != null ? summary.notification : "null");
47552941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            result += "\n    children size: " + children.size();
47652941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            for (NotificationData.Entry child : children) {
47752941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek                result += "\n      " + child.notification;
47852941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            }
47952941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek            return result;
48052941c5618d79c0fb7af655f47f558d956af28c8Selim Cinek        }
48125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
48225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
48325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    public interface OnGroupChangeListener {
48425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        /**
48525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * The expansion of a group has changed.
48625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         *
48725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param changedRow the row for which the expansion has changed, which is also the summary
48825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param expanded a boolean indicating the new expanded state
48925fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         */
49025fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        void onGroupExpansionChanged(ExpandableNotificationRow changedRow, boolean expanded);
49125fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek
49225fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        /**
49325fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * A group of children just received a summary notification and should therefore become
49425fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * children of it.
49525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         *
49625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         * @param group the group created
49725fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek         */
49825fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek        void onGroupCreatedFromChildren(NotificationGroup group);
499ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek
500ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek        /**
5012a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * The groups have changed. This can happen if the isolation of a child has changes or if a
5022a7393410b6390831143dca198438a4e58bdf88aSelim Cinek         * group became suppressed / unsuppressed
503ef5127ea5f34f7a4c961021f6b691174bcb81d2eSelim Cinek         */
5042a7393410b6390831143dca198438a4e58bdf88aSelim Cinek        void onGroupsChanged();
50525fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek    }
50625fd4e2be731fe893685faa48828d8fa4526cb1aSelim Cinek}
507