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.recents.views; 18 19 import android.content.Context; 20 import android.content.res.Configuration; 21 import android.content.res.Resources; 22 import android.graphics.Bitmap; 23 import android.graphics.BitmapShader; 24 import android.graphics.Canvas; 25 import android.graphics.Color; 26 import android.graphics.ColorMatrix; 27 import android.graphics.ColorMatrixColorFilter; 28 import android.graphics.LightingColorFilter; 29 import android.graphics.Matrix; 30 import android.graphics.Paint; 31 import android.graphics.Rect; 32 import android.graphics.Shader; 33 import android.util.AttributeSet; 34 import android.view.View; 35 import android.view.ViewDebug; 36 37 import com.android.systemui.R; 38 import com.android.systemui.recents.events.EventBus; 39 import com.android.systemui.recents.events.ui.TaskSnapshotChangedEvent; 40 import com.android.systemui.shared.recents.utilities.Utilities; 41 import com.android.systemui.shared.recents.model.Task; 42 import com.android.systemui.shared.recents.model.ThumbnailData; 43 import java.io.PrintWriter; 44 45 46 /** 47 * The task thumbnail view. It implements an image view that allows for animating the dim and 48 * alpha of the thumbnail image. 49 */ 50 public class TaskViewThumbnail extends View { 51 52 private static final ColorMatrix TMP_FILTER_COLOR_MATRIX = new ColorMatrix(); 53 private static final ColorMatrix TMP_BRIGHTNESS_COLOR_MATRIX = new ColorMatrix(); 54 55 private Task mTask; 56 57 private int mDisplayOrientation = Configuration.ORIENTATION_UNDEFINED; 58 private Rect mDisplayRect = new Rect(); 59 60 // Drawing 61 @ViewDebug.ExportedProperty(category="recents") 62 protected Rect mTaskViewRect = new Rect(); 63 @ViewDebug.ExportedProperty(category="recents") 64 protected Rect mThumbnailRect = new Rect(); 65 @ViewDebug.ExportedProperty(category="recents") 66 protected float mThumbnailScale; 67 private float mFullscreenThumbnailScale = 1f; 68 /** The height, in pixels, of the task view's title bar. */ 69 private int mTitleBarHeight; 70 private boolean mSizeToFit = false; 71 private boolean mOverlayHeaderOnThumbnailActionBar = true; 72 private ThumbnailData mThumbnailData; 73 74 protected int mCornerRadius; 75 @ViewDebug.ExportedProperty(category="recents") 76 private float mDimAlpha; 77 private Matrix mMatrix = new Matrix(); 78 private Paint mDrawPaint = new Paint(); 79 protected Paint mLockedPaint = new Paint(); 80 protected Paint mBgFillPaint = new Paint(); 81 protected BitmapShader mBitmapShader; 82 protected boolean mUserLocked = false; 83 private LightingColorFilter mLightingColorFilter = new LightingColorFilter(0xffffffff, 0); 84 85 // Clip the top of the thumbnail against the opaque header bar that overlaps this view 86 private View mTaskBar; 87 88 // Visibility optimization, if the thumbnail height is less than the height of the header 89 // bar for the task view, then just mark this thumbnail view as invisible 90 @ViewDebug.ExportedProperty(category="recents") 91 private boolean mInvisible; 92 93 @ViewDebug.ExportedProperty(category="recents") 94 private boolean mDisabledInSafeMode; 95 TaskViewThumbnail(Context context)96 public TaskViewThumbnail(Context context) { 97 this(context, null); 98 } 99 TaskViewThumbnail(Context context, AttributeSet attrs)100 public TaskViewThumbnail(Context context, AttributeSet attrs) { 101 this(context, attrs, 0); 102 } 103 TaskViewThumbnail(Context context, AttributeSet attrs, int defStyleAttr)104 public TaskViewThumbnail(Context context, AttributeSet attrs, int defStyleAttr) { 105 this(context, attrs, defStyleAttr, 0); 106 } 107 TaskViewThumbnail(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)108 public TaskViewThumbnail(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { 109 super(context, attrs, defStyleAttr, defStyleRes); 110 mDrawPaint.setColorFilter(mLightingColorFilter); 111 mDrawPaint.setFilterBitmap(true); 112 mDrawPaint.setAntiAlias(true); 113 Resources res = getResources(); 114 mCornerRadius = res.getDimensionPixelSize(R.dimen.recents_task_view_rounded_corners_radius); 115 mBgFillPaint.setColor(Color.WHITE); 116 mLockedPaint.setColor(Color.WHITE); 117 mTitleBarHeight = res.getDimensionPixelSize(R.dimen.recents_grid_task_view_header_height); 118 } 119 120 /** 121 * Called when the task view frame changes, allowing us to move the contents of the header 122 * to match the frame changes. 123 */ onTaskViewSizeChanged(int width, int height)124 public void onTaskViewSizeChanged(int width, int height) { 125 // Return early if the bounds have not changed 126 if (mTaskViewRect.width() == width && mTaskViewRect.height() == height) { 127 return; 128 } 129 130 mTaskViewRect.set(0, 0, width, height); 131 setLeftTopRightBottom(0, 0, width, height); 132 updateThumbnailMatrix(); 133 } 134 135 @Override onDraw(Canvas canvas)136 protected void onDraw(Canvas canvas) { 137 if (mInvisible) { 138 return; 139 } 140 141 int viewWidth = mTaskViewRect.width(); 142 int viewHeight = mTaskViewRect.height(); 143 int thumbnailWidth = Math.min(viewWidth, 144 (int) (mThumbnailRect.width() * mThumbnailScale)); 145 int thumbnailHeight = Math.min(viewHeight, 146 (int) (mThumbnailRect.height() * mThumbnailScale)); 147 148 if (mUserLocked) { 149 canvas.drawRoundRect(0, 0, viewWidth, viewHeight, mCornerRadius, mCornerRadius, 150 mLockedPaint); 151 } else if (mBitmapShader != null && thumbnailWidth > 0 && thumbnailHeight > 0) { 152 int topOffset = 0; 153 if (mTaskBar != null && mOverlayHeaderOnThumbnailActionBar) { 154 topOffset = mTaskBar.getHeight() - mCornerRadius; 155 } 156 157 // Draw the background, there will be some small overdraw with the thumbnail 158 if (thumbnailWidth < viewWidth) { 159 // Portrait thumbnail on a landscape task view 160 canvas.drawRoundRect(Math.max(0, thumbnailWidth - mCornerRadius), topOffset, 161 viewWidth, viewHeight, 162 mCornerRadius, mCornerRadius, mBgFillPaint); 163 } 164 if (thumbnailHeight < viewHeight) { 165 // Landscape thumbnail on a portrait task view 166 canvas.drawRoundRect(0, Math.max(topOffset, thumbnailHeight - mCornerRadius), 167 viewWidth, viewHeight, 168 mCornerRadius, mCornerRadius, mBgFillPaint); 169 } 170 171 // Draw the thumbnail 172 canvas.drawRoundRect(0, topOffset, thumbnailWidth, thumbnailHeight, 173 mCornerRadius, mCornerRadius, mDrawPaint); 174 } else { 175 canvas.drawRoundRect(0, 0, viewWidth, viewHeight, mCornerRadius, mCornerRadius, 176 mBgFillPaint); 177 } 178 } 179 180 /** Sets the thumbnail to a given bitmap. */ setThumbnail(ThumbnailData thumbnailData)181 void setThumbnail(ThumbnailData thumbnailData) { 182 if (thumbnailData != null && thumbnailData.thumbnail != null) { 183 Bitmap bm = thumbnailData.thumbnail; 184 bm.prepareToDraw(); 185 mFullscreenThumbnailScale = thumbnailData.scale; 186 mBitmapShader = new BitmapShader(bm, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP); 187 mDrawPaint.setShader(mBitmapShader); 188 mThumbnailRect.set(0, 0, 189 bm.getWidth() - thumbnailData.insets.left - thumbnailData.insets.right, 190 bm.getHeight() - thumbnailData.insets.top - thumbnailData.insets.bottom); 191 mThumbnailData = thumbnailData; 192 updateThumbnailMatrix(); 193 updateThumbnailPaintFilter(); 194 } else { 195 mBitmapShader = null; 196 mDrawPaint.setShader(null); 197 mThumbnailRect.setEmpty(); 198 mThumbnailData = null; 199 } 200 } 201 202 /** Updates the paint to draw the thumbnail. */ updateThumbnailPaintFilter()203 void updateThumbnailPaintFilter() { 204 if (mInvisible) { 205 return; 206 } 207 int mul = (int) ((1.0f - mDimAlpha) * 255); 208 if (mBitmapShader != null) { 209 if (mDisabledInSafeMode) { 210 // Brightness: C-new = C-old*(1-amount) + amount 211 TMP_FILTER_COLOR_MATRIX.setSaturation(0); 212 float scale = 1f - mDimAlpha; 213 float[] mat = TMP_BRIGHTNESS_COLOR_MATRIX.getArray(); 214 mat[0] = scale; 215 mat[6] = scale; 216 mat[12] = scale; 217 mat[4] = mDimAlpha * 255f; 218 mat[9] = mDimAlpha * 255f; 219 mat[14] = mDimAlpha * 255f; 220 TMP_FILTER_COLOR_MATRIX.preConcat(TMP_BRIGHTNESS_COLOR_MATRIX); 221 ColorMatrixColorFilter filter = new ColorMatrixColorFilter(TMP_FILTER_COLOR_MATRIX); 222 mDrawPaint.setColorFilter(filter); 223 mBgFillPaint.setColorFilter(filter); 224 mLockedPaint.setColorFilter(filter); 225 } else { 226 mLightingColorFilter.setColorMultiply(Color.argb(255, mul, mul, mul)); 227 mDrawPaint.setColorFilter(mLightingColorFilter); 228 mDrawPaint.setColor(0xFFffffff); 229 mBgFillPaint.setColorFilter(mLightingColorFilter); 230 mLockedPaint.setColorFilter(mLightingColorFilter); 231 } 232 } else { 233 int grey = mul; 234 mDrawPaint.setColorFilter(null); 235 mDrawPaint.setColor(Color.argb(255, grey, grey, grey)); 236 } 237 if (!mInvisible) { 238 invalidate(); 239 } 240 } 241 242 /** 243 * Updates the scale of the bitmap relative to this view. 244 */ updateThumbnailMatrix()245 public void updateThumbnailMatrix() { 246 mThumbnailScale = 1f; 247 if (mBitmapShader != null && mThumbnailData != null) { 248 if (mTaskViewRect.isEmpty()) { 249 // If we haven't measured , skip the thumbnail drawing and only draw the background 250 // color 251 mThumbnailScale = 0f; 252 } else if (mSizeToFit) { 253 // Make sure we fill the entire space regardless of the orientation. 254 float viewAspectRatio = (float) mTaskViewRect.width() / 255 (float) (mTaskViewRect.height() - mTitleBarHeight); 256 float thumbnailAspectRatio = 257 (float) mThumbnailRect.width() / (float) mThumbnailRect.height(); 258 if (viewAspectRatio > thumbnailAspectRatio) { 259 mThumbnailScale = 260 (float) mTaskViewRect.width() / (float) mThumbnailRect.width(); 261 } else { 262 mThumbnailScale = (float) (mTaskViewRect.height() - mTitleBarHeight) 263 / (float) mThumbnailRect.height(); 264 } 265 } else { 266 float invThumbnailScale = 1f / mFullscreenThumbnailScale; 267 if (mDisplayOrientation == Configuration.ORIENTATION_PORTRAIT) { 268 if (mThumbnailData.orientation == Configuration.ORIENTATION_PORTRAIT) { 269 // If we are in the same orientation as the screenshot, just scale it to the 270 // width of the task view 271 mThumbnailScale = (float) mTaskViewRect.width() / mThumbnailRect.width(); 272 } else { 273 // Scale the landscape thumbnail up to app size, then scale that to the task 274 // view size to match other portrait screenshots 275 mThumbnailScale = invThumbnailScale * 276 ((float) mTaskViewRect.width() / mDisplayRect.width()); 277 } 278 } else { 279 // Otherwise, scale the screenshot to fit 1:1 in the current orientation 280 mThumbnailScale = invThumbnailScale; 281 } 282 } 283 mMatrix.setTranslate(-mThumbnailData.insets.left * mFullscreenThumbnailScale, 284 -mThumbnailData.insets.top * mFullscreenThumbnailScale); 285 mMatrix.postScale(mThumbnailScale, mThumbnailScale); 286 mBitmapShader.setLocalMatrix(mMatrix); 287 } 288 if (!mInvisible) { 289 invalidate(); 290 } 291 } 292 293 /** Sets whether the thumbnail should be resized to fit the task view in all orientations. */ setSizeToFit(boolean flag)294 public void setSizeToFit(boolean flag) { 295 mSizeToFit = flag; 296 } 297 298 /** 299 * Sets whether the header should overlap (and hide) the action bar in the thumbnail, or 300 * be stacked just above it. 301 */ setOverlayHeaderOnThumbnailActionBar(boolean flag)302 public void setOverlayHeaderOnThumbnailActionBar(boolean flag) { 303 mOverlayHeaderOnThumbnailActionBar = flag; 304 } 305 306 /** Updates the clip rect based on the given task bar. */ updateClipToTaskBar(View taskBar)307 void updateClipToTaskBar(View taskBar) { 308 mTaskBar = taskBar; 309 invalidate(); 310 } 311 312 /** Updates the visibility of the the thumbnail. */ updateThumbnailVisibility(int clipBottom)313 void updateThumbnailVisibility(int clipBottom) { 314 boolean invisible = mTaskBar != null && (getHeight() - clipBottom) <= mTaskBar.getHeight(); 315 if (invisible != mInvisible) { 316 mInvisible = invisible; 317 if (!mInvisible) { 318 updateThumbnailPaintFilter(); 319 } 320 } 321 } 322 323 /** 324 * Sets the dim alpha, only used when we are not using hardware layers. 325 * (see RecentsConfiguration.useHardwareLayers) 326 */ setDimAlpha(float dimAlpha)327 public void setDimAlpha(float dimAlpha) { 328 mDimAlpha = dimAlpha; 329 updateThumbnailPaintFilter(); 330 } 331 332 /** 333 * Returns the {@link Paint} used to draw a task screenshot, or {@link #mLockedPaint} if the 334 * thumbnail shouldn't be drawn because it belongs to a locked user. 335 */ getDrawPaint()336 protected Paint getDrawPaint() { 337 if (mUserLocked) { 338 return mLockedPaint; 339 } 340 return mDrawPaint; 341 } 342 343 /** 344 * Binds the thumbnail view to the task. 345 */ bindToTask(Task t, boolean disabledInSafeMode, int displayOrientation, Rect displayRect)346 void bindToTask(Task t, boolean disabledInSafeMode, int displayOrientation, Rect displayRect) { 347 mTask = t; 348 mDisabledInSafeMode = disabledInSafeMode; 349 mDisplayOrientation = displayOrientation; 350 mDisplayRect.set(displayRect); 351 if (t.colorBackground != 0) { 352 mBgFillPaint.setColor(t.colorBackground); 353 } 354 if (t.colorPrimary != 0) { 355 mLockedPaint.setColor(t.colorPrimary); 356 } 357 mUserLocked = t.isLocked; 358 EventBus.getDefault().register(this); 359 } 360 361 /** 362 * Called when the bound task's data has loaded and this view should update to reflect the 363 * changes. 364 */ onTaskDataLoaded(ThumbnailData thumbnailData)365 void onTaskDataLoaded(ThumbnailData thumbnailData) { 366 setThumbnail(thumbnailData); 367 } 368 369 /** Unbinds the thumbnail view from the task */ unbindFromTask()370 void unbindFromTask() { 371 mTask = null; 372 setThumbnail(null); 373 EventBus.getDefault().unregister(this); 374 } 375 onBusEvent(TaskSnapshotChangedEvent event)376 public final void onBusEvent(TaskSnapshotChangedEvent event) { 377 if (mTask == null || event.taskId != mTask.key.id || event.thumbnailData == null 378 || event.thumbnailData.thumbnail == null) { 379 return; 380 } 381 setThumbnail(event.thumbnailData); 382 } 383 dump(String prefix, PrintWriter writer)384 public void dump(String prefix, PrintWriter writer) { 385 writer.print(prefix); writer.print("TaskViewThumbnail"); 386 writer.print(" mTaskViewRect="); writer.print(Utilities.dumpRect(mTaskViewRect)); 387 writer.print(" mThumbnailRect="); writer.print(Utilities.dumpRect(mThumbnailRect)); 388 writer.print(" mThumbnailScale="); writer.print(mThumbnailScale); 389 writer.print(" mDimAlpha="); writer.print(mDimAlpha); 390 writer.println(); 391 } 392 } 393