1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License
15  */
16 
17 package com.android.systemui.statusbar;
18 
19 import android.app.Notification;
20 import android.graphics.PorterDuff;
21 import android.graphics.drawable.Icon;
22 import android.text.TextUtils;
23 import android.view.NotificationHeaderView;
24 import android.view.View;
25 import android.widget.ImageView;
26 import android.widget.TextView;
27 
28 import java.util.ArrayList;
29 import java.util.HashSet;
30 import java.util.List;
31 
32 /**
33  * A Util to manage {@link android.view.NotificationHeaderView} objects and their redundancies.
34  */
35 public class NotificationHeaderUtil {
36 
37     private static final TextViewComparator sTextViewComparator = new TextViewComparator();
38     private static final VisibilityApplicator sVisibilityApplicator = new VisibilityApplicator();
39     private static  final DataExtractor sIconExtractor = new DataExtractor() {
40         @Override
41         public Object extractData(ExpandableNotificationRow row) {
42             return row.getStatusBarNotification().getNotification();
43         }
44     };
45     private static final IconComparator sIconVisibilityComparator = new IconComparator() {
46         public boolean compare(View parent, View child, Object parentData,
47                 Object childData) {
48             return hasSameIcon(parentData, childData)
49                     && hasSameColor(parentData, childData);
50         }
51     };
52     private static final IconComparator sGreyComparator = new IconComparator() {
53         public boolean compare(View parent, View child, Object parentData,
54                 Object childData) {
55             return !hasSameIcon(parentData, childData)
56                     || hasSameColor(parentData, childData);
57         }
58     };
59     private final static ResultApplicator mGreyApplicator = new ResultApplicator() {
60         @Override
61         public void apply(View view, boolean apply) {
62             NotificationHeaderView header = (NotificationHeaderView) view;
63             ImageView icon = (ImageView) view.findViewById(
64                     com.android.internal.R.id.icon);
65             ImageView expand = (ImageView) view.findViewById(
66                     com.android.internal.R.id.expand_button);
67             applyToChild(icon, apply, header.getOriginalIconColor());
68             applyToChild(expand, apply, header.getOriginalNotificationColor());
69         }
70 
71         private void applyToChild(View view, boolean shouldApply, int originalColor) {
72             if (originalColor != NotificationHeaderView.NO_COLOR) {
73                 ImageView imageView = (ImageView) view;
74                 imageView.getDrawable().mutate();
75                 if (shouldApply) {
76                     // lets gray it out
77                     int grey = view.getContext().getColor(
78                             com.android.internal.R.color.notification_icon_default_color);
79                     imageView.getDrawable().setColorFilter(grey, PorterDuff.Mode.SRC_ATOP);
80                 } else {
81                     // lets reset it
82                     imageView.getDrawable().setColorFilter(originalColor,
83                             PorterDuff.Mode.SRC_ATOP);
84                 }
85             }
86         }
87     };
88 
89     private final ExpandableNotificationRow mRow;
90     private final ArrayList<HeaderProcessor> mComparators = new ArrayList<>();
91     private final HashSet<Integer> mDividers = new HashSet<>();
92 
NotificationHeaderUtil(ExpandableNotificationRow row)93     public NotificationHeaderUtil(ExpandableNotificationRow row) {
94         mRow = row;
95         // To hide the icons if they are the same and the color is the same
96         mComparators.add(new HeaderProcessor(mRow,
97                 com.android.internal.R.id.icon,
98                 sIconExtractor,
99                 sIconVisibilityComparator,
100                 sVisibilityApplicator));
101         // To grey them out the icons and expand button when the icons are not the same
102         mComparators.add(new HeaderProcessor(mRow,
103                 com.android.internal.R.id.notification_header,
104                 sIconExtractor,
105                 sGreyComparator,
106                 mGreyApplicator));
107         mComparators.add(new HeaderProcessor(mRow,
108                 com.android.internal.R.id.profile_badge,
109                 null /* Extractor */,
110                 new ViewComparator() {
111                     @Override
112                     public boolean compare(View parent, View child, Object parentData,
113                             Object childData) {
114                         return parent.getVisibility() != View.GONE;
115                     }
116 
117                     @Override
118                     public boolean isEmpty(View view) {
119                         if (view instanceof ImageView) {
120                             return ((ImageView) view).getDrawable() == null;
121                         }
122                         return false;
123                     }
124                 },
125                 sVisibilityApplicator));
126         mComparators.add(HeaderProcessor.forTextView(mRow,
127                 com.android.internal.R.id.app_name_text));
128         mComparators.add(HeaderProcessor.forTextView(mRow,
129                 com.android.internal.R.id.header_text));
130         mDividers.add(com.android.internal.R.id.header_text_divider);
131         mDividers.add(com.android.internal.R.id.time_divider);
132     }
133 
updateChildrenHeaderAppearance()134     public void updateChildrenHeaderAppearance() {
135         List<ExpandableNotificationRow> notificationChildren = mRow.getNotificationChildren();
136         if (notificationChildren == null) {
137             return;
138         }
139         // Initialize the comparators
140         for (int compI = 0; compI < mComparators.size(); compI++) {
141             mComparators.get(compI).init();
142         }
143 
144         // Compare all notification headers
145         for (int i = 0; i < notificationChildren.size(); i++) {
146             ExpandableNotificationRow row = notificationChildren.get(i);
147             for (int compI = 0; compI < mComparators.size(); compI++) {
148                 mComparators.get(compI).compareToHeader(row);
149             }
150         }
151 
152         // Apply the comparison to the row
153         for (int i = 0; i < notificationChildren.size(); i++) {
154             ExpandableNotificationRow row = notificationChildren.get(i);
155             for (int compI = 0; compI < mComparators.size(); compI++) {
156                 mComparators.get(compI).apply(row);
157             }
158             // We need to sanitize the dividers since they might be off-balance now
159             sanitizeHeaderViews(row);
160         }
161     }
162 
sanitizeHeaderViews(ExpandableNotificationRow row)163     private void sanitizeHeaderViews(ExpandableNotificationRow row) {
164         if (row.isSummaryWithChildren()) {
165             sanitizeHeader(row.getNotificationHeader());
166             return;
167         }
168         final NotificationContentView layout = row.getPrivateLayout();
169         sanitizeChild(layout.getContractedChild());
170         sanitizeChild(layout.getHeadsUpChild());
171         sanitizeChild(layout.getExpandedChild());
172     }
173 
sanitizeChild(View child)174     private void sanitizeChild(View child) {
175         if (child != null) {
176             NotificationHeaderView header = (NotificationHeaderView) child.findViewById(
177                     com.android.internal.R.id.notification_header);
178             sanitizeHeader(header);
179         }
180     }
181 
sanitizeHeader(NotificationHeaderView rowHeader)182     private void sanitizeHeader(NotificationHeaderView rowHeader) {
183         if (rowHeader == null) {
184             return;
185         }
186         final int childCount = rowHeader.getChildCount();
187         View time = rowHeader.findViewById(com.android.internal.R.id.time);
188         boolean hasVisibleText = false;
189         for (int i = 1; i < childCount - 1 ; i++) {
190             View child = rowHeader.getChildAt(i);
191             if (child instanceof TextView
192                     && child.getVisibility() != View.GONE
193                     && !mDividers.contains(Integer.valueOf(child.getId()))
194                     && child != time) {
195                 hasVisibleText = true;
196                 break;
197             }
198         }
199         // in case no view is visible we make sure the time is visible
200         int timeVisibility = !hasVisibleText
201                 || mRow.getStatusBarNotification().getNotification().showsTime()
202                 ? View.VISIBLE : View.GONE;
203         time.setVisibility(timeVisibility);
204         View left = null;
205         View right;
206         for (int i = 1; i < childCount - 1 ; i++) {
207             View child = rowHeader.getChildAt(i);
208             if (mDividers.contains(Integer.valueOf(child.getId()))) {
209                 boolean visible = false;
210                 // Lets find the item to the right
211                 for (i++; i < childCount - 1; i++) {
212                     right = rowHeader.getChildAt(i);
213                     if (mDividers.contains(Integer.valueOf(right.getId()))) {
214                         // A divider was found, this needs to be hidden
215                         i--;
216                         break;
217                     } else if (right.getVisibility() != View.GONE && right instanceof TextView) {
218                         visible = left != null;
219                         left = right;
220                         break;
221                     }
222                 }
223                 child.setVisibility(visible ? View.VISIBLE : View.GONE);
224             } else if (child.getVisibility() != View.GONE && child instanceof TextView) {
225                 left = child;
226             }
227         }
228     }
229 
restoreNotificationHeader(ExpandableNotificationRow row)230     public void restoreNotificationHeader(ExpandableNotificationRow row) {
231         for (int compI = 0; compI < mComparators.size(); compI++) {
232             mComparators.get(compI).apply(row, true /* reset */);
233         }
234         sanitizeHeaderViews(row);
235     }
236 
237     private static class HeaderProcessor {
238         private final int mId;
239         private final DataExtractor mExtractor;
240         private final ResultApplicator mApplicator;
241         private final ExpandableNotificationRow mParentRow;
242         private boolean mApply;
243         private View mParentView;
244         private ViewComparator mComparator;
245         private Object mParentData;
246 
forTextView(ExpandableNotificationRow row, int id)247         public static HeaderProcessor forTextView(ExpandableNotificationRow row, int id) {
248             return new HeaderProcessor(row, id, null, sTextViewComparator, sVisibilityApplicator);
249         }
250 
HeaderProcessor(ExpandableNotificationRow row, int id, DataExtractor extractor, ViewComparator comparator, ResultApplicator applicator)251         HeaderProcessor(ExpandableNotificationRow row, int id, DataExtractor extractor,
252                 ViewComparator comparator,
253                 ResultApplicator applicator) {
254             mId = id;
255             mExtractor = extractor;
256             mApplicator = applicator;
257             mComparator = comparator;
258             mParentRow = row;
259         }
260 
init()261         public void init() {
262             mParentView = mParentRow.getNotificationHeader().findViewById(mId);
263             mParentData = mExtractor == null ? null : mExtractor.extractData(mParentRow);
264             mApply = !mComparator.isEmpty(mParentView);
265         }
compareToHeader(ExpandableNotificationRow row)266         public void compareToHeader(ExpandableNotificationRow row) {
267             if (!mApply) {
268                 return;
269             }
270             NotificationHeaderView header = row.getNotificationHeader();
271             if (header == null) {
272                 mApply = false;
273                 return;
274             }
275             Object childData = mExtractor == null ? null : mExtractor.extractData(row);
276             mApply = mComparator.compare(mParentView, header.findViewById(mId),
277                     mParentData, childData);
278         }
279 
apply(ExpandableNotificationRow row)280         public void apply(ExpandableNotificationRow row) {
281             apply(row, false /* reset */);
282         }
283 
apply(ExpandableNotificationRow row, boolean reset)284         public void apply(ExpandableNotificationRow row, boolean reset) {
285             boolean apply = mApply && !reset;
286             if (row.isSummaryWithChildren()) {
287                 applyToView(apply, row.getNotificationHeader());
288                 return;
289             }
290             applyToView(apply, row.getPrivateLayout().getContractedChild());
291             applyToView(apply, row.getPrivateLayout().getHeadsUpChild());
292             applyToView(apply, row.getPrivateLayout().getExpandedChild());
293         }
294 
applyToView(boolean apply, View parent)295         private void applyToView(boolean apply, View parent) {
296             if (parent != null) {
297                 View view = parent.findViewById(mId);
298                 if (view != null && !mComparator.isEmpty(view)) {
299                     mApplicator.apply(view, apply);
300                 }
301             }
302         }
303     }
304 
305     private interface ViewComparator {
306         /**
307          * @param parent the parent view
308          * @param child the child view
309          * @param parentData optional data for the parent
310          * @param childData optional data for the child
311          * @return whether to views are the same
312          */
compare(View parent, View child, Object parentData, Object childData)313         boolean compare(View parent, View child, Object parentData, Object childData);
isEmpty(View view)314         boolean isEmpty(View view);
315     }
316 
317     private interface DataExtractor {
extractData(ExpandableNotificationRow row)318         Object extractData(ExpandableNotificationRow row);
319     }
320 
321     private static class TextViewComparator implements ViewComparator {
322         @Override
compare(View parent, View child, Object parentData, Object childData)323         public boolean compare(View parent, View child, Object parentData, Object childData) {
324             TextView parentView = (TextView) parent;
325             TextView childView = (TextView) child;
326             return parentView.getText().equals(childView.getText());
327         }
328 
329         @Override
isEmpty(View view)330         public boolean isEmpty(View view) {
331             return TextUtils.isEmpty(((TextView) view).getText());
332         }
333     }
334 
335     private static abstract class IconComparator implements ViewComparator {
336         @Override
compare(View parent, View child, Object parentData, Object childData)337         public boolean compare(View parent, View child, Object parentData, Object childData) {
338             return false;
339         }
340 
hasSameIcon(Object parentData, Object childData)341         protected boolean hasSameIcon(Object parentData, Object childData) {
342             Icon parentIcon = ((Notification) parentData).getSmallIcon();
343             Icon childIcon = ((Notification) childData).getSmallIcon();
344             return parentIcon.sameAs(childIcon);
345         }
346 
347         /**
348          * @return whether two ImageViews have the same colorFilterSet or none at all
349          */
hasSameColor(Object parentData, Object childData)350         protected boolean hasSameColor(Object parentData, Object childData) {
351             int parentColor = ((Notification) parentData).color;
352             int childColor = ((Notification) childData).color;
353             return parentColor == childColor;
354         }
355 
356         @Override
isEmpty(View view)357         public boolean isEmpty(View view) {
358             return false;
359         }
360     }
361 
362     private interface ResultApplicator {
apply(View view, boolean apply)363         void apply(View view, boolean apply);
364     }
365 
366     private static class VisibilityApplicator implements ResultApplicator {
367 
368         @Override
apply(View view, boolean apply)369         public void apply(View view, boolean apply) {
370             view.setVisibility(apply ? View.GONE : View.VISIBLE);
371         }
372     }
373 }
374