2009-03-03 19:32:27 -08:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2008 The Android Open Source Project
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
2013-06-05 22:57:57 -04:00
|
|
|
package com.android.launcher3;
|
2009-03-03 19:32:27 -08:00
|
|
|
|
2014-08-14 10:53:27 -07:00
|
|
|
import android.animation.ObjectAnimator;
|
|
|
|
|
import android.animation.TimeInterpolator;
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.graphics.Bitmap;
|
|
|
|
|
import android.graphics.Canvas;
|
2014-08-14 10:53:27 -07:00
|
|
|
import android.graphics.Color;
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.graphics.ColorFilter;
|
2014-07-14 12:02:01 -07:00
|
|
|
import android.graphics.ColorMatrix;
|
|
|
|
|
import android.graphics.ColorMatrixColorFilter;
|
2010-11-09 17:19:49 -08:00
|
|
|
import android.graphics.Paint;
|
|
|
|
|
import android.graphics.PixelFormat;
|
2014-08-14 10:53:27 -07:00
|
|
|
import android.graphics.PorterDuff;
|
|
|
|
|
import android.graphics.PorterDuffColorFilter;
|
2010-11-09 17:19:49 -08:00
|
|
|
import android.graphics.drawable.Drawable;
|
2017-02-02 12:57:18 -08:00
|
|
|
import android.util.Property;
|
2014-08-14 10:53:27 -07:00
|
|
|
import android.util.SparseArray;
|
2009-03-03 19:32:27 -08:00
|
|
|
|
2017-12-19 16:49:24 -08:00
|
|
|
import com.android.launcher3.graphics.BitmapInfo;
|
2017-01-11 09:53:12 -08:00
|
|
|
|
2015-04-08 19:01:34 -07:00
|
|
|
public class FastBitmapDrawable extends Drawable {
|
2017-02-16 13:33:15 -08:00
|
|
|
|
|
|
|
|
private static final float PRESSED_BRIGHTNESS = 100f / 255f;
|
2016-11-08 10:40:34 -08:00
|
|
|
private static final float DISABLED_DESATURATION = 1f;
|
|
|
|
|
private static final float DISABLED_BRIGHTNESS = 0.5f;
|
2014-07-14 12:02:01 -07:00
|
|
|
|
2017-12-19 16:49:24 -08:00
|
|
|
public static final TimeInterpolator CLICK_FEEDBACK_INTERPOLATOR = (input) ->
|
|
|
|
|
(input < 0.05f) ? (input / 0.05f) : ((input < 0.3f) ? 1 : (1 - input) / 0.7f);
|
2014-08-14 10:53:27 -07:00
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
public static final int CLICK_FEEDBACK_DURATION = 2000;
|
2014-08-14 10:53:27 -07:00
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
// Since we don't need 256^2 values for combinations of both the brightness and saturation, we
|
|
|
|
|
// reduce the value space to a smaller value V, which reduces the number of cached
|
|
|
|
|
// ColorMatrixColorFilters that we need to keep to V^2
|
|
|
|
|
private static final int REDUCED_FILTER_VALUE_SPACE = 48;
|
2014-08-04 10:53:22 -07:00
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
// A cache of ColorFilters for optimizing brightness and saturation animations
|
|
|
|
|
private static final SparseArray<ColorFilter> sCachedFilter = new SparseArray<>();
|
2014-08-14 10:53:27 -07:00
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
// Temporary matrices used for calculation
|
|
|
|
|
private static final ColorMatrix sTempBrightnessMatrix = new ColorMatrix();
|
|
|
|
|
private static final ColorMatrix sTempFilterMatrix = new ColorMatrix();
|
2014-07-14 12:02:01 -07:00
|
|
|
|
2016-11-12 09:58:29 -08:00
|
|
|
protected final Paint mPaint = new Paint(Paint.FILTER_BITMAP_FLAG | Paint.ANTI_ALIAS_FLAG);
|
2014-08-14 10:53:27 -07:00
|
|
|
private final Bitmap mBitmap;
|
2017-12-19 16:49:24 -08:00
|
|
|
protected final int mIconColor;
|
2017-02-16 13:33:15 -08:00
|
|
|
|
|
|
|
|
private boolean mIsPressed;
|
2016-11-08 10:40:34 -08:00
|
|
|
private boolean mIsDisabled;
|
2014-07-14 12:02:01 -07:00
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
private static final Property<FastBitmapDrawable, Float> BRIGHTNESS
|
|
|
|
|
= new Property<FastBitmapDrawable, Float>(Float.TYPE, "brightness") {
|
|
|
|
|
@Override
|
|
|
|
|
public Float get(FastBitmapDrawable fastBitmapDrawable) {
|
|
|
|
|
return fastBitmapDrawable.getBrightness();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void set(FastBitmapDrawable fastBitmapDrawable, Float value) {
|
|
|
|
|
fastBitmapDrawable.setBrightness(value);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
// The saturation and brightness are values that are mapped to REDUCED_FILTER_VALUE_SPACE and
|
|
|
|
|
// as a result, can be used to compose the key for the cached ColorMatrixColorFilters
|
|
|
|
|
private int mDesaturation = 0;
|
2014-07-14 12:02:01 -07:00
|
|
|
private int mBrightness = 0;
|
2015-08-21 11:16:27 -07:00
|
|
|
private int mAlpha = 255;
|
|
|
|
|
private int mPrevUpdateKey = Integer.MAX_VALUE;
|
2009-03-03 19:32:27 -08:00
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
// Animators for the fast bitmap drawable's brightness
|
|
|
|
|
private ObjectAnimator mBrightnessAnimator;
|
2014-08-14 10:53:27 -07:00
|
|
|
|
2015-04-08 19:01:34 -07:00
|
|
|
public FastBitmapDrawable(Bitmap b) {
|
2017-12-19 16:49:24 -08:00
|
|
|
this(b, Color.TRANSPARENT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public FastBitmapDrawable(BitmapInfo info) {
|
|
|
|
|
this(info.icon, info.color);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public FastBitmapDrawable(ItemInfoWithIcon info) {
|
|
|
|
|
this(info.iconBitmap, info.iconColor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected FastBitmapDrawable(Bitmap b, int iconColor) {
|
2009-03-03 19:32:27 -08:00
|
|
|
mBitmap = b;
|
2017-12-19 16:49:24 -08:00
|
|
|
mIconColor = iconColor;
|
2017-02-02 16:37:21 -08:00
|
|
|
setFilterBitmap(true);
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void draw(Canvas canvas) {
|
2015-08-21 11:16:27 -07:00
|
|
|
canvas.drawBitmap(mBitmap, null, getBounds(), mPaint);
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
2011-05-26 19:08:29 -07:00
|
|
|
@Override
|
|
|
|
|
public void setColorFilter(ColorFilter cf) {
|
2014-07-14 12:02:01 -07:00
|
|
|
// No op
|
2011-05-26 19:08:29 -07:00
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
@Override
|
|
|
|
|
public int getOpacity() {
|
|
|
|
|
return PixelFormat.TRANSLUCENT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void setAlpha(int alpha) {
|
2010-12-01 15:47:31 -08:00
|
|
|
mAlpha = alpha;
|
2010-08-19 14:51:28 -07:00
|
|
|
mPaint.setAlpha(alpha);
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
2014-07-14 12:02:01 -07:00
|
|
|
@Override
|
2011-06-17 13:26:23 -07:00
|
|
|
public void setFilterBitmap(boolean filterBitmap) {
|
|
|
|
|
mPaint.setFilterBitmap(filterBitmap);
|
2013-10-25 15:24:24 -07:00
|
|
|
mPaint.setAntiAlias(filterBitmap);
|
2011-06-17 13:26:23 -07:00
|
|
|
}
|
|
|
|
|
|
2010-12-01 15:47:31 -08:00
|
|
|
public int getAlpha() {
|
|
|
|
|
return mAlpha;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
@Override
|
|
|
|
|
public int getIntrinsicWidth() {
|
2014-09-05 07:04:59 -07:00
|
|
|
return mBitmap.getWidth();
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getIntrinsicHeight() {
|
2014-09-05 07:04:59 -07:00
|
|
|
return mBitmap.getHeight();
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getMinimumWidth() {
|
2013-11-13 15:47:05 -08:00
|
|
|
return getBounds().width();
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public int getMinimumHeight() {
|
2013-11-13 15:47:05 -08:00
|
|
|
return getBounds().height();
|
2010-02-08 13:44:00 -08:00
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
public Bitmap getBitmap() {
|
|
|
|
|
return mBitmap;
|
|
|
|
|
}
|
2014-07-14 12:02:01 -07:00
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
@Override
|
|
|
|
|
public boolean isStateful() {
|
|
|
|
|
return true;
|
2014-07-14 12:02:01 -07:00
|
|
|
}
|
|
|
|
|
|
2017-06-21 17:28:23 -07:00
|
|
|
@Override
|
|
|
|
|
public ColorFilter getColorFilter() {
|
|
|
|
|
return mPaint.getColorFilter();
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
@Override
|
|
|
|
|
protected boolean onStateChange(int[] state) {
|
|
|
|
|
boolean isPressed = false;
|
|
|
|
|
for (int s : state) {
|
|
|
|
|
if (s == android.R.attr.state_pressed) {
|
|
|
|
|
isPressed = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (mIsPressed != isPressed) {
|
|
|
|
|
mIsPressed = isPressed;
|
2015-08-21 11:16:27 -07:00
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
if (mBrightnessAnimator != null) {
|
|
|
|
|
mBrightnessAnimator.cancel();
|
|
|
|
|
}
|
2015-08-21 11:16:27 -07:00
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
if (mIsPressed) {
|
|
|
|
|
// Animate when going to pressed state
|
|
|
|
|
mBrightnessAnimator = ObjectAnimator.ofFloat(
|
|
|
|
|
this, BRIGHTNESS, getExpectedBrightness());
|
|
|
|
|
mBrightnessAnimator.setDuration(CLICK_FEEDBACK_DURATION);
|
|
|
|
|
mBrightnessAnimator.setInterpolator(CLICK_FEEDBACK_INTERPOLATOR);
|
|
|
|
|
mBrightnessAnimator.start();
|
|
|
|
|
} else {
|
|
|
|
|
setBrightness(getExpectedBrightness());
|
|
|
|
|
}
|
2015-08-21 11:16:27 -07:00
|
|
|
return true;
|
2014-08-14 10:53:27 -07:00
|
|
|
}
|
2015-08-21 11:16:27 -07:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-08 10:40:34 -08:00
|
|
|
private void invalidateDesaturationAndBrightness() {
|
2017-02-16 13:33:15 -08:00
|
|
|
setDesaturation(mIsDisabled ? DISABLED_DESATURATION : 0);
|
|
|
|
|
setBrightness(getExpectedBrightness());
|
2016-11-08 10:40:34 -08:00
|
|
|
}
|
|
|
|
|
|
2017-02-16 13:33:15 -08:00
|
|
|
private float getExpectedBrightness() {
|
|
|
|
|
return mIsDisabled ? DISABLED_BRIGHTNESS :
|
|
|
|
|
(mIsPressed ? PRESSED_BRIGHTNESS : 0);
|
2014-08-14 10:53:27 -07:00
|
|
|
}
|
|
|
|
|
|
2016-11-08 10:40:34 -08:00
|
|
|
public void setIsDisabled(boolean isDisabled) {
|
|
|
|
|
if (mIsDisabled != isDisabled) {
|
|
|
|
|
mIsDisabled = isDisabled;
|
|
|
|
|
invalidateDesaturationAndBrightness();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
/**
|
|
|
|
|
* Sets the saturation of this icon, 0 [full color] -> 1 [desaturated]
|
|
|
|
|
*/
|
2016-11-12 09:58:29 -08:00
|
|
|
private void setDesaturation(float desaturation) {
|
2015-08-21 11:16:27 -07:00
|
|
|
int newDesaturation = (int) Math.floor(desaturation * REDUCED_FILTER_VALUE_SPACE);
|
|
|
|
|
if (mDesaturation != newDesaturation) {
|
|
|
|
|
mDesaturation = newDesaturation;
|
|
|
|
|
updateFilter();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public float getDesaturation() {
|
|
|
|
|
return (float) mDesaturation / REDUCED_FILTER_VALUE_SPACE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Sets the brightness of this icon, 0 [no add. brightness] -> 1 [2bright2furious]
|
|
|
|
|
*/
|
2016-11-12 09:58:29 -08:00
|
|
|
private void setBrightness(float brightness) {
|
2015-08-21 11:16:27 -07:00
|
|
|
int newBrightness = (int) Math.floor(brightness * REDUCED_FILTER_VALUE_SPACE);
|
|
|
|
|
if (mBrightness != newBrightness) {
|
|
|
|
|
mBrightness = newBrightness;
|
2014-08-04 10:53:22 -07:00
|
|
|
updateFilter();
|
|
|
|
|
}
|
2014-07-14 12:02:01 -07:00
|
|
|
}
|
|
|
|
|
|
2016-11-12 09:58:29 -08:00
|
|
|
private float getBrightness() {
|
2015-08-21 11:16:27 -07:00
|
|
|
return (float) mBrightness / REDUCED_FILTER_VALUE_SPACE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Updates the paint to reflect the current brightness and saturation.
|
|
|
|
|
*/
|
2014-07-14 12:02:01 -07:00
|
|
|
private void updateFilter() {
|
2015-08-21 11:16:27 -07:00
|
|
|
boolean usePorterDuffFilter = false;
|
|
|
|
|
int key = -1;
|
|
|
|
|
if (mDesaturation > 0) {
|
|
|
|
|
key = (mDesaturation << 16) | mBrightness;
|
|
|
|
|
} else if (mBrightness > 0) {
|
|
|
|
|
// Compose a key with a fully saturated icon if we are just animating brightness
|
|
|
|
|
key = (1 << 16) | mBrightness;
|
|
|
|
|
|
|
|
|
|
// We found that in L, ColorFilters cause drawing artifacts with shadows baked into
|
|
|
|
|
// icons, so just use a PorterDuff filter when we aren't animating saturation
|
|
|
|
|
usePorterDuffFilter = true;
|
|
|
|
|
}
|
2014-08-04 10:53:22 -07:00
|
|
|
|
2015-08-21 11:16:27 -07:00
|
|
|
// Debounce multiple updates on the same frame
|
|
|
|
|
if (key == mPrevUpdateKey) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mPrevUpdateKey = key;
|
|
|
|
|
|
|
|
|
|
if (key != -1) {
|
|
|
|
|
ColorFilter filter = sCachedFilter.get(key);
|
2014-08-14 10:53:27 -07:00
|
|
|
if (filter == null) {
|
2015-08-21 11:16:27 -07:00
|
|
|
float brightnessF = getBrightness();
|
|
|
|
|
int brightnessI = (int) (255 * brightnessF);
|
|
|
|
|
if (usePorterDuffFilter) {
|
|
|
|
|
filter = new PorterDuffColorFilter(Color.argb(brightnessI, 255, 255, 255),
|
|
|
|
|
PorterDuff.Mode.SRC_ATOP);
|
|
|
|
|
} else {
|
|
|
|
|
float saturationF = 1f - getDesaturation();
|
|
|
|
|
sTempFilterMatrix.setSaturation(saturationF);
|
|
|
|
|
if (mBrightness > 0) {
|
|
|
|
|
// Brightness: C-new = C-old*(1-amount) + amount
|
|
|
|
|
float scale = 1f - brightnessF;
|
|
|
|
|
float[] mat = sTempBrightnessMatrix.getArray();
|
|
|
|
|
mat[0] = scale;
|
|
|
|
|
mat[6] = scale;
|
|
|
|
|
mat[12] = scale;
|
|
|
|
|
mat[4] = brightnessI;
|
|
|
|
|
mat[9] = brightnessI;
|
|
|
|
|
mat[14] = brightnessI;
|
|
|
|
|
sTempFilterMatrix.preConcat(sTempBrightnessMatrix);
|
|
|
|
|
}
|
|
|
|
|
filter = new ColorMatrixColorFilter(sTempFilterMatrix);
|
|
|
|
|
}
|
|
|
|
|
sCachedFilter.append(key, filter);
|
2014-08-14 10:53:27 -07:00
|
|
|
}
|
|
|
|
|
mPaint.setColorFilter(filter);
|
2014-07-14 12:02:01 -07:00
|
|
|
} else {
|
|
|
|
|
mPaint.setColorFilter(null);
|
|
|
|
|
}
|
2015-08-21 11:16:27 -07:00
|
|
|
invalidateSelf();
|
2014-07-14 12:02:01 -07:00
|
|
|
}
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|