blob: d80a33632f3e77c197780a3537fb9aa6696da3fa [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
19import android.app.PendingIntent;
Lucas Dupin6bd86012017-12-05 17:58:57 -080020import android.arch.lifecycle.LiveData;
21import android.arch.lifecycle.Observer;
Lucas Dupin957e50c2017-10-10 11:23:27 -070022import android.content.Context;
Lucas Dupin6bd86012017-12-05 17:58:57 -080023import android.graphics.Canvas;
Lucas Dupin957e50c2017-10-10 11:23:27 -070024import android.graphics.Color;
Lucas Dupin6bd86012017-12-05 17:58:57 -080025import android.graphics.Paint;
26import android.graphics.drawable.Drawable;
Lucas Dupin957e50c2017-10-10 11:23:27 -070027import android.net.Uri;
Lucas Dupin6bd86012017-12-05 17:58:57 -080028import android.provider.Settings;
Lucas Dupin2a3c3e32018-01-05 17:02:43 -080029import android.text.Layout;
30import android.text.TextUtils;
31import android.text.TextUtils.TruncateAt;
Lucas Dupin957e50c2017-10-10 11:23:27 -070032import android.util.AttributeSet;
Lucas Dupin6bd86012017-12-05 17:58:57 -080033import android.util.Log;
34import android.view.View;
35import android.widget.Button;
Lucas Dupin957e50c2017-10-10 11:23:27 -070036import android.widget.LinearLayout;
37import android.widget.TextView;
38
39import com.android.internal.graphics.ColorUtils;
Lucas Dupin6bd86012017-12-05 17:58:57 -080040import com.android.settingslib.Utils;
41import com.android.systemui.Dependency;
Lucas Dupin957e50c2017-10-10 11:23:27 -070042import com.android.systemui.R;
43import com.android.systemui.keyguard.KeyguardSliceProvider;
Lucas Dupin6bd86012017-12-05 17:58:57 -080044import com.android.systemui.tuner.TunerService;
Lucas Dupin957e50c2017-10-10 11:23:27 -070045
Lucas Dupin2a3c3e32018-01-05 17:02:43 -080046import java.util.ArrayList;
Lucas Dupin6bd86012017-12-05 17:58:57 -080047import java.util.HashMap;
48import java.util.List;
49import java.util.function.Consumer;
50
51import androidx.app.slice.Slice;
52import androidx.app.slice.SliceItem;
53import androidx.app.slice.core.SliceQuery;
54import androidx.app.slice.widget.SliceLiveData;
Jason Monk2af19982017-11-07 19:38:27 -050055
Lucas Dupin957e50c2017-10-10 11:23:27 -070056/**
57 * View visible under the clock on the lock screen and AoD.
58 */
Lucas Dupin6bd86012017-12-05 17:58:57 -080059public class KeyguardSliceView extends LinearLayout implements View.OnClickListener,
60 Observer<Slice>, TunerService.Tunable {
Lucas Dupin957e50c2017-10-10 11:23:27 -070061
Lucas Dupin6bd86012017-12-05 17:58:57 -080062 private static final String TAG = "KeyguardSliceView";
63 private final HashMap<View, PendingIntent> mClickActions;
64 private Uri mKeyguardSliceUri;
Lucas Dupin957e50c2017-10-10 11:23:27 -070065 private TextView mTitle;
Lucas Dupin6bd86012017-12-05 17:58:57 -080066 private LinearLayout mRow;
Lucas Dupin957e50c2017-10-10 11:23:27 -070067 private int mTextColor;
68 private float mDarkAmount = 0;
69
Lucas Dupin6bd86012017-12-05 17:58:57 -080070 private LiveData<Slice> mLiveData;
71 private int mIconSize;
72 private Consumer<Boolean> mListener;
73 private boolean mHasHeader;
Lucas Dupin957e50c2017-10-10 11:23:27 -070074
75 public KeyguardSliceView(Context context) {
76 this(context, null, 0);
77 }
78
79 public KeyguardSliceView(Context context, AttributeSet attrs) {
80 this(context, attrs, 0);
81 }
82
83 public KeyguardSliceView(Context context, AttributeSet attrs, int defStyle) {
84 super(context, attrs, defStyle);
Lucas Dupin6bd86012017-12-05 17:58:57 -080085
86 TunerService tunerService = Dependency.get(TunerService.class);
87 tunerService.addTunable(this, Settings.Secure.KEYGUARD_SLICE_URI);
88
89 mClickActions = new HashMap<>();
Lucas Dupin957e50c2017-10-10 11:23:27 -070090 }
91
92 @Override
93 protected void onFinishInflate() {
94 super.onFinishInflate();
95 mTitle = findViewById(R.id.title);
Lucas Dupin6bd86012017-12-05 17:58:57 -080096 mRow = findViewById(R.id.row);
97 mTextColor = Utils.getColorAttr(mContext, R.attr.wallpaperTextColor);
98 mIconSize = (int) mContext.getResources().getDimension(R.dimen.widget_icon_size);
Lucas Dupin957e50c2017-10-10 11:23:27 -070099 }
100
101 @Override
102 protected void onAttachedToWindow() {
103 super.onAttachedToWindow();
104
Lucas Dupin957e50c2017-10-10 11:23:27 -0700105 // Make sure we always have the most current slice
Lucas Dupin6bd86012017-12-05 17:58:57 -0800106 mLiveData.observeForever(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700107 }
108
109 @Override
110 protected void onDetachedFromWindow() {
111 super.onDetachedFromWindow();
112
Lucas Dupin6bd86012017-12-05 17:58:57 -0800113 mLiveData.removeObserver(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700114 }
115
116 private void showSlice(Slice slice) {
Lucas Dupin957e50c2017-10-10 11:23:27 -0700117
Lucas Dupin6bd86012017-12-05 17:58:57 -0800118 // Main area
119 SliceItem mainItem = SliceQuery.find(slice, android.app.slice.SliceItem.FORMAT_SLICE,
120 null /* hints */, new String[]{android.app.slice.Slice.HINT_LIST_ITEM});
121 mHasHeader = mainItem != null;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700122
Lucas Dupin6bd86012017-12-05 17:58:57 -0800123 List<SliceItem> subItems = SliceQuery.findAll(slice,
124 android.app.slice.SliceItem.FORMAT_SLICE,
125 new String[]{android.app.slice.Slice.HINT_LIST_ITEM},
126 null /* nonHints */);
127
128 if (!mHasHeader) {
Lucas Dupin957e50c2017-10-10 11:23:27 -0700129 mTitle.setVisibility(GONE);
130 } else {
131 mTitle.setVisibility(VISIBLE);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800132 SliceItem mainTitle = SliceQuery.find(mainItem.getSlice(),
133 android.app.slice.SliceItem.FORMAT_TEXT,
134 new String[]{android.app.slice.Slice.HINT_TITLE},
135 null /* nonHints */);
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800136 CharSequence title = mainTitle.getText();
137 mTitle.setText(title);
138
139 // Check if we're already ellipsizing the text.
140 // We're going to figure out the best possible line break if not.
141 Layout layout = mTitle.getLayout();
142 if (layout != null){
143 final int lineCount = layout.getLineCount();
144 if (lineCount > 0) {
145 if (layout.getEllipsisCount(lineCount - 1) == 0) {
146 mTitle.setText(findBestLineBreak(title));
147 }
148 }
149 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700150 }
151
Lucas Dupin6bd86012017-12-05 17:58:57 -0800152 mClickActions.clear();
153 final int subItemsCount = subItems.size();
Lucas Dupina2a7a402018-01-12 17:45:51 -0800154 final int blendedColor = getTextColor();
Lucas Dupin6bd86012017-12-05 17:58:57 -0800155
156 for (int i = 0; i < subItemsCount; i++) {
157 SliceItem item = subItems.get(i);
158 final Uri itemTag = item.getSlice().getUri();
159 // Try to reuse the view if already exists in the layout
160 KeyguardSliceButton button = mRow.findViewWithTag(itemTag);
161 if (button == null) {
162 button = new KeyguardSliceButton(mContext);
Lucas Dupina2a7a402018-01-12 17:45:51 -0800163 button.setTextColor(blendedColor);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800164 button.setTag(itemTag);
165 } else {
166 mRow.removeView(button);
167 }
168 button.setHasDivider(i < subItemsCount - 1);
169 mRow.addView(button, i);
170
171 PendingIntent pendingIntent;
172 try {
173 pendingIntent = item.getAction();
174 } catch (RuntimeException e) {
175 Log.w(TAG, "Cannot retrieve action from keyguard slice", e);
176 pendingIntent = null;
177 }
178 mClickActions.put(button, pendingIntent);
179
180 SliceItem title = SliceQuery.find(item.getSlice(),
181 android.app.slice.SliceItem.FORMAT_TEXT,
182 new String[]{android.app.slice.Slice.HINT_TITLE},
183 null /* nonHints */);
184 button.setText(title.getText());
185
186 Drawable iconDrawable = null;
187 SliceItem icon = SliceQuery.find(item.getSlice(),
188 android.app.slice.SliceItem.FORMAT_IMAGE);
189 if (icon != null) {
190 iconDrawable = icon.getIcon().loadDrawable(mContext);
191 final int width = (int) (iconDrawable.getIntrinsicWidth()
192 / (float) iconDrawable.getIntrinsicHeight() * mIconSize);
193 iconDrawable.setBounds(0, 0, Math.max(width, 1), mIconSize);
194 }
195 button.setCompoundDrawablesRelative(iconDrawable, null, null, null);
196 button.setOnClickListener(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700197 }
198
Lucas Dupin6bd86012017-12-05 17:58:57 -0800199 // Removing old views
200 for (int i = 0; i < mRow.getChildCount(); i++) {
201 View child = mRow.getChildAt(i);
202 if (!mClickActions.containsKey(child)) {
203 mRow.removeView(child);
204 i--;
205 }
206 }
207
208 final int visibility = mHasHeader || subItemsCount > 0 ? VISIBLE : GONE;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700209 if (visibility != getVisibility()) {
210 setVisibility(visibility);
211 }
Lucas Dupin6bd86012017-12-05 17:58:57 -0800212
213 mListener.accept(mHasHeader);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700214 }
215
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800216 /**
217 * Breaks a string in 2 lines where both have similar character count
218 * but first line is always longer.
219 *
220 * @param charSequence Original text.
221 * @return Optimal string.
222 */
223 private CharSequence findBestLineBreak(CharSequence charSequence) {
224 if (TextUtils.isEmpty(charSequence)) {
225 return charSequence;
226 }
227
228 String source = charSequence.toString();
229 // Ignore if there is only 1 word,
230 // or if line breaks were manually set.
231 if (source.contains("\n") || !source.contains(" ")) {
232 return source;
233 }
234
235 final String[] words = source.split(" ");
236 final StringBuilder optimalString = new StringBuilder(source.length());
237 int current = 0;
238 while (optimalString.length() < source.length() - optimalString.length()) {
239 optimalString.append(words[current]);
240 if (current < words.length - 1) {
241 optimalString.append(" ");
242 }
243 current++;
244 }
245 optimalString.append("\n");
246 for (int i = current; i < words.length; i++) {
247 optimalString.append(words[i]);
248 if (current < words.length - 1) {
249 optimalString.append(" ");
250 }
251 }
252
253 return optimalString.toString();
254 }
255
Lucas Dupin957e50c2017-10-10 11:23:27 -0700256 public void setDark(float darkAmount) {
257 mDarkAmount = darkAmount;
258 updateTextColors();
259 }
260
Lucas Dupin957e50c2017-10-10 11:23:27 -0700261 private void updateTextColors() {
Lucas Dupina2a7a402018-01-12 17:45:51 -0800262 final int blendedColor = getTextColor();
Lucas Dupin957e50c2017-10-10 11:23:27 -0700263 mTitle.setTextColor(blendedColor);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800264 int childCount = mRow.getChildCount();
265 for (int i = 0; i < childCount; i++) {
266 View v = mRow.getChildAt(i);
267 if (v instanceof Button) {
268 ((Button) v).setTextColor(blendedColor);
269 }
270 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700271 }
272
Lucas Dupin6bd86012017-12-05 17:58:57 -0800273 @Override
274 public void onClick(View v) {
275 final PendingIntent action = mClickActions.get(v);
276 if (action != null) {
277 try {
278 action.send();
279 } catch (PendingIntent.CanceledException e) {
280 Log.i(TAG, "Pending intent cancelled, nothing to launch", e);
281 }
282 }
283 }
284
285 public void setListener(Consumer<Boolean> listener) {
286 mListener = listener;
287 }
288
289 public boolean hasHeader() {
290 return mHasHeader;
291 }
292
293 /**
294 * LiveData observer lifecycle.
295 * @param slice the new slice content.
296 */
297 @Override
298 public void onChanged(Slice slice) {
299 showSlice(slice);
300 }
301
302 @Override
303 public void onTuningChanged(String key, String newValue) {
304 setupUri(newValue);
305 }
306
307 public void setupUri(String uriString) {
308 if (uriString == null) {
309 uriString = KeyguardSliceProvider.KEYGUARD_SLICE_URI;
310 }
311
312 boolean wasObserving = false;
313 if (mLiveData != null && mLiveData.hasActiveObservers()) {
314 wasObserving = true;
315 mLiveData.removeObserver(this);
316 }
317
318 mKeyguardSliceUri = Uri.parse(uriString);
319 mLiveData = SliceLiveData.fromUri(mContext, mKeyguardSliceUri);
320
321 if (wasObserving) {
322 mLiveData.observeForever(this);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800323 }
324 }
325
Lucas Dupina2a7a402018-01-12 17:45:51 -0800326 public int getTextColor() {
327 return ColorUtils.blendARGB(mTextColor, Color.WHITE, mDarkAmount);
328 }
329
Lucas Dupin6bd86012017-12-05 17:58:57 -0800330 /**
331 * Representation of an item that appears under the clock on main keyguard message.
332 * Shows optional separator.
333 */
334 private class KeyguardSliceButton extends Button {
335
336 private final Paint mPaint;
337 private boolean mHasDivider;
338
339 public KeyguardSliceButton(Context context) {
340 super(context, null /* attrs */,
341 com.android.keyguard.R.style.TextAppearance_Keyguard_Secondary);
342 mPaint = new Paint();
343 mPaint.setStyle(Paint.Style.STROKE);
344 float dividerWidth = context.getResources()
345 .getDimension(R.dimen.widget_separator_thickness);
346 mPaint.setStrokeWidth(dividerWidth);
347 int horizontalPadding = (int) context.getResources()
348 .getDimension(R.dimen.widget_horizontal_padding);
349 setPadding(horizontalPadding, 0, horizontalPadding, 0);
350 setCompoundDrawablePadding((int) context.getResources()
351 .getDimension(R.dimen.widget_icon_padding));
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800352 setMaxWidth(KeyguardSliceView.this.getWidth() / 2);
353 setMaxLines(1);
354 setEllipsize(TruncateAt.END);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800355 }
356
357 public void setHasDivider(boolean hasDivider) {
358 mHasDivider = hasDivider;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700359 }
360
361 @Override
Lucas Dupin6bd86012017-12-05 17:58:57 -0800362 public void setTextColor(int color) {
363 super.setTextColor(color);
364 mPaint.setColor(color);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700365 }
366
367 @Override
Lucas Dupin6bd86012017-12-05 17:58:57 -0800368 protected void onDraw(Canvas canvas) {
369 super.onDraw(canvas);
370 if (mHasDivider) {
371 final int lineX = getLayoutDirection() == LAYOUT_DIRECTION_RTL ? 0 : getWidth();
372 canvas.drawLine(lineX, 0, lineX, getHeight(), mPaint);
373 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700374 }
375 }
376}