1 /*
2  * Copyright (C) 2017 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.keyguard;
18 
19 import android.animation.LayoutTransition;
20 import android.animation.ObjectAnimator;
21 import android.animation.PropertyValuesHolder;
22 import android.annotation.ColorInt;
23 import android.app.PendingIntent;
24 import android.arch.lifecycle.LiveData;
25 import android.arch.lifecycle.Observer;
26 import android.content.Context;
27 import android.graphics.Color;
28 import android.graphics.drawable.Drawable;
29 import android.net.Uri;
30 import android.provider.Settings;
31 import android.text.Layout;
32 import android.text.TextUtils;
33 import android.text.TextUtils.TruncateAt;
34 import android.util.AttributeSet;
35 import android.util.Log;
36 import android.view.View;
37 import android.view.ViewGroup;
38 import android.view.animation.Animation;
39 import android.widget.Button;
40 import android.widget.LinearLayout;
41 import android.widget.TextView;
42 
43 import com.android.internal.annotations.VisibleForTesting;
44 import com.android.internal.graphics.ColorUtils;
45 import com.android.settingslib.Utils;
46 import com.android.systemui.Dependency;
47 import com.android.systemui.Interpolators;
48 import com.android.systemui.R;
49 import com.android.systemui.keyguard.KeyguardSliceProvider;
50 import com.android.systemui.statusbar.AlphaOptimizedTextView;
51 import com.android.systemui.statusbar.policy.ConfigurationController;
52 import com.android.systemui.tuner.TunerService;
53 import com.android.systemui.util.wakelock.KeepAwakeAnimationListener;
54 
55 import java.util.ArrayList;
56 import java.util.HashMap;
57 import java.util.List;
58 import java.util.function.Consumer;
59 
60 import androidx.slice.Slice;
61 import androidx.slice.SliceItem;
62 import androidx.slice.SliceViewManager;
63 import androidx.slice.core.SliceQuery;
64 import androidx.slice.widget.ListContent;
65 import androidx.slice.widget.RowContent;
66 import androidx.slice.widget.SliceLiveData;
67 
68 /**
69  * View visible under the clock on the lock screen and AoD.
70  */
71 public class KeyguardSliceView extends LinearLayout implements View.OnClickListener,
72         Observer<Slice>, TunerService.Tunable, ConfigurationController.ConfigurationListener {
73 
74     private static final String TAG = "KeyguardSliceView";
75     public static final int DEFAULT_ANIM_DURATION = 550;
76 
77     private final HashMap<View, PendingIntent> mClickActions;
78     private Uri mKeyguardSliceUri;
79     @VisibleForTesting
80     TextView mTitle;
81     private Row mRow;
82     private int mTextColor;
83     private float mDarkAmount = 0;
84 
85     private LiveData<Slice> mLiveData;
86     private int mIconSize;
87     /**
88      * Runnable called whenever the view contents change.
89      */
90     private Runnable mContentChangeListener;
91     private boolean mHasHeader;
92     private Slice mSlice;
93     private boolean mPulsing;
94 
KeyguardSliceView(Context context)95     public KeyguardSliceView(Context context) {
96         this(context, null, 0);
97     }
98 
KeyguardSliceView(Context context, AttributeSet attrs)99     public KeyguardSliceView(Context context, AttributeSet attrs) {
100         this(context, attrs, 0);
101     }
102 
KeyguardSliceView(Context context, AttributeSet attrs, int defStyle)103     public KeyguardSliceView(Context context, AttributeSet attrs, int defStyle) {
104         super(context, attrs, defStyle);
105 
106         TunerService tunerService = Dependency.get(TunerService.class);
107         tunerService.addTunable(this, Settings.Secure.KEYGUARD_SLICE_URI);
108 
109         mClickActions = new HashMap<>();
110 
111         LayoutTransition transition = new LayoutTransition();
112         transition.setStagger(LayoutTransition.CHANGE_APPEARING, DEFAULT_ANIM_DURATION / 2);
113         transition.setDuration(LayoutTransition.APPEARING, DEFAULT_ANIM_DURATION);
114         transition.setDuration(LayoutTransition.DISAPPEARING, DEFAULT_ANIM_DURATION / 2);
115         transition.disableTransitionType(LayoutTransition.CHANGE_APPEARING);
116         transition.disableTransitionType(LayoutTransition.CHANGE_DISAPPEARING);
117         transition.setInterpolator(LayoutTransition.APPEARING, Interpolators.FAST_OUT_SLOW_IN);
118         transition.setInterpolator(LayoutTransition.DISAPPEARING, Interpolators.ALPHA_OUT);
119         transition.setAnimateParentHierarchy(false);
120         transition.addTransitionListener(new SliceViewTransitionListener());
121         setLayoutTransition(transition);
122     }
123 
124     @Override
onFinishInflate()125     protected void onFinishInflate() {
126         super.onFinishInflate();
127         mTitle = findViewById(R.id.title);
128         mRow = findViewById(R.id.row);
129         mTextColor = Utils.getColorAttr(mContext, R.attr.wallpaperTextColor);
130     }
131 
132     @Override
onAttachedToWindow()133     protected void onAttachedToWindow() {
134         super.onAttachedToWindow();
135 
136         // Make sure we always have the most current slice
137         mLiveData.observeForever(this);
138         Dependency.get(ConfigurationController.class).addCallback(this);
139     }
140 
141     @Override
onDetachedFromWindow()142     protected void onDetachedFromWindow() {
143         super.onDetachedFromWindow();
144 
145         mLiveData.removeObserver(this);
146         Dependency.get(ConfigurationController.class).removeCallback(this);
147     }
148 
showSlice()149     private void showSlice() {
150         if (mPulsing || mSlice == null) {
151             mTitle.setVisibility(GONE);
152             mRow.setVisibility(GONE);
153             if (mContentChangeListener != null) {
154                 mContentChangeListener.run();
155             }
156             return;
157         }
158 
159         ListContent lc = new ListContent(getContext(), mSlice);
160         mHasHeader = lc.hasHeader();
161         List<SliceItem> subItems = new ArrayList<SliceItem>();
162         for (int i = 0; i < lc.getRowItems().size(); i++) {
163             SliceItem subItem = lc.getRowItems().get(i);
164             String itemUri = subItem.getSlice().getUri().toString();
165             // Filter out the action row
166             if (!KeyguardSliceProvider.KEYGUARD_ACTION_URI.equals(itemUri)) {
167                 subItems.add(subItem);
168             }
169         }
170         if (!mHasHeader) {
171             mTitle.setVisibility(GONE);
172         } else {
173             mTitle.setVisibility(VISIBLE);
174 
175             // If there's a header it'll be the first subitem
176             RowContent header = new RowContent(getContext(), subItems.get(0),
177                     true /* showStartItem */);
178             SliceItem mainTitle = header.getTitleItem();
179             CharSequence title = mainTitle != null ? mainTitle.getText() : null;
180             mTitle.setText(title);
181         }
182 
183         mClickActions.clear();
184         final int subItemsCount = subItems.size();
185         final int blendedColor = getTextColor();
186         final int startIndex = mHasHeader ? 1 : 0; // First item is header; skip it
187         mRow.setVisibility(subItemsCount > 0 ? VISIBLE : GONE);
188         for (int i = startIndex; i < subItemsCount; i++) {
189             SliceItem item = subItems.get(i);
190             RowContent rc = new RowContent(getContext(), item, true /* showStartItem */);
191             final Uri itemTag = item.getSlice().getUri();
192             // Try to reuse the view if already exists in the layout
193             KeyguardSliceButton button = mRow.findViewWithTag(itemTag);
194             if (button == null) {
195                 button = new KeyguardSliceButton(mContext);
196                 button.setTextColor(blendedColor);
197                 button.setTag(itemTag);
198                 final int viewIndex = i - (mHasHeader ? 1 : 0);
199                 mRow.addView(button, viewIndex);
200             }
201 
202             PendingIntent pendingIntent = null;
203             if (rc.getPrimaryAction() != null) {
204                 pendingIntent = rc.getPrimaryAction().getAction();
205             }
206             mClickActions.put(button, pendingIntent);
207 
208             final SliceItem titleItem = rc.getTitleItem();
209             button.setText(titleItem == null ? null : titleItem.getText());
210             button.setContentDescription(rc.getContentDescription());
211 
212             Drawable iconDrawable = null;
213             SliceItem icon = SliceQuery.find(item.getSlice(),
214                     android.app.slice.SliceItem.FORMAT_IMAGE);
215             if (icon != null) {
216                 iconDrawable = icon.getIcon().loadDrawable(mContext);
217                 final int width = (int) (iconDrawable.getIntrinsicWidth()
218                         / (float) iconDrawable.getIntrinsicHeight() * mIconSize);
219                 iconDrawable.setBounds(0, 0, Math.max(width, 1), mIconSize);
220             }
221             button.setCompoundDrawables(iconDrawable, null, null, null);
222             button.setOnClickListener(this);
223             button.setClickable(pendingIntent != null);
224         }
225 
226         // Removing old views
227         for (int i = 0; i < mRow.getChildCount(); i++) {
228             View child = mRow.getChildAt(i);
229             if (!mClickActions.containsKey(child)) {
230                 mRow.removeView(child);
231                 i--;
232             }
233         }
234 
235         if (mContentChangeListener != null) {
236             mContentChangeListener.run();
237         }
238     }
239 
setPulsing(boolean pulsing, boolean animate)240     public void setPulsing(boolean pulsing, boolean animate) {
241         mPulsing = pulsing;
242         LayoutTransition transition = getLayoutTransition();
243         if (!animate) {
244             setLayoutTransition(null);
245         }
246         showSlice();
247         if (!animate) {
248             setLayoutTransition(transition);
249         }
250     }
251 
252     /**
253      * Breaks a string in 2 lines where both have similar character count
254      * but first line is always longer.
255      *
256      * @param charSequence Original text.
257      * @return Optimal string.
258      */
findBestLineBreak(CharSequence charSequence)259     private static CharSequence findBestLineBreak(CharSequence charSequence) {
260         if (TextUtils.isEmpty(charSequence)) {
261             return charSequence;
262         }
263 
264         String source = charSequence.toString();
265         // Ignore if there is only 1 word,
266         // or if line breaks were manually set.
267         if (source.contains("\n") || !source.contains(" ")) {
268             return source;
269         }
270 
271         final String[] words = source.split(" ");
272         final StringBuilder optimalString = new StringBuilder(source.length());
273         int current = 0;
274         while (optimalString.length() < source.length() - optimalString.length()) {
275             optimalString.append(words[current]);
276             if (current < words.length - 1) {
277                 optimalString.append(" ");
278             }
279             current++;
280         }
281         optimalString.append("\n");
282         for (int i = current; i < words.length; i++) {
283             optimalString.append(words[i]);
284             if (current < words.length - 1) {
285                 optimalString.append(" ");
286             }
287         }
288 
289         return optimalString.toString();
290     }
291 
setDarkAmount(float darkAmount)292     public void setDarkAmount(float darkAmount) {
293         mDarkAmount = darkAmount;
294         mRow.setDarkAmount(darkAmount);
295         updateTextColors();
296     }
297 
updateTextColors()298     private void updateTextColors() {
299         final int blendedColor = getTextColor();
300         mTitle.setTextColor(blendedColor);
301         int childCount = mRow.getChildCount();
302         for (int i = 0; i < childCount; i++) {
303             View v = mRow.getChildAt(i);
304             if (v instanceof Button) {
305                 ((Button) v).setTextColor(blendedColor);
306             }
307         }
308     }
309 
310     @Override
onClick(View v)311     public void onClick(View v) {
312         final PendingIntent action = mClickActions.get(v);
313         if (action != null) {
314             try {
315                 action.send();
316             } catch (PendingIntent.CanceledException e) {
317                 Log.i(TAG, "Pending intent cancelled, nothing to launch", e);
318             }
319         }
320     }
321 
322     /**
323      * Runnable that gets invoked every time the title or the row visibility changes.
324      * @param contentChangeListener The listener.
325      */
setContentChangeListener(Runnable contentChangeListener)326     public void setContentChangeListener(Runnable contentChangeListener) {
327         mContentChangeListener = contentChangeListener;
328     }
329 
hasHeader()330     public boolean hasHeader() {
331         return mHasHeader;
332     }
333 
334     /**
335      * LiveData observer lifecycle.
336      * @param slice the new slice content.
337      */
338     @Override
onChanged(Slice slice)339     public void onChanged(Slice slice) {
340         mSlice = slice;
341         showSlice();
342     }
343 
344     @Override
onTuningChanged(String key, String newValue)345     public void onTuningChanged(String key, String newValue) {
346         setupUri(newValue);
347     }
348 
setupUri(String uriString)349     public void setupUri(String uriString) {
350         if (uriString == null) {
351             uriString = KeyguardSliceProvider.KEYGUARD_SLICE_URI;
352         }
353 
354         boolean wasObserving = false;
355         if (mLiveData != null && mLiveData.hasActiveObservers()) {
356             wasObserving = true;
357             mLiveData.removeObserver(this);
358         }
359 
360         mKeyguardSliceUri = Uri.parse(uriString);
361         mLiveData = SliceLiveData.fromUri(mContext, mKeyguardSliceUri);
362 
363         if (wasObserving) {
364             mLiveData.observeForever(this);
365         }
366     }
367 
368     @VisibleForTesting
getTextColor()369     int getTextColor() {
370         return ColorUtils.blendARGB(mTextColor, Color.WHITE, mDarkAmount);
371     }
372 
373     @VisibleForTesting
setTextColor(@olorInt int textColor)374     void setTextColor(@ColorInt int textColor) {
375         mTextColor = textColor;
376         updateTextColors();
377     }
378 
379     @Override
onDensityOrFontScaleChanged()380     public void onDensityOrFontScaleChanged() {
381         mIconSize = mContext.getResources().getDimensionPixelSize(R.dimen.widget_icon_size);
382     }
383 
refresh()384     public void refresh() {
385         Slice slice = SliceViewManager.getInstance(getContext()).bindSlice(mKeyguardSliceUri);
386         onChanged(slice);
387     }
388 
389     public static class Row extends LinearLayout {
390 
391         /**
392          * This view is visible in AOD, which means that the device will sleep if we
393          * don't hold a wake lock. We want to enter doze only after all views have reached
394          * their desired positions.
395          */
396         private final Animation.AnimationListener mKeepAwakeListener;
397         private float mDarkAmount;
398 
Row(Context context)399         public Row(Context context) {
400             this(context, null);
401         }
402 
Row(Context context, AttributeSet attrs)403         public Row(Context context, AttributeSet attrs) {
404             this(context, attrs, 0);
405         }
406 
Row(Context context, AttributeSet attrs, int defStyleAttr)407         public Row(Context context, AttributeSet attrs, int defStyleAttr) {
408             this(context, attrs, defStyleAttr, 0);
409         }
410 
Row(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)411         public Row(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
412             super(context, attrs, defStyleAttr, defStyleRes);
413             mKeepAwakeListener = new KeepAwakeAnimationListener(mContext);
414         }
415 
416         @Override
onFinishInflate()417         protected void onFinishInflate() {
418             LayoutTransition transition = new LayoutTransition();
419             transition.setDuration(DEFAULT_ANIM_DURATION);
420 
421             PropertyValuesHolder left = PropertyValuesHolder.ofInt("left", 0, 1);
422             PropertyValuesHolder right = PropertyValuesHolder.ofInt("right", 0, 1);
423             ObjectAnimator changeAnimator = ObjectAnimator.ofPropertyValuesHolder((Object) null,
424                     left, right);
425             transition.setAnimator(LayoutTransition.CHANGE_APPEARING, changeAnimator);
426             transition.setAnimator(LayoutTransition.CHANGE_DISAPPEARING, changeAnimator);
427             transition.setInterpolator(LayoutTransition.CHANGE_APPEARING,
428                     Interpolators.ACCELERATE_DECELERATE);
429             transition.setInterpolator(LayoutTransition.CHANGE_DISAPPEARING,
430                     Interpolators.ACCELERATE_DECELERATE);
431             transition.setStartDelay(LayoutTransition.CHANGE_APPEARING, DEFAULT_ANIM_DURATION);
432             transition.setStartDelay(LayoutTransition.CHANGE_DISAPPEARING, DEFAULT_ANIM_DURATION);
433 
434             ObjectAnimator appearAnimator = ObjectAnimator.ofFloat(null, "alpha", 0f, 1f);
435             transition.setAnimator(LayoutTransition.APPEARING, appearAnimator);
436             transition.setInterpolator(LayoutTransition.APPEARING, Interpolators.ALPHA_IN);
437 
438             ObjectAnimator disappearAnimator = ObjectAnimator.ofFloat(null, "alpha", 1f, 0f);
439             transition.setInterpolator(LayoutTransition.DISAPPEARING, Interpolators.ALPHA_OUT);
440             transition.setDuration(LayoutTransition.DISAPPEARING, DEFAULT_ANIM_DURATION / 4);
441             transition.setAnimator(LayoutTransition.DISAPPEARING, disappearAnimator);
442 
443             transition.setAnimateParentHierarchy(false);
444             setLayoutTransition(transition);
445         }
446 
447         @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)448         protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
449             int width = MeasureSpec.getSize(widthMeasureSpec);
450             int childCount = getChildCount();
451             for (int i = 0; i < childCount; i++) {
452                 View child = getChildAt(i);
453                 if (child instanceof KeyguardSliceButton) {
454                     ((KeyguardSliceButton) child).setMaxWidth(width / childCount);
455                 }
456             }
457             super.onMeasure(widthMeasureSpec, heightMeasureSpec);
458         }
459 
setDarkAmount(float darkAmount)460         public void setDarkAmount(float darkAmount) {
461             boolean isAwake = darkAmount != 0;
462             boolean wasAwake = mDarkAmount != 0;
463             if (isAwake == wasAwake) {
464                 return;
465             }
466             mDarkAmount = darkAmount;
467             setLayoutAnimationListener(isAwake ? null : mKeepAwakeListener);
468         }
469 
470         @Override
hasOverlappingRendering()471         public boolean hasOverlappingRendering() {
472             return false;
473         }
474     }
475 
476     /**
477      * Representation of an item that appears under the clock on main keyguard message.
478      */
479     @VisibleForTesting
480     static class KeyguardSliceButton extends Button implements
481             ConfigurationController.ConfigurationListener {
482 
KeyguardSliceButton(Context context)483         public KeyguardSliceButton(Context context) {
484             super(context, null /* attrs */, 0 /* styleAttr */,
485                     com.android.keyguard.R.style.TextAppearance_Keyguard_Secondary);
486             onDensityOrFontScaleChanged();
487             setEllipsize(TruncateAt.END);
488         }
489 
490         @Override
onAttachedToWindow()491         protected void onAttachedToWindow() {
492             super.onAttachedToWindow();
493             Dependency.get(ConfigurationController.class).addCallback(this);
494         }
495 
496         @Override
onDetachedFromWindow()497         protected void onDetachedFromWindow() {
498             super.onDetachedFromWindow();
499             Dependency.get(ConfigurationController.class).removeCallback(this);
500         }
501 
502         @Override
onDensityOrFontScaleChanged()503         public void onDensityOrFontScaleChanged() {
504             updatePadding();
505         }
506 
507         @Override
setText(CharSequence text, BufferType type)508         public void setText(CharSequence text, BufferType type) {
509             super.setText(text, type);
510             updatePadding();
511         }
512 
updatePadding()513         private void updatePadding() {
514             boolean hasText = !TextUtils.isEmpty(getText());
515             int horizontalPadding = (int) getContext().getResources()
516                     .getDimension(R.dimen.widget_horizontal_padding) / 2;
517             setPadding(horizontalPadding, 0, horizontalPadding * (hasText ? 1 : -1), 0);
518             setCompoundDrawablePadding((int) mContext.getResources()
519                     .getDimension(R.dimen.widget_icon_padding));
520         }
521 
522         @Override
setTextColor(int color)523         public void setTextColor(int color) {
524             super.setTextColor(color);
525             updateDrawableColors();
526         }
527 
528         @Override
setCompoundDrawables(Drawable left, Drawable top, Drawable right, Drawable bottom)529         public void setCompoundDrawables(Drawable left, Drawable top, Drawable right,
530                 Drawable bottom) {
531             super.setCompoundDrawables(left, top, right, bottom);
532             updateDrawableColors();
533             updatePadding();
534         }
535 
updateDrawableColors()536         private void updateDrawableColors() {
537             final int color = getCurrentTextColor();
538             for (Drawable drawable : getCompoundDrawables()) {
539                 if (drawable != null) {
540                     drawable.setTint(color);
541                 }
542             }
543         }
544     }
545 
546     /**
547      * A text view that will split its contents in 2 lines when possible.
548      */
549     static class TitleView extends AlphaOptimizedTextView {
550 
TitleView(Context context)551         public TitleView(Context context) {
552             super(context);
553         }
554 
TitleView(Context context, AttributeSet attrs)555         public TitleView(Context context, AttributeSet attrs) {
556             super(context, attrs);
557         }
558 
TitleView(Context context, AttributeSet attrs, int defStyleAttr)559         public TitleView(Context context, AttributeSet attrs, int defStyleAttr) {
560             super(context, attrs, defStyleAttr);
561         }
562 
TitleView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)563         public TitleView(Context context, AttributeSet attrs, int defStyleAttr,
564                 int defStyleRes) {
565             super(context, attrs, defStyleAttr, defStyleRes);
566         }
567 
568         @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)569         protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
570             super.onMeasure(widthMeasureSpec, heightMeasureSpec);
571 
572             Layout layout = getLayout();
573             int lineCount = layout.getLineCount();
574             boolean ellipsizing = layout.getEllipsisCount(lineCount - 1) != 0;
575             if (lineCount > 0 && !ellipsizing) {
576                 CharSequence title = getText();
577                 CharSequence bestLineBreak = findBestLineBreak(title);
578                 if (!TextUtils.equals(title, bestLineBreak)) {
579                     setText(bestLineBreak);
580                     super.onMeasure(widthMeasureSpec, heightMeasureSpec);
581                 }
582             }
583         }
584     }
585 
586     private class SliceViewTransitionListener implements LayoutTransition.TransitionListener {
587         @Override
startTransition(LayoutTransition transition, ViewGroup container, View view, int transitionType)588         public void startTransition(LayoutTransition transition, ViewGroup container, View view,
589                 int transitionType) {
590             switch (transitionType) {
591                 case  LayoutTransition.APPEARING:
592                     int translation = getResources().getDimensionPixelSize(
593                             R.dimen.pulsing_notification_appear_translation);
594                     view.setTranslationY(translation);
595                     view.animate()
596                             .translationY(0)
597                             .setDuration(DEFAULT_ANIM_DURATION)
598                             .setInterpolator(Interpolators.ALPHA_IN)
599                             .start();
600                     break;
601                 case LayoutTransition.DISAPPEARING:
602                     if (view == mTitle) {
603                         // Translate the view to the inverse of its height, so the layout event
604                         // won't misposition it.
605                         LayoutParams params = (LayoutParams) mTitle.getLayoutParams();
606                         int margin = params.topMargin + params.bottomMargin;
607                         mTitle.setTranslationY(-mTitle.getHeight() - margin);
608                     }
609                     break;
610             }
611         }
612 
613         @Override
endTransition(LayoutTransition transition, ViewGroup container, View view, int transitionType)614         public void endTransition(LayoutTransition transition, ViewGroup container, View view,
615                 int transitionType) {
616 
617         }
618     }
619 }
620