1 /*
2  * Copyright (C) 2016 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.settingslib.drawable;
18 
19 import android.annotation.NonNull;
20 import android.app.admin.DevicePolicyManager;
21 import android.content.Context;
22 import android.content.res.ColorStateList;
23 import android.graphics.Bitmap;
24 import android.graphics.BitmapShader;
25 import android.graphics.Canvas;
26 import android.graphics.Color;
27 import android.graphics.ColorFilter;
28 import android.graphics.Matrix;
29 import android.graphics.Paint;
30 import android.graphics.PixelFormat;
31 import android.graphics.PorterDuff;
32 import android.graphics.PorterDuffColorFilter;
33 import android.graphics.PorterDuffXfermode;
34 import android.graphics.Rect;
35 import android.graphics.RectF;
36 import android.graphics.Shader;
37 import android.graphics.drawable.BitmapDrawable;
38 import android.graphics.drawable.Drawable;
39 
40 import com.android.settingslib.R;
41 
42 /**
43  * Converts the user avatar icon to a circularly clipped one with an optional badge and frame
44  */
45 public class UserIconDrawable extends Drawable implements Drawable.Callback {
46 
47     private Drawable mUserDrawable;
48     private Bitmap mUserIcon;
49     private Bitmap mBitmap; // baked representation. Required for transparent border around badge
50     private final Paint mIconPaint = new Paint();
51     private final Paint mPaint = new Paint();
52     private final Matrix mIconMatrix = new Matrix();
53     private float mIntrinsicRadius;
54     private float mDisplayRadius;
55     private float mPadding = 0;
56     private int mSize = 0; // custom "intrinsic" size for this drawable if non-zero
57     private boolean mInvalidated = true;
58     private ColorStateList mTintColor = null;
59     private PorterDuff.Mode mTintMode = PorterDuff.Mode.SRC_ATOP;
60 
61     private float mFrameWidth;
62     private float mFramePadding;
63     private ColorStateList mFrameColor = null;
64     private Paint mFramePaint;
65 
66     private Drawable mBadge;
67     private Paint mClearPaint;
68     private float mBadgeRadius;
69     private float mBadgeMargin;
70 
71     /**
72      * Gets the system default managed-user badge as a drawable
73      * @param context
74      * @return drawable containing just the badge
75      */
getManagedUserBadgeDrawable(Context context)76     public static Drawable getManagedUserBadgeDrawable(Context context) {
77         int displayDensity = context.getResources().getDisplayMetrics().densityDpi;
78         return context.getResources().getDrawableForDensity(
79                 com.android.internal.R.drawable.ic_corp_user_badge,
80                 displayDensity, context.getTheme());
81     }
82 
83     /**
84      * Gets the preferred list-item size of this drawable.
85      * @param context
86      * @return size in pixels
87      */
getSizeForList(Context context)88     public static int getSizeForList(Context context) {
89         return (int) context.getResources().getDimension(R.dimen.circle_avatar_size);
90     }
91 
UserIconDrawable()92     public UserIconDrawable() {
93         this(0);
94     }
95 
96     /**
97      * Use this constructor if the drawable is intended to be placed in listviews
98      * @param intrinsicSize if 0, the intrinsic size will come from the icon itself
99      */
UserIconDrawable(int intrinsicSize)100     public UserIconDrawable(int intrinsicSize) {
101         super();
102         mIconPaint.setAntiAlias(true);
103         mIconPaint.setFilterBitmap(true);
104         mPaint.setFilterBitmap(true);
105         mPaint.setAntiAlias(true);
106         if (intrinsicSize > 0) {
107             setBounds(0, 0, intrinsicSize, intrinsicSize);
108             setIntrinsicSize(intrinsicSize);
109         }
110         setIcon(null);
111     }
112 
setIcon(Bitmap icon)113     public UserIconDrawable setIcon(Bitmap icon) {
114         if (mUserDrawable != null) {
115             mUserDrawable.setCallback(null);
116             mUserDrawable = null;
117         }
118         mUserIcon = icon;
119         if (mUserIcon == null) {
120             mIconPaint.setShader(null);
121             mBitmap = null;
122         } else {
123             mIconPaint.setShader(new BitmapShader(icon, Shader.TileMode.CLAMP,
124                     Shader.TileMode.CLAMP));
125         }
126         onBoundsChange(getBounds());
127         return this;
128     }
129 
setIconDrawable(Drawable icon)130     public UserIconDrawable setIconDrawable(Drawable icon) {
131         if (mUserDrawable != null) {
132             mUserDrawable.setCallback(null);
133         }
134         mUserIcon = null;
135         mUserDrawable = icon;
136         if (mUserDrawable == null) {
137             mBitmap = null;
138         } else {
139             mUserDrawable.setCallback(this);
140         }
141         onBoundsChange(getBounds());
142         return this;
143     }
144 
setBadge(Drawable badge)145     public UserIconDrawable setBadge(Drawable badge) {
146         mBadge = badge;
147         if (mBadge != null) {
148             if (mClearPaint == null) {
149                 mClearPaint = new Paint();
150                 mClearPaint.setAntiAlias(true);
151                 mClearPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
152                 mClearPaint.setStyle(Paint.Style.FILL);
153             }
154             // update metrics
155             onBoundsChange(getBounds());
156         } else {
157             invalidateSelf();
158         }
159         return this;
160     }
161 
setBadgeIfManagedUser(Context context, int userId)162     public UserIconDrawable setBadgeIfManagedUser(Context context, int userId) {
163         Drawable badge = null;
164         boolean isManaged = context.getSystemService(DevicePolicyManager.class)
165                 .getProfileOwnerAsUser(userId) != null;
166         if (isManaged) {
167             badge = getManagedUserBadgeDrawable(context);
168         }
169         return setBadge(badge);
170     }
171 
setBadgeRadius(float radius)172     public void setBadgeRadius(float radius) {
173         mBadgeRadius = radius;
174         onBoundsChange(getBounds());
175     }
176 
setBadgeMargin(float margin)177     public void setBadgeMargin(float margin) {
178         mBadgeMargin = margin;
179         onBoundsChange(getBounds());
180     }
181 
182     /**
183      * Sets global padding of icon/frame. Doesn't effect the badge.
184      * @param padding
185      */
setPadding(float padding)186     public void setPadding(float padding) {
187         mPadding = padding;
188         onBoundsChange(getBounds());
189     }
190 
initFramePaint()191     private void initFramePaint() {
192         if (mFramePaint == null) {
193             mFramePaint = new Paint();
194             mFramePaint.setStyle(Paint.Style.STROKE);
195             mFramePaint.setAntiAlias(true);
196         }
197     }
198 
setFrameWidth(float width)199     public void setFrameWidth(float width) {
200         initFramePaint();
201         mFrameWidth = width;
202         mFramePaint.setStrokeWidth(width);
203         onBoundsChange(getBounds());
204     }
205 
setFramePadding(float padding)206     public void setFramePadding(float padding) {
207         initFramePaint();
208         mFramePadding = padding;
209         onBoundsChange(getBounds());
210     }
211 
setFrameColor(int color)212     public void setFrameColor(int color) {
213         initFramePaint();
214         mFramePaint.setColor(color);
215         invalidateSelf();
216     }
217 
setFrameColor(ColorStateList colorList)218     public void setFrameColor(ColorStateList colorList) {
219         initFramePaint();
220         mFrameColor = colorList;
221         invalidateSelf();
222     }
223 
224     /**
225      * This sets the "intrinsic" size of this drawable. Useful for views which use the drawable's
226      * intrinsic size for layout. It is independent of the bounds.
227      * @param size if 0, the intrinsic size will be set to the displayed icon's size
228      */
setIntrinsicSize(int size)229     public void setIntrinsicSize(int size) {
230         mSize = size;
231     }
232 
233     @Override
draw(Canvas canvas)234     public void draw(Canvas canvas) {
235         if (mInvalidated) {
236             rebake();
237         }
238         if (mBitmap != null) {
239             if (mTintColor == null) {
240                 mPaint.setColorFilter(null);
241             } else {
242                 int color = mTintColor.getColorForState(getState(), mTintColor.getDefaultColor());
243                 if (mPaint.getColorFilter() == null) {
244                     mPaint.setColorFilter(new PorterDuffColorFilter(color, mTintMode));
245                 } else {
246                     ((PorterDuffColorFilter) mPaint.getColorFilter()).setMode(mTintMode);
247                     ((PorterDuffColorFilter) mPaint.getColorFilter()).setColor(color);
248                 }
249             }
250 
251             canvas.drawBitmap(mBitmap, 0, 0, mPaint);
252         }
253     }
254 
255     @Override
setAlpha(int alpha)256     public void setAlpha(int alpha) {
257         mPaint.setAlpha(alpha);
258         super.invalidateSelf();
259     }
260 
261     @Override
setColorFilter(ColorFilter colorFilter)262     public void setColorFilter(ColorFilter colorFilter) {
263     }
264 
265     @Override
setTintList(ColorStateList tintList)266     public void setTintList(ColorStateList tintList) {
267         mTintColor = tintList;
268         super.invalidateSelf();
269     }
270 
271     @Override
setTintMode(@onNull PorterDuff.Mode mode)272     public void setTintMode(@NonNull PorterDuff.Mode mode) {
273         mTintMode = mode;
274         super.invalidateSelf();
275     }
276 
277     @Override
getConstantState()278     public ConstantState getConstantState() {
279         return new BitmapDrawable(mBitmap).getConstantState();
280     }
281 
282     /**
283      * This 'bakes' the current state of this icon into a bitmap and removes/recycles the source
284      * bitmap/drawable. Use this when no more changes will be made and an intrinsic size is set.
285      * This effectively turns this into a static drawable.
286      */
bake()287     public UserIconDrawable bake() {
288         if (mSize <= 0) {
289             throw new IllegalStateException("Baking requires an explicit intrinsic size");
290         }
291         onBoundsChange(new Rect(0, 0, mSize, mSize));
292         rebake();
293         mFrameColor = null;
294         mFramePaint = null;
295         mClearPaint = null;
296         if (mUserDrawable != null) {
297             mUserDrawable.setCallback(null);
298             mUserDrawable = null;
299         } else if (mUserIcon != null) {
300             mUserIcon.recycle();
301             mUserIcon = null;
302         }
303         return this;
304     }
305 
rebake()306     private void rebake() {
307         mInvalidated = false;
308 
309         if (mBitmap == null || (mUserDrawable == null && mUserIcon == null)) {
310             return;
311         }
312 
313         final Canvas canvas = new Canvas(mBitmap);
314         canvas.drawColor(0, PorterDuff.Mode.CLEAR);
315 
316         if(mUserDrawable != null) {
317             mUserDrawable.draw(canvas);
318         } else if (mUserIcon != null) {
319             int saveId = canvas.save();
320             canvas.concat(mIconMatrix);
321             canvas.drawCircle(mUserIcon.getWidth() * 0.5f, mUserIcon.getHeight() * 0.5f,
322                     mIntrinsicRadius, mIconPaint);
323             canvas.restoreToCount(saveId);
324         }
325 
326         if (mFrameColor != null) {
327             mFramePaint.setColor(mFrameColor.getColorForState(getState(), Color.TRANSPARENT));
328         }
329         if ((mFrameWidth + mFramePadding) > 0.001f) {
330             float radius = mDisplayRadius - mPadding - mFrameWidth * 0.5f;
331             canvas.drawCircle(getBounds().exactCenterX(), getBounds().exactCenterY(),
332                     radius, mFramePaint);
333         }
334 
335         if ((mBadge != null) && (mBadgeRadius > 0.001f)) {
336             final float badgeDiameter = mBadgeRadius * 2f;
337             final float badgeTop = mBitmap.getHeight() - badgeDiameter;
338             float badgeLeft = mBitmap.getWidth() - badgeDiameter;
339 
340             mBadge.setBounds((int) badgeLeft, (int) badgeTop,
341                     (int) (badgeLeft + badgeDiameter), (int) (badgeTop + badgeDiameter));
342 
343             final float borderRadius = mBadge.getBounds().width() * 0.5f + mBadgeMargin;
344             canvas.drawCircle(badgeLeft + mBadgeRadius, badgeTop + mBadgeRadius,
345                     borderRadius, mClearPaint);
346 
347             mBadge.draw(canvas);
348         }
349     }
350 
351     @Override
onBoundsChange(Rect bounds)352     protected void onBoundsChange(Rect bounds) {
353         if (bounds.isEmpty() || (mUserIcon == null && mUserDrawable == null)) {
354             return;
355         }
356 
357         // re-create bitmap if applicable
358         float newDisplayRadius = Math.min(bounds.width(), bounds.height()) * 0.5f;
359         int size = (int) (newDisplayRadius * 2);
360         if (mBitmap == null || size != ((int) (mDisplayRadius * 2))) {
361             mDisplayRadius = newDisplayRadius;
362             if (mBitmap != null) {
363                 mBitmap.recycle();
364             }
365             mBitmap = Bitmap.createBitmap(size, size, Bitmap.Config.ARGB_8888);
366         }
367 
368         // update metrics
369         mDisplayRadius = Math.min(bounds.width(), bounds.height()) * 0.5f;
370         final float iconRadius = mDisplayRadius - mFrameWidth - mFramePadding - mPadding;
371         RectF dstRect = new RectF(bounds.exactCenterX() - iconRadius,
372                                   bounds.exactCenterY() - iconRadius,
373                                   bounds.exactCenterX() + iconRadius,
374                                   bounds.exactCenterY() + iconRadius);
375         if (mUserDrawable != null) {
376             Rect rounded = new Rect();
377             dstRect.round(rounded);
378             mIntrinsicRadius = Math.min(mUserDrawable.getIntrinsicWidth(),
379                                         mUserDrawable.getIntrinsicHeight()) * 0.5f;
380             mUserDrawable.setBounds(rounded);
381         } else if (mUserIcon != null) {
382             // Build square-to-square transformation matrix
383             final float iconCX = mUserIcon.getWidth() * 0.5f;
384             final float iconCY = mUserIcon.getHeight() * 0.5f;
385             mIntrinsicRadius = Math.min(iconCX, iconCY);
386             RectF srcRect = new RectF(iconCX - mIntrinsicRadius, iconCY - mIntrinsicRadius,
387                                       iconCX + mIntrinsicRadius, iconCY + mIntrinsicRadius);
388             mIconMatrix.setRectToRect(srcRect, dstRect, Matrix.ScaleToFit.FILL);
389         }
390 
391         invalidateSelf();
392     }
393 
394     @Override
invalidateSelf()395     public void invalidateSelf() {
396         super.invalidateSelf();
397         mInvalidated = true;
398     }
399 
400     @Override
isStateful()401     public boolean isStateful() {
402         return mFrameColor != null && mFrameColor.isStateful();
403     }
404 
405     @Override
getOpacity()406     public int getOpacity() {
407         return PixelFormat.TRANSLUCENT;
408     }
409 
410     @Override
getIntrinsicWidth()411     public int getIntrinsicWidth() {
412         return (mSize <= 0 ? (int) mIntrinsicRadius * 2 : mSize);
413     }
414 
415     @Override
getIntrinsicHeight()416     public int getIntrinsicHeight() {
417         return getIntrinsicWidth();
418     }
419 
420     @Override
invalidateDrawable(@onNull Drawable who)421     public void invalidateDrawable(@NonNull Drawable who) {
422         invalidateSelf();
423     }
424 
425     @Override
scheduleDrawable(@onNull Drawable who, @NonNull Runnable what, long when)426     public void scheduleDrawable(@NonNull Drawable who, @NonNull Runnable what, long when) {
427         scheduleSelf(what, when);
428     }
429 
430     @Override
unscheduleDrawable(@onNull Drawable who, @NonNull Runnable what)431     public void unscheduleDrawable(@NonNull Drawable who, @NonNull Runnable what) {
432         unscheduleSelf(what);
433     }
434 }
435