blob: 7ab88a0083da7c9b271e27bd3dd38e28cb5302b8 [file] [log] [blame]
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001/*
2 * Copyright (C) 2008 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
Daniel Sandler325dc232013-06-05 22:57:57 -040017package com.android.launcher3;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080018
Sunny Goyal726bee72018-03-05 12:54:24 -080019import static com.android.launcher3.anim.Interpolators.ACCEL;
Hyunyoung Songef468d82019-01-03 01:02:43 -080020import static com.android.launcher3.anim.Interpolators.DEACCEL;
Sunny Goyal726bee72018-03-05 12:54:24 -080021
Sunny Goyal508da152014-08-14 10:53:27 -070022import android.animation.ObjectAnimator;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080023import android.graphics.Bitmap;
24import android.graphics.Canvas;
Sunny Goyal508da152014-08-14 10:53:27 -070025import android.graphics.Color;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080026import android.graphics.ColorFilter;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070027import android.graphics.ColorMatrix;
28import android.graphics.ColorMatrixColorFilter;
Winson Chung45e1d6e2010-11-09 17:19:49 -080029import android.graphics.Paint;
30import android.graphics.PixelFormat;
Sunny Goyal508da152014-08-14 10:53:27 -070031import android.graphics.PorterDuff;
32import android.graphics.PorterDuffColorFilter;
Sunny Goyal726bee72018-03-05 12:54:24 -080033import android.graphics.Rect;
Winson Chung45e1d6e2010-11-09 17:19:49 -080034import android.graphics.drawable.Drawable;
Tony Wickham1e618492017-02-02 12:57:18 -080035import android.util.Property;
Sunny Goyal508da152014-08-14 10:53:27 -070036import android.util.SparseArray;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080037
Hyunyoung Song48cb7bc2018-09-25 17:03:34 -070038import com.android.launcher3.icons.BitmapInfo;
Tony Wickham9a8d11f2017-01-11 09:53:12 -080039
Hyunyoung Song3f471442015-04-08 19:01:34 -070040public class FastBitmapDrawable extends Drawable {
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080041
Sunny Goyal726bee72018-03-05 12:54:24 -080042 private static final float PRESSED_SCALE = 1.1f;
43
Tony Wickham6b910a22016-11-08 10:40:34 -080044 private static final float DISABLED_DESATURATION = 1f;
45 private static final float DISABLED_BRIGHTNESS = 0.5f;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070046
Sunny Goyal726bee72018-03-05 12:54:24 -080047 public static final int CLICK_FEEDBACK_DURATION = 200;
Sunny Goyal508da152014-08-14 10:53:27 -070048
Winsonc0880492015-08-21 11:16:27 -070049 // Since we don't need 256^2 values for combinations of both the brightness and saturation, we
50 // reduce the value space to a smaller value V, which reduces the number of cached
51 // ColorMatrixColorFilters that we need to keep to V^2
52 private static final int REDUCED_FILTER_VALUE_SPACE = 48;
Sunny Goyal95abbb32014-08-04 10:53:22 -070053
Winsonc0880492015-08-21 11:16:27 -070054 // A cache of ColorFilters for optimizing brightness and saturation animations
55 private static final SparseArray<ColorFilter> sCachedFilter = new SparseArray<>();
Sunny Goyal508da152014-08-14 10:53:27 -070056
Winsonc0880492015-08-21 11:16:27 -070057 // Temporary matrices used for calculation
58 private static final ColorMatrix sTempBrightnessMatrix = new ColorMatrix();
59 private static final ColorMatrix sTempFilterMatrix = new ColorMatrix();
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070060
Sunny Goyal55cb70b2016-11-12 09:58:29 -080061 protected final Paint mPaint = new Paint(Paint.FILTER_BITMAP_FLAG | Paint.ANTI_ALIAS_FLAG);
Sunny Goyal61e08462018-03-02 17:25:59 -080062 protected Bitmap mBitmap;
Sunny Goyal179249d2017-12-19 16:49:24 -080063 protected final int mIconColor;
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080064
65 private boolean mIsPressed;
Tony Wickham6b910a22016-11-08 10:40:34 -080066 private boolean mIsDisabled;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070067
Sunny Goyal726bee72018-03-05 12:54:24 -080068 // Animator and properties for the fast bitmap drawable's scale
69 private static final Property<FastBitmapDrawable, Float> SCALE
70 = new Property<FastBitmapDrawable, Float>(Float.TYPE, "scale") {
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080071 @Override
72 public Float get(FastBitmapDrawable fastBitmapDrawable) {
Sunny Goyal726bee72018-03-05 12:54:24 -080073 return fastBitmapDrawable.mScale;
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080074 }
75
76 @Override
77 public void set(FastBitmapDrawable fastBitmapDrawable, Float value) {
Sunny Goyal726bee72018-03-05 12:54:24 -080078 fastBitmapDrawable.mScale = value;
79 fastBitmapDrawable.invalidateSelf();
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080080 }
81 };
Sunny Goyal726bee72018-03-05 12:54:24 -080082 private ObjectAnimator mScaleAnimation;
83 private float mScale = 1;
84
Sunny Goyal2a76e3f2017-02-16 13:33:15 -080085
Winsonc0880492015-08-21 11:16:27 -070086 // The saturation and brightness are values that are mapped to REDUCED_FILTER_VALUE_SPACE and
87 // as a result, can be used to compose the key for the cached ColorMatrixColorFilters
88 private int mDesaturation = 0;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070089 private int mBrightness = 0;
Winsonc0880492015-08-21 11:16:27 -070090 private int mAlpha = 255;
91 private int mPrevUpdateKey = Integer.MAX_VALUE;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080092
Hyunyoung Song3f471442015-04-08 19:01:34 -070093 public FastBitmapDrawable(Bitmap b) {
Sunny Goyal179249d2017-12-19 16:49:24 -080094 this(b, Color.TRANSPARENT);
95 }
96
97 public FastBitmapDrawable(BitmapInfo info) {
98 this(info.icon, info.color);
99 }
100
101 public FastBitmapDrawable(ItemInfoWithIcon info) {
102 this(info.iconBitmap, info.iconColor);
103 }
104
105 protected FastBitmapDrawable(Bitmap b, int iconColor) {
Jon Mirandab6d686d2019-03-29 10:49:43 -0700106 this(b, iconColor, false);
107 }
108
109 protected FastBitmapDrawable(Bitmap b, int iconColor, boolean isDisabled) {
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800110 mBitmap = b;
Sunny Goyal179249d2017-12-19 16:49:24 -0800111 mIconColor = iconColor;
Sunny Goyal96ac68a2017-02-02 16:37:21 -0800112 setFilterBitmap(true);
Jon Mirandab6d686d2019-03-29 10:49:43 -0700113 setIsDisabled(isDisabled);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800114 }
115
116 @Override
Sunny Goyal726bee72018-03-05 12:54:24 -0800117 public final void draw(Canvas canvas) {
Matthew Ngeb9cc9d2018-06-25 15:32:24 -0700118 if (mScale != 1f) {
Sunny Goyal726bee72018-03-05 12:54:24 -0800119 int count = canvas.save();
120 Rect bounds = getBounds();
121 canvas.scale(mScale, mScale, bounds.exactCenterX(), bounds.exactCenterY());
122 drawInternal(canvas, bounds);
123 canvas.restoreToCount(count);
124 } else {
125 drawInternal(canvas, getBounds());
126 }
127 }
128
129 protected void drawInternal(Canvas canvas, Rect bounds) {
130 canvas.drawBitmap(mBitmap, null, bounds, mPaint);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800131 }
132
133 @Override
Adam Cohenbadf71e2011-05-26 19:08:29 -0700134 public void setColorFilter(ColorFilter cf) {
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700135 // No op
Adam Cohenbadf71e2011-05-26 19:08:29 -0700136 }
137
138 @Override
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800139 public int getOpacity() {
140 return PixelFormat.TRANSLUCENT;
141 }
142
143 @Override
144 public void setAlpha(int alpha) {
Winson Chung29d6fea2010-12-01 15:47:31 -0800145 mAlpha = alpha;
Winson Chungb3347bb2010-08-19 14:51:28 -0700146 mPaint.setAlpha(alpha);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800147 }
148
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700149 @Override
Adam Cohen76fc0852011-06-17 13:26:23 -0700150 public void setFilterBitmap(boolean filterBitmap) {
151 mPaint.setFilterBitmap(filterBitmap);
Winson Chung6e1c0d32013-10-25 15:24:24 -0700152 mPaint.setAntiAlias(filterBitmap);
Adam Cohen76fc0852011-06-17 13:26:23 -0700153 }
154
Winson Chung29d6fea2010-12-01 15:47:31 -0800155 public int getAlpha() {
156 return mAlpha;
157 }
158
Matthew Ngeb9cc9d2018-06-25 15:32:24 -0700159 public void setScale(float scale) {
160 if (mScaleAnimation != null) {
161 mScaleAnimation.cancel();
162 mScaleAnimation = null;
163 }
164 mScale = scale;
165 invalidateSelf();
166 }
167
Sunny Goyal726bee72018-03-05 12:54:24 -0800168 public float getAnimatedScale() {
169 return mScaleAnimation == null ? 1 : mScale;
170 }
171
Matthew Ngeb9cc9d2018-06-25 15:32:24 -0700172 public float getScale() {
173 return mScale;
174 }
175
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800176 @Override
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800177 public int getIntrinsicWidth() {
Sunny Goyalc424f222014-09-05 07:04:59 -0700178 return mBitmap.getWidth();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800179 }
180
181 @Override
182 public int getIntrinsicHeight() {
Sunny Goyalc424f222014-09-05 07:04:59 -0700183 return mBitmap.getHeight();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800184 }
185
186 @Override
187 public int getMinimumWidth() {
Winson Chungeeb5bbc2013-11-13 15:47:05 -0800188 return getBounds().width();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800189 }
190
191 @Override
192 public int getMinimumHeight() {
Winson Chungeeb5bbc2013-11-13 15:47:05 -0800193 return getBounds().height();
Joe Onorato0589f0f2010-02-08 13:44:00 -0800194 }
195
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800196 @Override
197 public boolean isStateful() {
198 return true;
Winsonc0880492015-08-21 11:16:27 -0700199 }
200
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800201 @Override
Sunny Goyal28141122017-06-21 17:28:23 -0700202 public ColorFilter getColorFilter() {
203 return mPaint.getColorFilter();
204 }
205
206 @Override
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800207 protected boolean onStateChange(int[] state) {
208 boolean isPressed = false;
209 for (int s : state) {
210 if (s == android.R.attr.state_pressed) {
211 isPressed = true;
212 break;
213 }
214 }
215 if (mIsPressed != isPressed) {
216 mIsPressed = isPressed;
Winsonc0880492015-08-21 11:16:27 -0700217
Sunny Goyal726bee72018-03-05 12:54:24 -0800218 if (mScaleAnimation != null) {
219 mScaleAnimation.cancel();
220 mScaleAnimation = null;
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800221 }
Winsonc0880492015-08-21 11:16:27 -0700222
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800223 if (mIsPressed) {
224 // Animate when going to pressed state
Sunny Goyal726bee72018-03-05 12:54:24 -0800225 mScaleAnimation = ObjectAnimator.ofFloat(this, SCALE, PRESSED_SCALE);
226 mScaleAnimation.setDuration(CLICK_FEEDBACK_DURATION);
227 mScaleAnimation.setInterpolator(ACCEL);
228 mScaleAnimation.start();
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800229 } else {
Hyunyoung Songef468d82019-01-03 01:02:43 -0800230 if (isVisible()) {
231 mScaleAnimation = ObjectAnimator.ofFloat(this, SCALE, 1f);
232 mScaleAnimation.setDuration(CLICK_FEEDBACK_DURATION);
233 mScaleAnimation.setInterpolator(DEACCEL);
234 mScaleAnimation.start();
235 } else {
236 mScale = 1f;
237 invalidateSelf();
238 }
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800239 }
Winsonc0880492015-08-21 11:16:27 -0700240 return true;
241 }
242 return false;
243 }
244
Tony Wickham6b910a22016-11-08 10:40:34 -0800245 private void invalidateDesaturationAndBrightness() {
Sunny Goyal2a76e3f2017-02-16 13:33:15 -0800246 setDesaturation(mIsDisabled ? DISABLED_DESATURATION : 0);
Sunny Goyal726bee72018-03-05 12:54:24 -0800247 setBrightness(mIsDisabled ? DISABLED_BRIGHTNESS : 0);
Winsonc0880492015-08-21 11:16:27 -0700248 }
249
Tony Wickham6b910a22016-11-08 10:40:34 -0800250 public void setIsDisabled(boolean isDisabled) {
251 if (mIsDisabled != isDisabled) {
252 mIsDisabled = isDisabled;
253 invalidateDesaturationAndBrightness();
254 }
255 }
256
Jon Mirandab6d686d2019-03-29 10:49:43 -0700257 protected boolean isDisabled() {
258 return mIsDisabled;
259 }
260
Winsonc0880492015-08-21 11:16:27 -0700261 /**
Winsonc0880492015-08-21 11:16:27 -0700262 * Sets the saturation of this icon, 0 [full color] -> 1 [desaturated]
263 */
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800264 private void setDesaturation(float desaturation) {
Winsonc0880492015-08-21 11:16:27 -0700265 int newDesaturation = (int) Math.floor(desaturation * REDUCED_FILTER_VALUE_SPACE);
266 if (mDesaturation != newDesaturation) {
267 mDesaturation = newDesaturation;
Sunny Goyal95abbb32014-08-04 10:53:22 -0700268 updateFilter();
269 }
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700270 }
271
Winsonc0880492015-08-21 11:16:27 -0700272 public float getDesaturation() {
273 return (float) mDesaturation / REDUCED_FILTER_VALUE_SPACE;
Sunny Goyal508da152014-08-14 10:53:27 -0700274 }
275
Winsonc0880492015-08-21 11:16:27 -0700276 /**
277 * Sets the brightness of this icon, 0 [no add. brightness] -> 1 [2bright2furious]
278 */
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800279 private void setBrightness(float brightness) {
Winsonc0880492015-08-21 11:16:27 -0700280 int newBrightness = (int) Math.floor(brightness * REDUCED_FILTER_VALUE_SPACE);
281 if (mBrightness != newBrightness) {
282 mBrightness = newBrightness;
Sunny Goyal95abbb32014-08-04 10:53:22 -0700283 updateFilter();
284 }
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700285 }
286
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800287 private float getBrightness() {
Winsonc0880492015-08-21 11:16:27 -0700288 return (float) mBrightness / REDUCED_FILTER_VALUE_SPACE;
289 }
290
291 /**
292 * Updates the paint to reflect the current brightness and saturation.
293 */
Sunny Goyal0ffab442018-06-07 17:31:48 -0700294 protected void updateFilter() {
Winsonc0880492015-08-21 11:16:27 -0700295 boolean usePorterDuffFilter = false;
296 int key = -1;
297 if (mDesaturation > 0) {
298 key = (mDesaturation << 16) | mBrightness;
299 } else if (mBrightness > 0) {
300 // Compose a key with a fully saturated icon if we are just animating brightness
301 key = (1 << 16) | mBrightness;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700302
Winsonc0880492015-08-21 11:16:27 -0700303 // We found that in L, ColorFilters cause drawing artifacts with shadows baked into
304 // icons, so just use a PorterDuff filter when we aren't animating saturation
305 usePorterDuffFilter = true;
306 }
Sunny Goyal95abbb32014-08-04 10:53:22 -0700307
Winsonc0880492015-08-21 11:16:27 -0700308 // Debounce multiple updates on the same frame
309 if (key == mPrevUpdateKey) {
310 return;
311 }
312 mPrevUpdateKey = key;
313
314 if (key != -1) {
315 ColorFilter filter = sCachedFilter.get(key);
Sunny Goyal508da152014-08-14 10:53:27 -0700316 if (filter == null) {
Winsonc0880492015-08-21 11:16:27 -0700317 float brightnessF = getBrightness();
318 int brightnessI = (int) (255 * brightnessF);
319 if (usePorterDuffFilter) {
320 filter = new PorterDuffColorFilter(Color.argb(brightnessI, 255, 255, 255),
321 PorterDuff.Mode.SRC_ATOP);
322 } else {
323 float saturationF = 1f - getDesaturation();
324 sTempFilterMatrix.setSaturation(saturationF);
325 if (mBrightness > 0) {
326 // Brightness: C-new = C-old*(1-amount) + amount
327 float scale = 1f - brightnessF;
328 float[] mat = sTempBrightnessMatrix.getArray();
329 mat[0] = scale;
330 mat[6] = scale;
331 mat[12] = scale;
332 mat[4] = brightnessI;
333 mat[9] = brightnessI;
334 mat[14] = brightnessI;
335 sTempFilterMatrix.preConcat(sTempBrightnessMatrix);
336 }
337 filter = new ColorMatrixColorFilter(sTempFilterMatrix);
338 }
339 sCachedFilter.append(key, filter);
Sunny Goyal508da152014-08-14 10:53:27 -0700340 }
341 mPaint.setColorFilter(filter);
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700342 } else {
343 mPaint.setColorFilter(null);
344 }
Winsonc0880492015-08-21 11:16:27 -0700345 invalidateSelf();
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700346 }
Sunny Goyal338d15d2018-02-23 12:19:44 -0800347
348 @Override
349 public ConstantState getConstantState() {
Jon Mirandab6d686d2019-03-29 10:49:43 -0700350 return new MyConstantState(mBitmap, mIconColor, mIsDisabled);
Sunny Goyal338d15d2018-02-23 12:19:44 -0800351 }
352
Sunny Goyal61e08462018-03-02 17:25:59 -0800353 protected static class MyConstantState extends ConstantState {
354 protected final Bitmap mBitmap;
355 protected final int mIconColor;
Jon Mirandab6d686d2019-03-29 10:49:43 -0700356 protected final boolean mIsDisabled;
Sunny Goyal338d15d2018-02-23 12:19:44 -0800357
Jon Mirandab6d686d2019-03-29 10:49:43 -0700358 public MyConstantState(Bitmap bitmap, int color, boolean isDisabled) {
Sunny Goyal338d15d2018-02-23 12:19:44 -0800359 mBitmap = bitmap;
360 mIconColor = color;
Jon Mirandab6d686d2019-03-29 10:49:43 -0700361 mIsDisabled = isDisabled;
Sunny Goyal338d15d2018-02-23 12:19:44 -0800362 }
363
364 @Override
365 public Drawable newDrawable() {
Jon Mirandab6d686d2019-03-29 10:49:43 -0700366 return new FastBitmapDrawable(mBitmap, mIconColor, mIsDisabled);
Sunny Goyal338d15d2018-02-23 12:19:44 -0800367 }
368
369 @Override
370 public int getChangingConfigurations() {
371 return 0;
372 }
373 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800374}