blob: df195471bc9473818f218926db1012fe8b5ce5cf [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
Winsonc0880492015-08-21 11:16:27 -070019import android.animation.AnimatorSet;
Sunny Goyal508da152014-08-14 10:53:27 -070020import android.animation.ObjectAnimator;
21import android.animation.TimeInterpolator;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080022import android.graphics.Bitmap;
23import android.graphics.Canvas;
Sunny Goyal508da152014-08-14 10:53:27 -070024import android.graphics.Color;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080025import android.graphics.ColorFilter;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070026import android.graphics.ColorMatrix;
27import android.graphics.ColorMatrixColorFilter;
Winson Chung45e1d6e2010-11-09 17:19:49 -080028import android.graphics.Paint;
29import android.graphics.PixelFormat;
Sunny Goyal508da152014-08-14 10:53:27 -070030import android.graphics.PorterDuff;
31import android.graphics.PorterDuffColorFilter;
Winson Chung45e1d6e2010-11-09 17:19:49 -080032import android.graphics.drawable.Drawable;
Tony Wickham010d2552017-01-20 08:15:28 -080033import android.util.Log;
Sunny Goyal508da152014-08-14 10:53:27 -070034import android.util.SparseArray;
Winsonc0880492015-08-21 11:16:27 -070035import android.view.animation.DecelerateInterpolator;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080036
Tony Wickham9a8d11f2017-01-11 09:53:12 -080037import com.android.launcher3.badge.BadgeInfo;
Tony Wickham010d2552017-01-20 08:15:28 -080038import com.android.launcher3.badge.BadgeRenderer;
39import com.android.launcher3.graphics.IconPalette;
Tony Wickham9a8d11f2017-01-11 09:53:12 -080040
Hyunyoung Song3f471442015-04-08 19:01:34 -070041public class FastBitmapDrawable extends Drawable {
Tony Wickham6b910a22016-11-08 10:40:34 -080042 private static final float DISABLED_DESATURATION = 1f;
43 private static final float DISABLED_BRIGHTNESS = 0.5f;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -070044
Winsonc0880492015-08-21 11:16:27 -070045 /**
46 * The possible states that a FastBitmapDrawable can be in.
47 */
48 public enum State {
49
50 NORMAL (0f, 0f, 1f, new DecelerateInterpolator()),
51 PRESSED (0f, 100f / 255f, 1f, CLICK_FEEDBACK_INTERPOLATOR),
Winsonc08c59d2015-10-28 15:30:38 -070052 FAST_SCROLL_HIGHLIGHTED (0f, 0f, 1.15f, new DecelerateInterpolator()),
Tony Wickham6b910a22016-11-08 10:40:34 -080053 FAST_SCROLL_UNHIGHLIGHTED (0f, 0f, 1f, new DecelerateInterpolator());
Winsonc0880492015-08-21 11:16:27 -070054
55 public final float desaturation;
56 public final float brightness;
57 /**
58 * Used specifically by the view drawing this FastBitmapDrawable.
59 */
60 public final float viewScale;
61 public final TimeInterpolator interpolator;
62
63 State(float desaturation, float brightness, float viewScale, TimeInterpolator interpolator) {
64 this.desaturation = desaturation;
65 this.brightness = brightness;
66 this.viewScale = viewScale;
67 this.interpolator = interpolator;
68 }
69 }
70
71 public static final TimeInterpolator CLICK_FEEDBACK_INTERPOLATOR = new TimeInterpolator() {
Sunny Goyal508da152014-08-14 10:53:27 -070072
73 @Override
74 public float getInterpolation(float input) {
75 if (input < 0.05f) {
76 return input / 0.05f;
77 } else if (input < 0.3f){
78 return 1;
79 } else {
80 return (1 - input) / 0.7f;
81 }
82 }
83 };
Winsonc0880492015-08-21 11:16:27 -070084 public static final int CLICK_FEEDBACK_DURATION = 2000;
85 public static final int FAST_SCROLL_HIGHLIGHT_DURATION = 225;
86 public static final int FAST_SCROLL_UNHIGHLIGHT_DURATION = 150;
87 public static final int FAST_SCROLL_UNHIGHLIGHT_FROM_NORMAL_DURATION = 225;
88 public static final int FAST_SCROLL_INACTIVE_DURATION = 275;
Sunny Goyal508da152014-08-14 10:53:27 -070089
Winsonc0880492015-08-21 11:16:27 -070090 // Since we don't need 256^2 values for combinations of both the brightness and saturation, we
91 // reduce the value space to a smaller value V, which reduces the number of cached
92 // ColorMatrixColorFilters that we need to keep to V^2
93 private static final int REDUCED_FILTER_VALUE_SPACE = 48;
Sunny Goyal95abbb32014-08-04 10:53:22 -070094
Winsonc0880492015-08-21 11:16:27 -070095 // A cache of ColorFilters for optimizing brightness and saturation animations
96 private static final SparseArray<ColorFilter> sCachedFilter = new SparseArray<>();
Sunny Goyal508da152014-08-14 10:53:27 -070097
Winsonc0880492015-08-21 11:16:27 -070098 // Temporary matrices used for calculation
99 private static final ColorMatrix sTempBrightnessMatrix = new ColorMatrix();
100 private static final ColorMatrix sTempFilterMatrix = new ColorMatrix();
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700101
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800102 protected final Paint mPaint = new Paint(Paint.FILTER_BITMAP_FLAG | Paint.ANTI_ALIAS_FLAG);
Sunny Goyal508da152014-08-14 10:53:27 -0700103 private final Bitmap mBitmap;
Winsonc0880492015-08-21 11:16:27 -0700104 private State mState = State.NORMAL;
Tony Wickham6b910a22016-11-08 10:40:34 -0800105 private boolean mIsDisabled;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700106
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800107 private BadgeInfo mBadgeInfo;
108 private BadgeRenderer mBadgeRenderer;
109 private IconPalette mIconPalette;
110
Winsonc0880492015-08-21 11:16:27 -0700111 // The saturation and brightness are values that are mapped to REDUCED_FILTER_VALUE_SPACE and
112 // as a result, can be used to compose the key for the cached ColorMatrixColorFilters
113 private int mDesaturation = 0;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700114 private int mBrightness = 0;
Winsonc0880492015-08-21 11:16:27 -0700115 private int mAlpha = 255;
116 private int mPrevUpdateKey = Integer.MAX_VALUE;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800117
Winsonc0880492015-08-21 11:16:27 -0700118 // Animators for the fast bitmap drawable's properties
119 private AnimatorSet mPropertyAnimator;
Sunny Goyal508da152014-08-14 10:53:27 -0700120
Hyunyoung Song3f471442015-04-08 19:01:34 -0700121 public FastBitmapDrawable(Bitmap b) {
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800122 mBitmap = b;
Winson Chung268f1c52013-11-18 14:04:41 -0800123 setBounds(0, 0, b.getWidth(), b.getHeight());
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800124 }
125
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800126 public void applyIconBadge(BadgeInfo badgeInfo, BadgeRenderer badgeRenderer) {
Tony Wickham010d2552017-01-20 08:15:28 -0800127 boolean wasBadged = mBadgeInfo != null;
128 boolean isBadged = badgeInfo != null;
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800129 mBadgeInfo = badgeInfo;
130 mBadgeRenderer = badgeRenderer;
Tony Wickham010d2552017-01-20 08:15:28 -0800131 if (wasBadged || isBadged) {
Tony Wickham9438ed42017-01-20 09:38:25 -0800132 mIconPalette = getIconPalette();
Tony Wickham010d2552017-01-20 08:15:28 -0800133 invalidateSelf();
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800134 }
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800135 }
136
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800137 @Override
138 public void draw(Canvas canvas) {
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800139 drawInternal(canvas);
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800140 // Draw the icon badge in the top right corner.
141 drawBadgeIfNecessary(canvas);
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800142 }
143
144 public void drawWithBrightness(Canvas canvas, float brightness) {
145 float oldBrightness = getBrightness();
146 setBrightness(brightness);
147 drawInternal(canvas);
148 setBrightness(oldBrightness);
149 }
150
151 protected void drawInternal(Canvas canvas) {
Winsonc0880492015-08-21 11:16:27 -0700152 canvas.drawBitmap(mBitmap, null, getBounds(), mPaint);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800153 }
154
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800155 protected void drawBadgeIfNecessary(Canvas canvas) {
156 if (hasBadge()) {
157 mBadgeRenderer.draw(canvas, mIconPalette, mBadgeInfo, getBounds());
158 }
159 }
160
Tony Wickham9438ed42017-01-20 09:38:25 -0800161 public IconPalette getIconPalette() {
162 if (mIconPalette == null) {
163 mIconPalette = IconPalette.fromDominantColor(Utilities
164 .findDominantColorByHue(mBitmap, 20));
165 }
166 return mIconPalette;
167 }
168
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800169 private boolean hasBadge() {
Tony Wickham010d2552017-01-20 08:15:28 -0800170 return mBadgeInfo != null && mBadgeInfo.getNotificationCount() != 0;
Tony Wickham9a8d11f2017-01-11 09:53:12 -0800171 }
172
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800173 @Override
Adam Cohenbadf71e2011-05-26 19:08:29 -0700174 public void setColorFilter(ColorFilter cf) {
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700175 // No op
Adam Cohenbadf71e2011-05-26 19:08:29 -0700176 }
177
178 @Override
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800179 public int getOpacity() {
180 return PixelFormat.TRANSLUCENT;
181 }
182
183 @Override
184 public void setAlpha(int alpha) {
Winson Chung29d6fea2010-12-01 15:47:31 -0800185 mAlpha = alpha;
Winson Chungb3347bb2010-08-19 14:51:28 -0700186 mPaint.setAlpha(alpha);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800187 }
188
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700189 @Override
Adam Cohen76fc0852011-06-17 13:26:23 -0700190 public void setFilterBitmap(boolean filterBitmap) {
191 mPaint.setFilterBitmap(filterBitmap);
Winson Chung6e1c0d32013-10-25 15:24:24 -0700192 mPaint.setAntiAlias(filterBitmap);
Adam Cohen76fc0852011-06-17 13:26:23 -0700193 }
194
Winson Chung29d6fea2010-12-01 15:47:31 -0800195 public int getAlpha() {
196 return mAlpha;
197 }
198
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800199 @Override
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800200 public int getIntrinsicWidth() {
Sunny Goyalc424f222014-09-05 07:04:59 -0700201 return mBitmap.getWidth();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800202 }
203
204 @Override
205 public int getIntrinsicHeight() {
Sunny Goyalc424f222014-09-05 07:04:59 -0700206 return mBitmap.getHeight();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800207 }
208
209 @Override
210 public int getMinimumWidth() {
Winson Chungeeb5bbc2013-11-13 15:47:05 -0800211 return getBounds().width();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800212 }
213
214 @Override
215 public int getMinimumHeight() {
Winson Chungeeb5bbc2013-11-13 15:47:05 -0800216 return getBounds().height();
Joe Onorato0589f0f2010-02-08 13:44:00 -0800217 }
218
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800219 public Bitmap getBitmap() {
220 return mBitmap;
221 }
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700222
Sunny Goyal95abbb32014-08-04 10:53:22 -0700223 /**
Winsonc0880492015-08-21 11:16:27 -0700224 * Animates this drawable to a new state.
225 *
226 * @return whether the state has changed.
Sunny Goyal95abbb32014-08-04 10:53:22 -0700227 */
Winsonc0880492015-08-21 11:16:27 -0700228 public boolean animateState(State newState) {
229 State prevState = mState;
230 if (mState != newState) {
231 mState = newState;
232
Tony Wickham6b910a22016-11-08 10:40:34 -0800233 float desaturation = mIsDisabled ? DISABLED_DESATURATION : newState.desaturation;
234 float brightness = mIsDisabled ? DISABLED_BRIGHTNESS: newState.brightness;
235
Winsonc0880492015-08-21 11:16:27 -0700236 mPropertyAnimator = cancelAnimator(mPropertyAnimator);
237 mPropertyAnimator = new AnimatorSet();
238 mPropertyAnimator.playTogether(
Tony Wickham6b910a22016-11-08 10:40:34 -0800239 ObjectAnimator.ofFloat(this, "desaturation", desaturation),
240 ObjectAnimator.ofFloat(this, "brightness", brightness));
Winsonc0880492015-08-21 11:16:27 -0700241 mPropertyAnimator.setInterpolator(newState.interpolator);
242 mPropertyAnimator.setDuration(getDurationForStateChange(prevState, newState));
243 mPropertyAnimator.setStartDelay(getStartDelayForStateChange(prevState, newState));
244 mPropertyAnimator.start();
245 return true;
246 }
247 return false;
248 }
249
250 /**
251 * Immediately sets this drawable to a new state.
252 *
253 * @return whether the state has changed.
254 */
255 public boolean setState(State newState) {
256 if (mState != newState) {
257 mState = newState;
258
259 mPropertyAnimator = cancelAnimator(mPropertyAnimator);
260
Tony Wickham6b910a22016-11-08 10:40:34 -0800261 invalidateDesaturationAndBrightness();
Winsonc0880492015-08-21 11:16:27 -0700262 return true;
263 }
264 return false;
265 }
266
Tony Wickham6b910a22016-11-08 10:40:34 -0800267 private void invalidateDesaturationAndBrightness() {
268 setDesaturation(mIsDisabled ? DISABLED_DESATURATION : mState.desaturation);
269 setBrightness(mIsDisabled ? DISABLED_BRIGHTNESS: mState.brightness);
270 }
271
Winsonc0880492015-08-21 11:16:27 -0700272 /**
273 * Returns the current state.
274 */
275 public State getCurrentState() {
276 return mState;
277 }
278
Tony Wickham6b910a22016-11-08 10:40:34 -0800279 public void setIsDisabled(boolean isDisabled) {
280 if (mIsDisabled != isDisabled) {
281 mIsDisabled = isDisabled;
282 invalidateDesaturationAndBrightness();
283 }
284 }
285
Winsonc0880492015-08-21 11:16:27 -0700286 /**
287 * Returns the duration for the state change animation.
288 */
289 public static int getDurationForStateChange(State fromState, State toState) {
290 switch (toState) {
291 case NORMAL:
292 switch (fromState) {
293 case PRESSED:
294 return 0;
295 case FAST_SCROLL_HIGHLIGHTED:
296 case FAST_SCROLL_UNHIGHLIGHTED:
297 return FAST_SCROLL_INACTIVE_DURATION;
298 }
299 case PRESSED:
300 return CLICK_FEEDBACK_DURATION;
301 case FAST_SCROLL_HIGHLIGHTED:
302 return FAST_SCROLL_HIGHLIGHT_DURATION;
303 case FAST_SCROLL_UNHIGHLIGHTED:
304 switch (fromState) {
305 case NORMAL:
306 // When animating from normal state, take a little longer
307 return FAST_SCROLL_UNHIGHLIGHT_FROM_NORMAL_DURATION;
308 default:
309 return FAST_SCROLL_UNHIGHLIGHT_DURATION;
310 }
311 }
312 return 0;
313 }
314
315 /**
316 * Returns the start delay when animating between certain fast scroll states.
317 */
318 public static int getStartDelayForStateChange(State fromState, State toState) {
319 switch (toState) {
320 case FAST_SCROLL_UNHIGHLIGHTED:
321 switch (fromState) {
322 case NORMAL:
323 return FAST_SCROLL_UNHIGHLIGHT_DURATION / 4;
324 }
325 }
326 return 0;
327 }
328
329 /**
330 * Sets the saturation of this icon, 0 [full color] -> 1 [desaturated]
331 */
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800332 private void setDesaturation(float desaturation) {
Winsonc0880492015-08-21 11:16:27 -0700333 int newDesaturation = (int) Math.floor(desaturation * REDUCED_FILTER_VALUE_SPACE);
334 if (mDesaturation != newDesaturation) {
335 mDesaturation = newDesaturation;
Sunny Goyal95abbb32014-08-04 10:53:22 -0700336 updateFilter();
337 }
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700338 }
339
Winsonc0880492015-08-21 11:16:27 -0700340 public float getDesaturation() {
341 return (float) mDesaturation / REDUCED_FILTER_VALUE_SPACE;
Sunny Goyal508da152014-08-14 10:53:27 -0700342 }
343
Winsonc0880492015-08-21 11:16:27 -0700344 /**
345 * Sets the brightness of this icon, 0 [no add. brightness] -> 1 [2bright2furious]
346 */
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800347 private void setBrightness(float brightness) {
Winsonc0880492015-08-21 11:16:27 -0700348 int newBrightness = (int) Math.floor(brightness * REDUCED_FILTER_VALUE_SPACE);
349 if (mBrightness != newBrightness) {
350 mBrightness = newBrightness;
Sunny Goyal95abbb32014-08-04 10:53:22 -0700351 updateFilter();
352 }
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700353 }
354
Sunny Goyal55cb70b2016-11-12 09:58:29 -0800355 private float getBrightness() {
Winsonc0880492015-08-21 11:16:27 -0700356 return (float) mBrightness / REDUCED_FILTER_VALUE_SPACE;
357 }
358
359 /**
360 * Updates the paint to reflect the current brightness and saturation.
361 */
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700362 private void updateFilter() {
Winsonc0880492015-08-21 11:16:27 -0700363 boolean usePorterDuffFilter = false;
364 int key = -1;
365 if (mDesaturation > 0) {
366 key = (mDesaturation << 16) | mBrightness;
367 } else if (mBrightness > 0) {
368 // Compose a key with a fully saturated icon if we are just animating brightness
369 key = (1 << 16) | mBrightness;
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700370
Winsonc0880492015-08-21 11:16:27 -0700371 // We found that in L, ColorFilters cause drawing artifacts with shadows baked into
372 // icons, so just use a PorterDuff filter when we aren't animating saturation
373 usePorterDuffFilter = true;
374 }
Sunny Goyal95abbb32014-08-04 10:53:22 -0700375
Winsonc0880492015-08-21 11:16:27 -0700376 // Debounce multiple updates on the same frame
377 if (key == mPrevUpdateKey) {
378 return;
379 }
380 mPrevUpdateKey = key;
381
382 if (key != -1) {
383 ColorFilter filter = sCachedFilter.get(key);
Sunny Goyal508da152014-08-14 10:53:27 -0700384 if (filter == null) {
Winsonc0880492015-08-21 11:16:27 -0700385 float brightnessF = getBrightness();
386 int brightnessI = (int) (255 * brightnessF);
387 if (usePorterDuffFilter) {
388 filter = new PorterDuffColorFilter(Color.argb(brightnessI, 255, 255, 255),
389 PorterDuff.Mode.SRC_ATOP);
390 } else {
391 float saturationF = 1f - getDesaturation();
392 sTempFilterMatrix.setSaturation(saturationF);
393 if (mBrightness > 0) {
394 // Brightness: C-new = C-old*(1-amount) + amount
395 float scale = 1f - brightnessF;
396 float[] mat = sTempBrightnessMatrix.getArray();
397 mat[0] = scale;
398 mat[6] = scale;
399 mat[12] = scale;
400 mat[4] = brightnessI;
401 mat[9] = brightnessI;
402 mat[14] = brightnessI;
403 sTempFilterMatrix.preConcat(sTempBrightnessMatrix);
404 }
405 filter = new ColorMatrixColorFilter(sTempFilterMatrix);
406 }
407 sCachedFilter.append(key, filter);
Sunny Goyal508da152014-08-14 10:53:27 -0700408 }
409 mPaint.setColorFilter(filter);
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700410 } else {
411 mPaint.setColorFilter(null);
412 }
Winsonc0880492015-08-21 11:16:27 -0700413 invalidateSelf();
Sunny Goyalc5c60ad2014-07-14 12:02:01 -0700414 }
Sunny Goyal95abbb32014-08-04 10:53:22 -0700415
Winsonc0880492015-08-21 11:16:27 -0700416 private AnimatorSet cancelAnimator(AnimatorSet animator) {
417 if (animator != null) {
Winsonc0880492015-08-21 11:16:27 -0700418 animator.cancel();
419 }
420 return null;
Sunny Goyal95abbb32014-08-04 10:53:22 -0700421 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800422}