blob: f5abd0631892aefa099d33024ce1c76db15b32e2 [file] [log] [blame]
Lucas Dupin957e50c2017-10-10 11:23:27 -07001/*
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
17package com.android.keyguard;
18
Lucas Dupinc68ec482018-03-30 14:24:53 -070019import android.animation.LayoutTransition;
20import android.animation.ObjectAnimator;
21import android.animation.PropertyValuesHolder;
Lucas Dupinf8e274c2018-02-22 17:01:55 -080022import android.annotation.ColorInt;
Lucas Dupin957e50c2017-10-10 11:23:27 -070023import android.app.PendingIntent;
Lucas Dupin6bd86012017-12-05 17:58:57 -080024import android.arch.lifecycle.LiveData;
25import android.arch.lifecycle.Observer;
Lucas Dupin957e50c2017-10-10 11:23:27 -070026import android.content.Context;
Lucas Dupin957e50c2017-10-10 11:23:27 -070027import android.graphics.Color;
Lucas Dupin6bd86012017-12-05 17:58:57 -080028import android.graphics.drawable.Drawable;
Lucas Dupin957e50c2017-10-10 11:23:27 -070029import android.net.Uri;
Lucas Dupin6bd86012017-12-05 17:58:57 -080030import android.provider.Settings;
Lucas Dupin2a3c3e32018-01-05 17:02:43 -080031import android.text.Layout;
32import android.text.TextUtils;
33import android.text.TextUtils.TruncateAt;
Lucas Dupin957e50c2017-10-10 11:23:27 -070034import android.util.AttributeSet;
Lucas Dupin6bd86012017-12-05 17:58:57 -080035import android.util.Log;
36import android.view.View;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070037import android.view.ViewGroup;
Lucas Dupinc68ec482018-03-30 14:24:53 -070038import android.view.animation.Animation;
Lucas Dupin6bd86012017-12-05 17:58:57 -080039import android.widget.Button;
Lucas Dupin957e50c2017-10-10 11:23:27 -070040import android.widget.LinearLayout;
41import android.widget.TextView;
42
Lucas Dupinf8e274c2018-02-22 17:01:55 -080043import com.android.internal.annotations.VisibleForTesting;
Lucas Dupin957e50c2017-10-10 11:23:27 -070044import com.android.internal.graphics.ColorUtils;
Lucas Dupin6bd86012017-12-05 17:58:57 -080045import com.android.settingslib.Utils;
46import com.android.systemui.Dependency;
Lucas Dupinc68ec482018-03-30 14:24:53 -070047import com.android.systemui.Interpolators;
Lucas Dupin957e50c2017-10-10 11:23:27 -070048import com.android.systemui.R;
49import com.android.systemui.keyguard.KeyguardSliceProvider;
Lucas Dupinb9afc962018-05-21 14:41:52 -070050import com.android.systemui.statusbar.AlphaOptimizedTextView;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070051import com.android.systemui.statusbar.policy.ConfigurationController;
Lucas Dupin6bd86012017-12-05 17:58:57 -080052import com.android.systemui.tuner.TunerService;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070053import com.android.systemui.util.wakelock.KeepAwakeAnimationListener;
Lucas Dupin957e50c2017-10-10 11:23:27 -070054
Lucas Dupin6bd86012017-12-05 17:58:57 -080055import java.util.HashMap;
56import java.util.List;
57import java.util.function.Consumer;
58
Alan Viverettee8935882018-03-15 21:19:36 +000059import androidx.slice.Slice;
60import androidx.slice.SliceItem;
Lucas Dupin9fedb892018-05-04 17:42:33 -070061import androidx.slice.SliceManager;
Alan Viverettee8935882018-03-15 21:19:36 +000062import androidx.slice.core.SliceQuery;
63import androidx.slice.widget.ListContent;
64import androidx.slice.widget.RowContent;
65import androidx.slice.widget.SliceLiveData;
Jason Monk2af19982017-11-07 19:38:27 -050066
Lucas Dupin957e50c2017-10-10 11:23:27 -070067/**
68 * View visible under the clock on the lock screen and AoD.
69 */
Lucas Dupin6bd86012017-12-05 17:58:57 -080070public class KeyguardSliceView extends LinearLayout implements View.OnClickListener,
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070071 Observer<Slice>, TunerService.Tunable, ConfigurationController.ConfigurationListener {
Lucas Dupin957e50c2017-10-10 11:23:27 -070072
Lucas Dupin6bd86012017-12-05 17:58:57 -080073 private static final String TAG = "KeyguardSliceView";
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070074 public static final int DEFAULT_ANIM_DURATION = 550;
75
Lucas Dupin6bd86012017-12-05 17:58:57 -080076 private final HashMap<View, PendingIntent> mClickActions;
77 private Uri mKeyguardSliceUri;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070078 @VisibleForTesting
79 TextView mTitle;
80 private Row mRow;
Lucas Dupin957e50c2017-10-10 11:23:27 -070081 private int mTextColor;
82 private float mDarkAmount = 0;
83
Lucas Dupin6bd86012017-12-05 17:58:57 -080084 private LiveData<Slice> mLiveData;
85 private int mIconSize;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070086 /**
87 * Listener called whenever the view contents change.
88 * Boolean will be true when the change happens animated.
89 */
90 private Consumer<Boolean> mContentChangeListener;
Lucas Dupin6bd86012017-12-05 17:58:57 -080091 private boolean mHasHeader;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -070092 private Slice mSlice;
93 private boolean mPulsing;
Lucas Dupin957e50c2017-10-10 11:23:27 -070094
95 public KeyguardSliceView(Context context) {
96 this(context, null, 0);
97 }
98
99 public KeyguardSliceView(Context context, AttributeSet attrs) {
100 this(context, attrs, 0);
101 }
102
103 public KeyguardSliceView(Context context, AttributeSet attrs, int defStyle) {
104 super(context, attrs, defStyle);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800105
106 TunerService tunerService = Dependency.get(TunerService.class);
107 tunerService.addTunable(this, Settings.Secure.KEYGUARD_SLICE_URI);
108
109 mClickActions = new HashMap<>();
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700110
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);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700122 }
123
124 @Override
125 protected void onFinishInflate() {
126 super.onFinishInflate();
127 mTitle = findViewById(R.id.title);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800128 mRow = findViewById(R.id.row);
129 mTextColor = Utils.getColorAttr(mContext, R.attr.wallpaperTextColor);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700130 }
131
132 @Override
133 protected void onAttachedToWindow() {
134 super.onAttachedToWindow();
135
Lucas Dupin957e50c2017-10-10 11:23:27 -0700136 // Make sure we always have the most current slice
Lucas Dupin6bd86012017-12-05 17:58:57 -0800137 mLiveData.observeForever(this);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700138 Dependency.get(ConfigurationController.class).addCallback(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700139 }
140
141 @Override
142 protected void onDetachedFromWindow() {
143 super.onDetachedFromWindow();
144
Lucas Dupin6bd86012017-12-05 17:58:57 -0800145 mLiveData.removeObserver(this);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700146 Dependency.get(ConfigurationController.class).removeCallback(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700147 }
148
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700149 private void showSlice() {
Lucas Dupinf39ce102018-05-02 14:23:44 -0700150 if (mPulsing || mSlice == null) {
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700151 mTitle.setVisibility(GONE);
152 mRow.setVisibility(GONE);
153 mContentChangeListener.accept(getLayoutTransition() != null);
154 return;
155 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700156
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700157 ListContent lc = new ListContent(getContext(), mSlice);
Mady Mellor90e9fce2018-01-22 17:30:40 -0800158 mHasHeader = lc.hasHeader();
159 List<SliceItem> subItems = lc.getRowItems();
Lucas Dupin6bd86012017-12-05 17:58:57 -0800160 if (!mHasHeader) {
Lucas Dupin957e50c2017-10-10 11:23:27 -0700161 mTitle.setVisibility(GONE);
162 } else {
163 mTitle.setVisibility(VISIBLE);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700164
Mady Mellor90e9fce2018-01-22 17:30:40 -0800165 // If there's a header it'll be the first subitem
Alan Viverettee8935882018-03-15 21:19:36 +0000166 RowContent header = new RowContent(getContext(), subItems.get(0),
167 true /* showStartItem */);
Mady Mellor90e9fce2018-01-22 17:30:40 -0800168 SliceItem mainTitle = header.getTitleItem();
169 CharSequence title = mainTitle != null ? mainTitle.getText() : null;
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800170 mTitle.setText(title);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700171 }
172
Lucas Dupin6bd86012017-12-05 17:58:57 -0800173 mClickActions.clear();
174 final int subItemsCount = subItems.size();
Lucas Dupina2a7a402018-01-12 17:45:51 -0800175 final int blendedColor = getTextColor();
Mady Mellor90e9fce2018-01-22 17:30:40 -0800176 final int startIndex = mHasHeader ? 1 : 0; // First item is header; skip it
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700177 mRow.setVisibility(subItemsCount > 0 ? VISIBLE : GONE);
Mady Mellor90e9fce2018-01-22 17:30:40 -0800178 for (int i = startIndex; i < subItemsCount; i++) {
Lucas Dupin6bd86012017-12-05 17:58:57 -0800179 SliceItem item = subItems.get(i);
Alan Viverettee8935882018-03-15 21:19:36 +0000180 RowContent rc = new RowContent(getContext(), item, true /* showStartItem */);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800181 final Uri itemTag = item.getSlice().getUri();
182 // Try to reuse the view if already exists in the layout
183 KeyguardSliceButton button = mRow.findViewWithTag(itemTag);
184 if (button == null) {
185 button = new KeyguardSliceButton(mContext);
Lucas Dupina2a7a402018-01-12 17:45:51 -0800186 button.setTextColor(blendedColor);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800187 button.setTag(itemTag);
Lucas Dupinc68ec482018-03-30 14:24:53 -0700188 final int viewIndex = i - (mHasHeader ? 1 : 0);
189 mRow.addView(button, viewIndex);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800190 }
Lucas Dupin6bd86012017-12-05 17:58:57 -0800191
Mady Mellor90e9fce2018-01-22 17:30:40 -0800192 PendingIntent pendingIntent = null;
Alan Viverette85935282018-02-27 22:10:18 +0000193 if (rc.getPrimaryAction() != null) {
194 pendingIntent = rc.getPrimaryAction().getAction();
Lucas Dupin6bd86012017-12-05 17:58:57 -0800195 }
196 mClickActions.put(button, pendingIntent);
197
Lucas Dupin1f7374a2018-02-26 18:08:33 -0800198 final SliceItem titleItem = rc.getTitleItem();
199 button.setText(titleItem == null ? null : titleItem.getText());
Lucas Dupin6bd86012017-12-05 17:58:57 -0800200
201 Drawable iconDrawable = null;
202 SliceItem icon = SliceQuery.find(item.getSlice(),
203 android.app.slice.SliceItem.FORMAT_IMAGE);
204 if (icon != null) {
205 iconDrawable = icon.getIcon().loadDrawable(mContext);
206 final int width = (int) (iconDrawable.getIntrinsicWidth()
207 / (float) iconDrawable.getIntrinsicHeight() * mIconSize);
208 iconDrawable.setBounds(0, 0, Math.max(width, 1), mIconSize);
209 }
Lucas Dupinacb0eed2018-03-03 14:05:04 -0800210 button.setCompoundDrawables(iconDrawable, null, null, null);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800211 button.setOnClickListener(this);
Lucas Dupin2c0d52e2018-03-14 12:39:00 -0700212 button.setClickable(pendingIntent != null);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700213 }
214
Lucas Dupin6bd86012017-12-05 17:58:57 -0800215 // Removing old views
216 for (int i = 0; i < mRow.getChildCount(); i++) {
217 View child = mRow.getChildAt(i);
218 if (!mClickActions.containsKey(child)) {
219 mRow.removeView(child);
220 i--;
221 }
222 }
223
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700224 if (mContentChangeListener != null) {
225 mContentChangeListener.accept(getLayoutTransition() != null);
226 }
Lucas Dupin2c5fce62018-03-05 19:06:00 -0800227 }
228
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700229 public void setPulsing(boolean pulsing, boolean animate) {
230 mPulsing = pulsing;
231 LayoutTransition transition = getLayoutTransition();
232 if (!animate) {
233 setLayoutTransition(null);
234 }
235 showSlice();
236 if (!animate) {
237 setLayoutTransition(transition);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700238 }
239 }
240
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800241 /**
242 * Breaks a string in 2 lines where both have similar character count
243 * but first line is always longer.
244 *
245 * @param charSequence Original text.
246 * @return Optimal string.
247 */
Lucas Dupinb9afc962018-05-21 14:41:52 -0700248 private static CharSequence findBestLineBreak(CharSequence charSequence) {
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800249 if (TextUtils.isEmpty(charSequence)) {
250 return charSequence;
251 }
252
253 String source = charSequence.toString();
254 // Ignore if there is only 1 word,
255 // or if line breaks were manually set.
256 if (source.contains("\n") || !source.contains(" ")) {
257 return source;
258 }
259
260 final String[] words = source.split(" ");
261 final StringBuilder optimalString = new StringBuilder(source.length());
262 int current = 0;
263 while (optimalString.length() < source.length() - optimalString.length()) {
264 optimalString.append(words[current]);
265 if (current < words.length - 1) {
266 optimalString.append(" ");
267 }
268 current++;
269 }
270 optimalString.append("\n");
271 for (int i = current; i < words.length; i++) {
272 optimalString.append(words[i]);
273 if (current < words.length - 1) {
274 optimalString.append(" ");
275 }
276 }
277
278 return optimalString.toString();
279 }
280
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700281 public void setDarkAmount(float darkAmount) {
Lucas Dupin957e50c2017-10-10 11:23:27 -0700282 mDarkAmount = darkAmount;
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700283 mRow.setDarkAmount(darkAmount);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700284 updateTextColors();
285 }
286
Lucas Dupin957e50c2017-10-10 11:23:27 -0700287 private void updateTextColors() {
Lucas Dupina2a7a402018-01-12 17:45:51 -0800288 final int blendedColor = getTextColor();
Lucas Dupin957e50c2017-10-10 11:23:27 -0700289 mTitle.setTextColor(blendedColor);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800290 int childCount = mRow.getChildCount();
291 for (int i = 0; i < childCount; i++) {
292 View v = mRow.getChildAt(i);
293 if (v instanceof Button) {
294 ((Button) v).setTextColor(blendedColor);
295 }
296 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700297 }
298
Lucas Dupin6bd86012017-12-05 17:58:57 -0800299 @Override
300 public void onClick(View v) {
301 final PendingIntent action = mClickActions.get(v);
302 if (action != null) {
303 try {
304 action.send();
305 } catch (PendingIntent.CanceledException e) {
306 Log.i(TAG, "Pending intent cancelled, nothing to launch", e);
307 }
308 }
309 }
310
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700311 /**
312 * Listener that gets invoked every time the title or the row visibility changes.
313 * Parameter will be {@code true} whenever the change happens animated.
314 * @param contentChangeListener The listener.
315 */
316 public void setContentChangeListener(Consumer<Boolean> contentChangeListener) {
317 mContentChangeListener = contentChangeListener;
Lucas Dupin6bd86012017-12-05 17:58:57 -0800318 }
319
320 public boolean hasHeader() {
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700321 return mTitle.getVisibility() == VISIBLE;
Lucas Dupin6bd86012017-12-05 17:58:57 -0800322 }
323
324 /**
325 * LiveData observer lifecycle.
326 * @param slice the new slice content.
327 */
328 @Override
329 public void onChanged(Slice slice) {
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700330 mSlice = slice;
331 showSlice();
Lucas Dupin6bd86012017-12-05 17:58:57 -0800332 }
333
334 @Override
335 public void onTuningChanged(String key, String newValue) {
336 setupUri(newValue);
337 }
338
339 public void setupUri(String uriString) {
340 if (uriString == null) {
341 uriString = KeyguardSliceProvider.KEYGUARD_SLICE_URI;
342 }
343
344 boolean wasObserving = false;
345 if (mLiveData != null && mLiveData.hasActiveObservers()) {
346 wasObserving = true;
347 mLiveData.removeObserver(this);
348 }
349
350 mKeyguardSliceUri = Uri.parse(uriString);
351 mLiveData = SliceLiveData.fromUri(mContext, mKeyguardSliceUri);
352
353 if (wasObserving) {
354 mLiveData.observeForever(this);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800355 }
356 }
357
Lucas Dupinf8e274c2018-02-22 17:01:55 -0800358 @VisibleForTesting
359 int getTextColor() {
Lucas Dupina2a7a402018-01-12 17:45:51 -0800360 return ColorUtils.blendARGB(mTextColor, Color.WHITE, mDarkAmount);
361 }
362
Lucas Dupinf8e274c2018-02-22 17:01:55 -0800363 @VisibleForTesting
364 void setTextColor(@ColorInt int textColor) {
365 mTextColor = textColor;
366 updateTextColors();
367 }
368
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700369 @Override
370 public void onDensityOrFontScaleChanged() {
371 mIconSize = mContext.getResources().getDimensionPixelSize(R.dimen.widget_icon_size);
Lucas Dupin2c5fce62018-03-05 19:06:00 -0800372 }
373
Lucas Dupin9fedb892018-05-04 17:42:33 -0700374 public void refresh() {
375 Slice slice = SliceManager.getInstance(getContext()).bindSlice(mKeyguardSliceUri);
376 onChanged(slice);
377 }
378
Lucas Dupin4603fe22018-03-08 16:51:16 -0800379 public static class Row extends LinearLayout {
380
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700381 /**
382 * This view is visible in AOD, which means that the device will sleep if we
383 * don't hold a wake lock. We want to enter doze only after all views have reached
384 * their desired positions.
385 */
386 private final Animation.AnimationListener mKeepAwakeListener;
387 private float mDarkAmount;
Lucas Dupinc68ec482018-03-30 14:24:53 -0700388
Lucas Dupin4603fe22018-03-08 16:51:16 -0800389 public Row(Context context) {
Lucas Dupinc68ec482018-03-30 14:24:53 -0700390 this(context, null);
Lucas Dupin4603fe22018-03-08 16:51:16 -0800391 }
392
393 public Row(Context context, AttributeSet attrs) {
Lucas Dupinc68ec482018-03-30 14:24:53 -0700394 this(context, attrs, 0);
Lucas Dupin4603fe22018-03-08 16:51:16 -0800395 }
396
397 public Row(Context context, AttributeSet attrs, int defStyleAttr) {
Lucas Dupinc68ec482018-03-30 14:24:53 -0700398 this(context, attrs, defStyleAttr, 0);
Lucas Dupin4603fe22018-03-08 16:51:16 -0800399 }
400
401 public Row(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
402 super(context, attrs, defStyleAttr, defStyleRes);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700403 mKeepAwakeListener = new KeepAwakeAnimationListener(mContext);
Lucas Dupinc68ec482018-03-30 14:24:53 -0700404 }
405
406 @Override
407 protected void onFinishInflate() {
408 LayoutTransition transition = new LayoutTransition();
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700409 transition.setDuration(DEFAULT_ANIM_DURATION);
Lucas Dupinc68ec482018-03-30 14:24:53 -0700410
411 PropertyValuesHolder left = PropertyValuesHolder.ofInt("left", 0, 1);
412 PropertyValuesHolder right = PropertyValuesHolder.ofInt("right", 0, 1);
413 ObjectAnimator changeAnimator = ObjectAnimator.ofPropertyValuesHolder((Object) null,
414 left, right);
415 transition.setAnimator(LayoutTransition.CHANGE_APPEARING, changeAnimator);
416 transition.setAnimator(LayoutTransition.CHANGE_DISAPPEARING, changeAnimator);
417 transition.setInterpolator(LayoutTransition.CHANGE_APPEARING,
418 Interpolators.ACCELERATE_DECELERATE);
419 transition.setInterpolator(LayoutTransition.CHANGE_DISAPPEARING,
420 Interpolators.ACCELERATE_DECELERATE);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700421 transition.setStartDelay(LayoutTransition.CHANGE_APPEARING, DEFAULT_ANIM_DURATION);
422 transition.setStartDelay(LayoutTransition.CHANGE_DISAPPEARING, DEFAULT_ANIM_DURATION);
Lucas Dupinc68ec482018-03-30 14:24:53 -0700423
424 ObjectAnimator appearAnimator = ObjectAnimator.ofFloat(null, "alpha", 0f, 1f);
425 transition.setAnimator(LayoutTransition.APPEARING, appearAnimator);
426 transition.setInterpolator(LayoutTransition.APPEARING, Interpolators.ALPHA_IN);
427
428 ObjectAnimator disappearAnimator = ObjectAnimator.ofFloat(null, "alpha", 1f, 0f);
429 transition.setInterpolator(LayoutTransition.DISAPPEARING, Interpolators.ALPHA_OUT);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700430 transition.setDuration(LayoutTransition.DISAPPEARING, DEFAULT_ANIM_DURATION / 4);
Lucas Dupinc68ec482018-03-30 14:24:53 -0700431 transition.setAnimator(LayoutTransition.DISAPPEARING, disappearAnimator);
432
433 transition.setAnimateParentHierarchy(false);
434 setLayoutTransition(transition);
Lucas Dupin4603fe22018-03-08 16:51:16 -0800435 }
436
437 @Override
438 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
439 int width = MeasureSpec.getSize(widthMeasureSpec);
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700440 int childCount = getChildCount();
441 for (int i = 0; i < childCount; i++) {
Lucas Dupin4603fe22018-03-08 16:51:16 -0800442 View child = getChildAt(i);
443 if (child instanceof KeyguardSliceButton) {
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700444 ((KeyguardSliceButton) child).setMaxWidth(width / childCount);
Lucas Dupin4603fe22018-03-08 16:51:16 -0800445 }
446 }
447 super.onMeasure(widthMeasureSpec, heightMeasureSpec);
448 }
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700449
450 public void setDarkAmount(float darkAmount) {
451 boolean isAwake = darkAmount != 0;
452 boolean wasAwake = mDarkAmount != 0;
453 if (isAwake == wasAwake) {
454 return;
455 }
456 mDarkAmount = darkAmount;
457 setLayoutAnimationListener(isAwake ? null : mKeepAwakeListener);
458 }
459
460 @Override
461 public boolean hasOverlappingRendering() {
462 return false;
463 }
Lucas Dupin4603fe22018-03-08 16:51:16 -0800464 }
465
Lucas Dupin6bd86012017-12-05 17:58:57 -0800466 /**
467 * Representation of an item that appears under the clock on main keyguard message.
Lucas Dupin6bd86012017-12-05 17:58:57 -0800468 */
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700469 @VisibleForTesting
470 static class KeyguardSliceButton extends Button implements
471 ConfigurationController.ConfigurationListener {
Lucas Dupin6bd86012017-12-05 17:58:57 -0800472
Lucas Dupin6bd86012017-12-05 17:58:57 -0800473 public KeyguardSliceButton(Context context) {
Lucas Dupinb16d8232018-01-26 12:44:52 -0800474 super(context, null /* attrs */, 0 /* styleAttr */,
Lucas Dupin6bd86012017-12-05 17:58:57 -0800475 com.android.keyguard.R.style.TextAppearance_Keyguard_Secondary);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700476 onDensityOrFontScaleChanged();
477 setEllipsize(TruncateAt.END);
478 }
479
480 @Override
481 protected void onAttachedToWindow() {
482 super.onAttachedToWindow();
483 Dependency.get(ConfigurationController.class).addCallback(this);
484 }
485
486 @Override
487 protected void onDetachedFromWindow() {
488 super.onDetachedFromWindow();
489 Dependency.get(ConfigurationController.class).removeCallback(this);
490 }
491
492 @Override
493 public void onDensityOrFontScaleChanged() {
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700494 updatePadding();
495 }
496
497 @Override
498 public void setText(CharSequence text, BufferType type) {
499 super.setText(text, type);
500 updatePadding();
501 }
502
503 private void updatePadding() {
504 boolean hasText = !TextUtils.isEmpty(getText());
505 int horizontalPadding = (int) getContext().getResources()
506 .getDimension(R.dimen.widget_horizontal_padding) / 2;
507 setPadding(horizontalPadding, 0, horizontalPadding * (hasText ? 1 : -1), 0);
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700508 setCompoundDrawablePadding((int) mContext.getResources()
Lucas Dupin6bd86012017-12-05 17:58:57 -0800509 .getDimension(R.dimen.widget_icon_padding));
510 }
Lucas Dupinacb0eed2018-03-03 14:05:04 -0800511
512 @Override
513 public void setTextColor(int color) {
514 super.setTextColor(color);
515 updateDrawableColors();
516 }
517
518 @Override
519 public void setCompoundDrawables(Drawable left, Drawable top, Drawable right,
520 Drawable bottom) {
521 super.setCompoundDrawables(left, top, right, bottom);
522 updateDrawableColors();
Lucas Dupin8b77a3b2018-05-01 13:23:03 -0700523 updatePadding();
Lucas Dupinacb0eed2018-03-03 14:05:04 -0800524 }
525
526 private void updateDrawableColors() {
527 final int color = getCurrentTextColor();
528 for (Drawable drawable : getCompoundDrawables()) {
529 if (drawable != null) {
530 drawable.setTint(color);
531 }
532 }
533 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700534 }
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700535
Lucas Dupinb9afc962018-05-21 14:41:52 -0700536 /**
537 * A text view that will split its contents in 2 lines when possible.
538 */
539 static class TitleView extends AlphaOptimizedTextView {
540
541 public TitleView(Context context) {
542 super(context);
543 }
544
545 public TitleView(Context context, AttributeSet attrs) {
546 super(context, attrs);
547 }
548
549 public TitleView(Context context, AttributeSet attrs, int defStyleAttr) {
550 super(context, attrs, defStyleAttr);
551 }
552
553 public TitleView(Context context, AttributeSet attrs, int defStyleAttr,
554 int defStyleRes) {
555 super(context, attrs, defStyleAttr, defStyleRes);
556 }
557
558 @Override
559 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
560 super.onMeasure(widthMeasureSpec, heightMeasureSpec);
561
562 Layout layout = getLayout();
563 int lineCount = layout.getLineCount();
564 boolean ellipsizing = layout.getEllipsisCount(lineCount - 1) != 0;
565 if (lineCount > 0 && !ellipsizing) {
566 CharSequence title = getText();
567 CharSequence bestLineBreak = findBestLineBreak(title);
568 if (!TextUtils.equals(title, bestLineBreak)) {
569 setText(bestLineBreak);
570 super.onMeasure(widthMeasureSpec, heightMeasureSpec);
571 }
572 }
573 }
574 }
575
Lucas Dupin3d7ccaf2018-04-02 21:19:23 -0700576 private class SliceViewTransitionListener implements LayoutTransition.TransitionListener {
577 @Override
578 public void startTransition(LayoutTransition transition, ViewGroup container, View view,
579 int transitionType) {
580 switch (transitionType) {
581 case LayoutTransition.APPEARING:
582 int translation = getResources().getDimensionPixelSize(
583 R.dimen.pulsing_notification_appear_translation);
584 view.setTranslationY(translation);
585 view.animate()
586 .translationY(0)
587 .setDuration(DEFAULT_ANIM_DURATION)
588 .setInterpolator(Interpolators.ALPHA_IN)
589 .start();
590 break;
591 case LayoutTransition.DISAPPEARING:
592 if (view == mTitle) {
593 // Translate the view to the inverse of its height, so the layout event
594 // won't misposition it.
595 LayoutParams params = (LayoutParams) mTitle.getLayoutParams();
596 int margin = params.topMargin + params.bottomMargin;
597 mTitle.setTranslationY(-mTitle.getHeight() - margin);
598 }
599 break;
600 }
601 }
602
603 @Override
604 public void endTransition(LayoutTransition transition, ViewGroup container, View view,
605 int transitionType) {
606
607 }
608 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700609}