blob: 8135c616d87ec6fcd4ce053cda0e27337c5b5dff [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();
154
155 for (int i = 0; i < subItemsCount; i++) {
156 SliceItem item = subItems.get(i);
157 final Uri itemTag = item.getSlice().getUri();
158 // Try to reuse the view if already exists in the layout
159 KeyguardSliceButton button = mRow.findViewWithTag(itemTag);
160 if (button == null) {
161 button = new KeyguardSliceButton(mContext);
162 button.setTextColor(mTextColor);
163 button.setTag(itemTag);
164 } else {
165 mRow.removeView(button);
166 }
167 button.setHasDivider(i < subItemsCount - 1);
168 mRow.addView(button, i);
169
170 PendingIntent pendingIntent;
171 try {
172 pendingIntent = item.getAction();
173 } catch (RuntimeException e) {
174 Log.w(TAG, "Cannot retrieve action from keyguard slice", e);
175 pendingIntent = null;
176 }
177 mClickActions.put(button, pendingIntent);
178
179 SliceItem title = SliceQuery.find(item.getSlice(),
180 android.app.slice.SliceItem.FORMAT_TEXT,
181 new String[]{android.app.slice.Slice.HINT_TITLE},
182 null /* nonHints */);
183 button.setText(title.getText());
184
185 Drawable iconDrawable = null;
186 SliceItem icon = SliceQuery.find(item.getSlice(),
187 android.app.slice.SliceItem.FORMAT_IMAGE);
188 if (icon != null) {
189 iconDrawable = icon.getIcon().loadDrawable(mContext);
190 final int width = (int) (iconDrawable.getIntrinsicWidth()
191 / (float) iconDrawable.getIntrinsicHeight() * mIconSize);
192 iconDrawable.setBounds(0, 0, Math.max(width, 1), mIconSize);
193 }
194 button.setCompoundDrawablesRelative(iconDrawable, null, null, null);
195 button.setOnClickListener(this);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700196 }
197
Lucas Dupin6bd86012017-12-05 17:58:57 -0800198 // Removing old views
199 for (int i = 0; i < mRow.getChildCount(); i++) {
200 View child = mRow.getChildAt(i);
201 if (!mClickActions.containsKey(child)) {
202 mRow.removeView(child);
203 i--;
204 }
205 }
206
207 final int visibility = mHasHeader || subItemsCount > 0 ? VISIBLE : GONE;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700208 if (visibility != getVisibility()) {
209 setVisibility(visibility);
210 }
Lucas Dupin6bd86012017-12-05 17:58:57 -0800211
212 mListener.accept(mHasHeader);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700213 }
214
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800215 /**
216 * Breaks a string in 2 lines where both have similar character count
217 * but first line is always longer.
218 *
219 * @param charSequence Original text.
220 * @return Optimal string.
221 */
222 private CharSequence findBestLineBreak(CharSequence charSequence) {
223 if (TextUtils.isEmpty(charSequence)) {
224 return charSequence;
225 }
226
227 String source = charSequence.toString();
228 // Ignore if there is only 1 word,
229 // or if line breaks were manually set.
230 if (source.contains("\n") || !source.contains(" ")) {
231 return source;
232 }
233
234 final String[] words = source.split(" ");
235 final StringBuilder optimalString = new StringBuilder(source.length());
236 int current = 0;
237 while (optimalString.length() < source.length() - optimalString.length()) {
238 optimalString.append(words[current]);
239 if (current < words.length - 1) {
240 optimalString.append(" ");
241 }
242 current++;
243 }
244 optimalString.append("\n");
245 for (int i = current; i < words.length; i++) {
246 optimalString.append(words[i]);
247 if (current < words.length - 1) {
248 optimalString.append(" ");
249 }
250 }
251
252 return optimalString.toString();
253 }
254
Lucas Dupin957e50c2017-10-10 11:23:27 -0700255 public void setDark(float darkAmount) {
256 mDarkAmount = darkAmount;
257 updateTextColors();
258 }
259
Lucas Dupin957e50c2017-10-10 11:23:27 -0700260 private void updateTextColors() {
261 final int blendedColor = ColorUtils.blendARGB(mTextColor, Color.WHITE, mDarkAmount);
262 mTitle.setTextColor(blendedColor);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800263 int childCount = mRow.getChildCount();
264 for (int i = 0; i < childCount; i++) {
265 View v = mRow.getChildAt(i);
266 if (v instanceof Button) {
267 ((Button) v).setTextColor(blendedColor);
268 }
269 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700270 }
271
Lucas Dupin6bd86012017-12-05 17:58:57 -0800272 @Override
273 public void onClick(View v) {
274 final PendingIntent action = mClickActions.get(v);
275 if (action != null) {
276 try {
277 action.send();
278 } catch (PendingIntent.CanceledException e) {
279 Log.i(TAG, "Pending intent cancelled, nothing to launch", e);
280 }
281 }
282 }
283
284 public void setListener(Consumer<Boolean> listener) {
285 mListener = listener;
286 }
287
288 public boolean hasHeader() {
289 return mHasHeader;
290 }
291
292 /**
293 * LiveData observer lifecycle.
294 * @param slice the new slice content.
295 */
296 @Override
297 public void onChanged(Slice slice) {
298 showSlice(slice);
299 }
300
301 @Override
302 public void onTuningChanged(String key, String newValue) {
303 setupUri(newValue);
304 }
305
306 public void setupUri(String uriString) {
307 if (uriString == null) {
308 uriString = KeyguardSliceProvider.KEYGUARD_SLICE_URI;
309 }
310
311 boolean wasObserving = false;
312 if (mLiveData != null && mLiveData.hasActiveObservers()) {
313 wasObserving = true;
314 mLiveData.removeObserver(this);
315 }
316
317 mKeyguardSliceUri = Uri.parse(uriString);
318 mLiveData = SliceLiveData.fromUri(mContext, mKeyguardSliceUri);
319
320 if (wasObserving) {
321 mLiveData.observeForever(this);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800322 }
323 }
324
325 /**
326 * Representation of an item that appears under the clock on main keyguard message.
327 * Shows optional separator.
328 */
329 private class KeyguardSliceButton extends Button {
330
331 private final Paint mPaint;
332 private boolean mHasDivider;
333
334 public KeyguardSliceButton(Context context) {
335 super(context, null /* attrs */,
336 com.android.keyguard.R.style.TextAppearance_Keyguard_Secondary);
337 mPaint = new Paint();
338 mPaint.setStyle(Paint.Style.STROKE);
339 float dividerWidth = context.getResources()
340 .getDimension(R.dimen.widget_separator_thickness);
341 mPaint.setStrokeWidth(dividerWidth);
342 int horizontalPadding = (int) context.getResources()
343 .getDimension(R.dimen.widget_horizontal_padding);
344 setPadding(horizontalPadding, 0, horizontalPadding, 0);
345 setCompoundDrawablePadding((int) context.getResources()
346 .getDimension(R.dimen.widget_icon_padding));
Lucas Dupin2a3c3e32018-01-05 17:02:43 -0800347 setMaxWidth(KeyguardSliceView.this.getWidth() / 2);
348 setMaxLines(1);
349 setEllipsize(TruncateAt.END);
Lucas Dupin6bd86012017-12-05 17:58:57 -0800350 }
351
352 public void setHasDivider(boolean hasDivider) {
353 mHasDivider = hasDivider;
Lucas Dupin957e50c2017-10-10 11:23:27 -0700354 }
355
356 @Override
Lucas Dupin6bd86012017-12-05 17:58:57 -0800357 public void setTextColor(int color) {
358 super.setTextColor(color);
359 mPaint.setColor(color);
Lucas Dupin957e50c2017-10-10 11:23:27 -0700360 }
361
362 @Override
Lucas Dupin6bd86012017-12-05 17:58:57 -0800363 protected void onDraw(Canvas canvas) {
364 super.onDraw(canvas);
365 if (mHasDivider) {
366 final int lineX = getLayoutDirection() == LAYOUT_DIRECTION_RTL ? 0 : getWidth();
367 canvas.drawLine(lineX, 0, lineX, getHeight(), mPaint);
368 }
Lucas Dupin957e50c2017-10-10 11:23:27 -0700369 }
370 }
371}