• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2014 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.launcher3.widget;
18 
19 import android.content.Context;
20 import android.graphics.Bitmap;
21 import android.graphics.Canvas;
22 import android.graphics.Color;
23 import android.graphics.PorterDuff;
24 import android.graphics.Rect;
25 import android.graphics.drawable.Drawable;
26 import android.os.Bundle;
27 import android.text.Layout;
28 import android.text.StaticLayout;
29 import android.text.TextPaint;
30 import android.util.TypedValue;
31 import android.view.ContextThemeWrapper;
32 import android.view.View;
33 import android.view.View.OnClickListener;
34 
35 import com.android.launcher3.DeviceProfile;
36 import com.android.launcher3.FastBitmapDrawable;
37 import com.android.launcher3.IconCache;
38 import com.android.launcher3.IconCache.ItemInfoUpdateReceiver;
39 import com.android.launcher3.ItemInfoWithIcon;
40 import com.android.launcher3.Launcher;
41 import com.android.launcher3.LauncherAppWidgetInfo;
42 import com.android.launcher3.R;
43 import com.android.launcher3.Utilities;
44 import com.android.launcher3.graphics.DrawableFactory;
45 import com.android.launcher3.model.PackageItemInfo;
46 import com.android.launcher3.touch.ItemClickHandler;
47 import com.android.launcher3.util.Themes;
48 import com.android.launcher3.widget.LauncherAppWidgetHostView;
49 
50 public class PendingAppWidgetHostView extends LauncherAppWidgetHostView
51         implements OnClickListener, ItemInfoUpdateReceiver {
52     private static final float SETUP_ICON_SIZE_FACTOR = 2f / 5;
53     private static final float MIN_SATUNATION = 0.7f;
54 
55     private final Rect mRect = new Rect();
56     private View mDefaultView;
57     private OnClickListener mClickListener;
58     private final LauncherAppWidgetInfo mInfo;
59     private final int mStartState;
60     private final boolean mDisabledForSafeMode;
61 
62     private Drawable mCenterDrawable;
63     private Drawable mSettingIconDrawable;
64 
65     private boolean mDrawableSizeChanged;
66 
67     private final TextPaint mPaint;
68     private Layout mSetupTextLayout;
69 
PendingAppWidgetHostView(Context context, LauncherAppWidgetInfo info, IconCache cache, boolean disabledForSafeMode)70     public PendingAppWidgetHostView(Context context, LauncherAppWidgetInfo info,
71             IconCache cache, boolean disabledForSafeMode) {
72         super(new ContextThemeWrapper(context, R.style.WidgetContainerTheme));
73 
74         mInfo = info;
75         mStartState = info.restoreStatus;
76         mDisabledForSafeMode = disabledForSafeMode;
77 
78         mPaint = new TextPaint();
79         mPaint.setColor(Themes.getAttrColor(getContext(), android.R.attr.textColorPrimary));
80         mPaint.setTextSize(TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_PX,
81                 mLauncher.getDeviceProfile().iconTextSizePx, getResources().getDisplayMetrics()));
82         setBackgroundResource(R.drawable.pending_widget_bg);
83         setWillNotDraw(false);
84 
85         setElevation(getResources().getDimension(R.dimen.pending_widget_elevation));
86         updateAppWidget(null);
87         setOnClickListener(ItemClickHandler.INSTANCE);
88 
89         if (info.pendingItemInfo == null) {
90             info.pendingItemInfo = new PackageItemInfo(info.providerName.getPackageName());
91             info.pendingItemInfo.user = info.user;
92             cache.updateIconInBackground(this, info.pendingItemInfo);
93         } else {
94             reapplyItemInfo(info.pendingItemInfo);
95         }
96     }
97 
98     @Override
updateAppWidgetSize(Bundle newOptions, int minWidth, int minHeight, int maxWidth, int maxHeight)99     public void updateAppWidgetSize(Bundle newOptions, int minWidth, int minHeight, int maxWidth,
100             int maxHeight) {
101         // No-op
102     }
103 
104     @Override
getDefaultView()105     protected View getDefaultView() {
106         if (mDefaultView == null) {
107             mDefaultView = mInflater.inflate(R.layout.appwidget_not_ready, this, false);
108             mDefaultView.setOnClickListener(this);
109             applyState();
110         }
111         return mDefaultView;
112     }
113 
114     @Override
setOnClickListener(OnClickListener l)115     public void setOnClickListener(OnClickListener l) {
116         mClickListener = l;
117     }
118 
isReinflateIfNeeded()119     public boolean isReinflateIfNeeded() {
120         return mStartState != mInfo.restoreStatus;
121     }
122 
123     @Override
onSizeChanged(int w, int h, int oldw, int oldh)124     protected void onSizeChanged(int w, int h, int oldw, int oldh) {
125         super.onSizeChanged(w, h, oldw, oldh);
126         mDrawableSizeChanged = true;
127     }
128 
129     @Override
reapplyItemInfo(ItemInfoWithIcon info)130     public void reapplyItemInfo(ItemInfoWithIcon info) {
131         if (mCenterDrawable != null) {
132             mCenterDrawable.setCallback(null);
133             mCenterDrawable = null;
134         }
135         if (info.iconBitmap != null) {
136             // The view displays three modes,
137             //   1) App icon in the center
138             //   2) Preload icon in the center
139             //   3) Setup icon in the center and app icon in the top right corner.
140             DrawableFactory drawableFactory = DrawableFactory.get(getContext());
141             if (mDisabledForSafeMode) {
142                 FastBitmapDrawable disabledIcon = drawableFactory.newIcon(info);
143                 disabledIcon.setIsDisabled(true);
144                 mCenterDrawable = disabledIcon;
145                 mSettingIconDrawable = null;
146             } else if (isReadyForClickSetup()) {
147                 mCenterDrawable = drawableFactory.newIcon(info);
148                 mSettingIconDrawable = getResources().getDrawable(R.drawable.ic_setting).mutate();
149                 updateSettingColor(info.iconColor);
150             } else {
151                 mCenterDrawable = DrawableFactory.get(getContext())
152                         .newPendingIcon(info, getContext());
153                 mSettingIconDrawable = null;
154                 applyState();
155             }
156             mCenterDrawable.setCallback(this);
157             mDrawableSizeChanged = true;
158         }
159         invalidate();
160     }
161 
updateSettingColor(int dominantColor)162     private void updateSettingColor(int dominantColor) {
163         // Make the dominant color bright.
164         float[] hsv = new float[3];
165         Color.colorToHSV(dominantColor, hsv);
166         hsv[1] = Math.min(hsv[1], MIN_SATUNATION);
167         hsv[2] = 1;
168         mSettingIconDrawable.setColorFilter(Color.HSVToColor(hsv),  PorterDuff.Mode.SRC_IN);
169     }
170 
171     @Override
verifyDrawable(Drawable who)172     protected boolean verifyDrawable(Drawable who) {
173         return (who == mCenterDrawable) || super.verifyDrawable(who);
174     }
175 
applyState()176     public void applyState() {
177         if (mCenterDrawable != null) {
178             mCenterDrawable.setLevel(Math.max(mInfo.installProgress, 0));
179         }
180     }
181 
182     @Override
onClick(View v)183     public void onClick(View v) {
184         // AppWidgetHostView blocks all click events on the root view. Instead handle click events
185         // on the content and pass it along.
186         if (mClickListener != null) {
187             mClickListener.onClick(this);
188         }
189     }
190 
191     /**
192      * A pending widget is ready for setup after the provider is installed and
193      *   1) Widget id is not valid: the widget id is not yet bound to the provider, probably
194      *                              because the launcher doesn't have appropriate permissions.
195      *                              Note that we would still have an allocated id as that does not
196      *                              require any permissions and can be done during view inflation.
197      *   2) UI is not ready: the id is valid and the bound. But the widget has a configure activity
198      *                       which needs to be called once.
199      */
isReadyForClickSetup()200     public boolean isReadyForClickSetup() {
201         return !mInfo.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY)
202                 && (mInfo.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_UI_NOT_READY)
203                 || mInfo.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_ID_NOT_VALID));
204     }
205 
updateDrawableBounds()206     private void updateDrawableBounds() {
207         DeviceProfile grid = mLauncher.getDeviceProfile();
208         int paddingTop = getPaddingTop();
209         int paddingBottom = getPaddingBottom();
210         int paddingLeft = getPaddingLeft();
211         int paddingRight = getPaddingRight();
212 
213         int minPadding = getResources()
214                 .getDimensionPixelSize(R.dimen.pending_widget_min_padding);
215 
216         int availableWidth = getWidth() - paddingLeft - paddingRight - 2 * minPadding;
217         int availableHeight = getHeight() - paddingTop - paddingBottom - 2 * minPadding;
218 
219         if (mSettingIconDrawable == null) {
220             int maxSize = grid.iconSizePx;
221             int size = Math.min(maxSize, Math.min(availableWidth, availableHeight));
222 
223             mRect.set(0, 0, size, size);
224             mRect.offsetTo((getWidth() - mRect.width()) / 2, (getHeight() - mRect.height()) / 2);
225             mCenterDrawable.setBounds(mRect);
226         } else  {
227             float iconSize = Math.max(0, Math.min(availableWidth, availableHeight));
228 
229             // Use twice the setting size factor, as the setting is drawn at a corner and the
230             // icon is drawn in the center.
231             float settingIconScaleFactor = 1 + SETUP_ICON_SIZE_FACTOR * 2;
232             int maxSize = Math.max(availableWidth, availableHeight);
233             if (iconSize * settingIconScaleFactor > maxSize) {
234                 // There is an overlap
235                 iconSize = maxSize / settingIconScaleFactor;
236             }
237 
238             int actualIconSize = (int) Math.min(iconSize, grid.iconSizePx);
239 
240             // Icon top when we do not draw the text
241             int iconTop = (getHeight() - actualIconSize) / 2;
242             mSetupTextLayout = null;
243 
244             if (availableWidth > 0) {
245                 // Recreate the setup text.
246                 mSetupTextLayout = new StaticLayout(
247                         getResources().getText(R.string.gadget_setup_text), mPaint, availableWidth,
248                         Layout.Alignment.ALIGN_CENTER, 1, 0, true);
249                 int textHeight = mSetupTextLayout.getHeight();
250 
251                 // Extra icon size due to the setting icon
252                 float minHeightWithText = textHeight + actualIconSize * settingIconScaleFactor
253                         + grid.iconDrawablePaddingPx;
254 
255                 if (minHeightWithText < availableHeight) {
256                     // We can draw the text as well
257                     iconTop = (getHeight() - textHeight -
258                             grid.iconDrawablePaddingPx - actualIconSize) / 2;
259 
260                 } else {
261                     // We can't draw the text. Let the iconTop be same as before.
262                     mSetupTextLayout = null;
263                 }
264             }
265 
266             mRect.set(0, 0, actualIconSize, actualIconSize);
267             mRect.offset((getWidth() - actualIconSize) / 2, iconTop);
268             mCenterDrawable.setBounds(mRect);
269 
270             mRect.left = paddingLeft + minPadding;
271             mRect.right = mRect.left + (int) (SETUP_ICON_SIZE_FACTOR * actualIconSize);
272             mRect.top = paddingTop + minPadding;
273             mRect.bottom = mRect.top + (int) (SETUP_ICON_SIZE_FACTOR * actualIconSize);
274             mSettingIconDrawable.setBounds(mRect);
275 
276             if (mSetupTextLayout != null) {
277                 // Set up position for dragging the text
278                 mRect.left = paddingLeft + minPadding;
279                 mRect.top = mCenterDrawable.getBounds().bottom + grid.iconDrawablePaddingPx;
280             }
281         }
282     }
283 
284     @Override
onDraw(Canvas canvas)285     protected void onDraw(Canvas canvas) {
286         if (mCenterDrawable == null) {
287             // Nothing to draw
288             return;
289         }
290 
291         if (mDrawableSizeChanged) {
292             updateDrawableBounds();
293             mDrawableSizeChanged = false;
294         }
295 
296         mCenterDrawable.draw(canvas);
297         if (mSettingIconDrawable != null) {
298             mSettingIconDrawable.draw(canvas);
299         }
300         if (mSetupTextLayout != null) {
301             canvas.save();
302             canvas.translate(mRect.left, mRect.top);
303             mSetupTextLayout.draw(canvas);
304             canvas.restore();
305         }
306 
307     }
308 }
309