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.systemui.statusbar; 18 19 import static java.lang.Float.isNaN; 20 21 import android.annotation.NonNull; 22 import android.content.Context; 23 import android.graphics.Canvas; 24 import android.graphics.Color; 25 import android.graphics.PorterDuff; 26 import android.graphics.PorterDuff.Mode; 27 import android.graphics.PorterDuffColorFilter; 28 import android.graphics.drawable.Drawable; 29 import android.util.AttributeSet; 30 import android.view.View; 31 32 import androidx.core.graphics.ColorUtils; 33 34 import com.android.internal.annotations.VisibleForTesting; 35 import com.android.internal.colorextraction.ColorExtractor; 36 import com.android.internal.colorextraction.drawable.ScrimDrawable; 37 38 /** 39 * A view which can draw a scrim 40 */ 41 public class ScrimView extends View { 42 private final ColorExtractor.GradientColors mColors; 43 private float mViewAlpha = 1.0f; 44 private Drawable mDrawable; 45 private PorterDuffColorFilter mColorFilter; 46 private int mTintColor; 47 private Runnable mChangeRunnable; 48 ScrimView(Context context)49 public ScrimView(Context context) { 50 this(context, null); 51 } 52 ScrimView(Context context, AttributeSet attrs)53 public ScrimView(Context context, AttributeSet attrs) { 54 this(context, attrs, 0); 55 } 56 ScrimView(Context context, AttributeSet attrs, int defStyleAttr)57 public ScrimView(Context context, AttributeSet attrs, int defStyleAttr) { 58 this(context, attrs, defStyleAttr, 0); 59 } 60 ScrimView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)61 public ScrimView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { 62 super(context, attrs, defStyleAttr, defStyleRes); 63 64 mDrawable = new ScrimDrawable(); 65 mDrawable.setCallback(this); 66 mColors = new ColorExtractor.GradientColors(); 67 updateColorWithTint(false); 68 } 69 70 @Override onDraw(Canvas canvas)71 protected void onDraw(Canvas canvas) { 72 if (mDrawable.getAlpha() > 0) { 73 mDrawable.draw(canvas); 74 } 75 } 76 setDrawable(Drawable drawable)77 public void setDrawable(Drawable drawable) { 78 mDrawable = drawable; 79 mDrawable.setCallback(this); 80 mDrawable.setBounds(getLeft(), getTop(), getRight(), getBottom()); 81 mDrawable.setAlpha((int) (255 * mViewAlpha)); 82 invalidate(); 83 } 84 85 @Override invalidateDrawable(@onNull Drawable drawable)86 public void invalidateDrawable(@NonNull Drawable drawable) { 87 super.invalidateDrawable(drawable); 88 if (drawable == mDrawable) { 89 invalidate(); 90 } 91 } 92 93 @Override onLayout(boolean changed, int left, int top, int right, int bottom)94 protected void onLayout(boolean changed, int left, int top, int right, int bottom) { 95 super.onLayout(changed, left, top, right, bottom); 96 if (changed) { 97 mDrawable.setBounds(left, top, right, bottom); 98 invalidate(); 99 } 100 } 101 setColors(@onNull ColorExtractor.GradientColors colors)102 public void setColors(@NonNull ColorExtractor.GradientColors colors) { 103 setColors(colors, false); 104 } 105 setColors(@onNull ColorExtractor.GradientColors colors, boolean animated)106 public void setColors(@NonNull ColorExtractor.GradientColors colors, boolean animated) { 107 if (colors == null) { 108 throw new IllegalArgumentException("Colors cannot be null"); 109 } 110 if (mColors.equals(colors)) { 111 return; 112 } 113 mColors.set(colors); 114 updateColorWithTint(animated); 115 } 116 117 @VisibleForTesting getDrawable()118 Drawable getDrawable() { 119 return mDrawable; 120 } 121 getColors()122 public ColorExtractor.GradientColors getColors() { 123 return mColors; 124 } 125 setTint(int color)126 public void setTint(int color) { 127 setTint(color, false); 128 } 129 setTint(int color, boolean animated)130 public void setTint(int color, boolean animated) { 131 if (mTintColor == color) { 132 return; 133 } 134 mTintColor = color; 135 updateColorWithTint(animated); 136 } 137 updateColorWithTint(boolean animated)138 private void updateColorWithTint(boolean animated) { 139 if (mDrawable instanceof ScrimDrawable) { 140 // Optimization to blend colors and avoid a color filter 141 ScrimDrawable drawable = (ScrimDrawable) mDrawable; 142 float tintAmount = Color.alpha(mTintColor) / 255f; 143 int mainTinted = ColorUtils.blendARGB(mColors.getMainColor(), mTintColor, 144 tintAmount); 145 drawable.setColor(mainTinted, animated); 146 } else { 147 boolean hasAlpha = Color.alpha(mTintColor) != 0; 148 if (hasAlpha) { 149 PorterDuff.Mode targetMode = mColorFilter == null ? Mode.SRC_OVER : 150 mColorFilter.getMode(); 151 if (mColorFilter == null || mColorFilter.getColor() != mTintColor) { 152 mColorFilter = new PorterDuffColorFilter(mTintColor, targetMode); 153 } 154 } else { 155 mColorFilter = null; 156 } 157 158 mDrawable.setColorFilter(mColorFilter); 159 mDrawable.invalidateSelf(); 160 } 161 162 if (mChangeRunnable != null) { 163 mChangeRunnable.run(); 164 } 165 } 166 getTint()167 public int getTint() { 168 return mTintColor; 169 } 170 171 @Override hasOverlappingRendering()172 public boolean hasOverlappingRendering() { 173 return false; 174 } 175 176 /** 177 * It might look counterintuitive to have another method to set the alpha instead of 178 * only using {@link #setAlpha(float)}. In this case we're in a hardware layer 179 * optimizing blend modes, so it makes sense. 180 * 181 * @param alpha Gradient alpha from 0 to 1. 182 */ setViewAlpha(float alpha)183 public void setViewAlpha(float alpha) { 184 if (isNaN(alpha)) { 185 throw new IllegalArgumentException("alpha cannot be NaN: " + alpha); 186 } 187 if (alpha != mViewAlpha) { 188 mViewAlpha = alpha; 189 190 mDrawable.setAlpha((int) (255 * alpha)); 191 if (mChangeRunnable != null) { 192 mChangeRunnable.run(); 193 } 194 } 195 } 196 getViewAlpha()197 public float getViewAlpha() { 198 return mViewAlpha; 199 } 200 setChangeRunnable(Runnable changeRunnable)201 public void setChangeRunnable(Runnable changeRunnable) { 202 mChangeRunnable = changeRunnable; 203 } 204 205 @Override canReceivePointerEvents()206 protected boolean canReceivePointerEvents() { 207 return false; 208 } 209 } 210