blob: a8db330b32b9d7a441a12fa41b8481330beb11e3 [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
Joe Onoratoa5902522009-07-30 13:37:37 -070017package com.android.launcher2;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080018
Michael Jurka38b4f7c2010-12-14 16:46:39 -080019import com.android.launcher.R;
20
The Android Open Source Project31dd5032009-03-03 19:32:27 -080021import android.content.Context;
Winson Chung656d11c2010-11-29 17:15:47 -080022import android.content.res.Resources;
Michael Jurka67b2f6c2010-11-17 12:33:46 -080023import android.graphics.Bitmap;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080024import android.graphics.Canvas;
Winson Chung656d11c2010-11-29 17:15:47 -080025import android.graphics.Color;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080026import android.graphics.Paint;
Michael Jurka38b4f7c2010-12-14 16:46:39 -080027import android.graphics.Rect;
Michael Jurka137142e2011-01-05 20:57:04 -080028import android.graphics.Region;
Michael Jurka38b4f7c2010-12-14 16:46:39 -080029import android.graphics.Region.Op;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080030import android.graphics.drawable.Drawable;
Winson Chung656d11c2010-11-29 17:15:47 -080031import android.util.AttributeSet;
Michael Jurka38b4f7c2010-12-14 16:46:39 -080032import android.view.MotionEvent;
Michael Jurka99b6a5b2011-01-07 15:37:17 -080033import android.view.View;
Michael Jurkabdb5c532011-02-01 15:05:06 -080034import android.widget.TextView;
Romain Guyedcce092010-03-04 13:03:17 -080035
The Android Open Source Project31dd5032009-03-03 19:32:27 -080036/**
37 * TextView that draws a bubble behind the text. We cannot use a LineBackgroundSpan
38 * because we want to make the bubble taller than the text and TextView's clip is
39 * too aggressive.
40 */
Michael Jurkabdb5c532011-02-01 15:05:06 -080041public class BubbleTextView extends TextView implements VisibilityChangedBroadcaster {
Winson Chung656d11c2010-11-29 17:15:47 -080042 static final float CORNER_RADIUS = 4.0f;
Winson Chung88127032010-12-13 12:11:33 -080043 static final float SHADOW_LARGE_RADIUS = 4.0f;
44 static final float SHADOW_SMALL_RADIUS = 1.75f;
45 static final float SHADOW_Y_OFFSET = 2.0f;
46 static final int SHADOW_LARGE_COLOUR = 0xCC000000;
47 static final int SHADOW_SMALL_COLOUR = 0xBB000000;
Winson Chung656d11c2010-11-29 17:15:47 -080048 static final float PADDING_H = 8.0f;
49 static final float PADDING_V = 3.0f;
50
The Android Open Source Project31dd5032009-03-03 19:32:27 -080051 private Paint mPaint;
Winson Chung656d11c2010-11-29 17:15:47 -080052 private float mBubbleColorAlpha;
Winson Chunge22a8e92010-11-12 13:40:58 -080053 private int mPrevAlpha = -1;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080054
Michael Jurka38b4f7c2010-12-14 16:46:39 -080055 private final HolographicOutlineHelper mOutlineHelper = new HolographicOutlineHelper();
56 private final Canvas mTempCanvas = new Canvas();
57 private final Rect mTempRect = new Rect();
58 private final Paint mTempPaint = new Paint();
59 private boolean mDidInvalidateForPressedState;
60 private Bitmap mPressedOrFocusedBackground;
61 private int mFocusedOutlineColor;
62 private int mFocusedGlowColor;
63 private int mPressedOutlineColor;
64 private int mPressedGlowColor;
65
The Android Open Source Project31dd5032009-03-03 19:32:27 -080066 private boolean mBackgroundSizeChanged;
67 private Drawable mBackground;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080068
Michael Jurkaddd62e92011-02-16 17:49:14 -080069 private boolean mStayPressed;
70
Michael Jurka99b6a5b2011-01-07 15:37:17 -080071 private VisibilityChangedListener mOnVisibilityChangedListener;
72
The Android Open Source Project31dd5032009-03-03 19:32:27 -080073 public BubbleTextView(Context context) {
74 super(context);
75 init();
76 }
77
78 public BubbleTextView(Context context, AttributeSet attrs) {
79 super(context, attrs);
80 init();
81 }
82
83 public BubbleTextView(Context context, AttributeSet attrs, int defStyle) {
84 super(context, attrs, defStyle);
85 init();
86 }
87
88 private void init() {
The Android Open Source Project31dd5032009-03-03 19:32:27 -080089 mBackground = getBackground();
Winson Chung656d11c2010-11-29 17:15:47 -080090 setFocusable(true);
The Android Open Source Project31dd5032009-03-03 19:32:27 -080091 setBackgroundDrawable(null);
The Android Open Source Project31dd5032009-03-03 19:32:27 -080092
Winson Chung656d11c2010-11-29 17:15:47 -080093 final Resources res = getContext().getResources();
94 int bubbleColor = res.getColor(R.color.bubble_dark_background);
The Android Open Source Project31dd5032009-03-03 19:32:27 -080095 mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
Winson Chung656d11c2010-11-29 17:15:47 -080096 mPaint.setColor(bubbleColor);
97 mBubbleColorAlpha = Color.alpha(bubbleColor) / 255.0f;
Winson Chung59e1f9a2010-12-21 11:31:54 -080098 mFocusedOutlineColor = res.getColor(R.color.workspace_item_focused_outline_color);
99 mFocusedGlowColor = res.getColor(R.color.workspace_item_focused_glow_color);
100 mPressedOutlineColor = res.getColor(R.color.workspace_item_pressed_outline_color);
101 mPressedGlowColor = res.getColor(R.color.workspace_item_pressed_glow_color);
Michael Jurkae7e3f6c2011-02-01 21:08:29 -0800102
103 setShadowLayer(SHADOW_LARGE_RADIUS, 0.0f, SHADOW_Y_OFFSET, SHADOW_LARGE_COLOUR);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800104 }
105
Michael Jurka67b2f6c2010-11-17 12:33:46 -0800106 public void applyFromShortcutInfo(ShortcutInfo info, IconCache iconCache) {
107 Bitmap b = info.getIcon(iconCache);
108
109 setCompoundDrawablesWithIntrinsicBounds(null,
110 new FastBitmapDrawable(b),
111 null, null);
112 setText(info.title);
Michael Jurka67b2f6c2010-11-17 12:33:46 -0800113 setTag(info);
Michael Jurka67b2f6c2010-11-17 12:33:46 -0800114 }
115
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800116 @Override
117 protected boolean setFrame(int left, int top, int right, int bottom) {
118 if (mLeft != left || mRight != right || mTop != top || mBottom != bottom) {
119 mBackgroundSizeChanged = true;
120 }
121 return super.setFrame(left, top, right, bottom);
122 }
123
124 @Override
125 protected boolean verifyDrawable(Drawable who) {
126 return who == mBackground || super.verifyDrawable(who);
127 }
128
Michael Jurkaddd62e92011-02-16 17:49:14 -0800129 private void invalidatePressedOrFocusedBackground() {
130 int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS / 2;
Michael Jurkab02972c2011-02-23 16:21:57 -0800131 View parent = (View) getParent();
132 if (parent != null) {
133 parent.invalidate(getLeft() - padding, getTop() - padding,
134 getRight() + padding, getBottom() + padding);
135 }
Michael Jurkaddd62e92011-02-16 17:49:14 -0800136 invalidate();
137 }
138
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800139 @Override
140 protected void drawableStateChanged() {
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800141 if (isPressed()) {
142 // In this case, we have already created the pressed outline on ACTION_DOWN,
143 // so we just need to do an invalidate to trigger draw
144 if (!mDidInvalidateForPressedState) {
Michael Jurkaddd62e92011-02-16 17:49:14 -0800145 invalidatePressedOrFocusedBackground();
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800146 }
147 } else {
148 // Otherwise, either clear the pressed/focused background, or create a background
149 // for the focused state
150 final boolean backgroundEmptyBefore = mPressedOrFocusedBackground == null;
Michael Jurkaddd62e92011-02-16 17:49:14 -0800151 if (!mStayPressed) {
152 mPressedOrFocusedBackground = null;
153 }
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800154 if (isFocused()) {
Patrick Dubroya017c032011-03-09 15:58:32 -0800155 if (mLayout == null) {
156 // In some cases, we get focus before we have been layed out. Set the
157 // background to null so that it will get created when the view is drawn.
158 mPressedOrFocusedBackground = null;
159 } else {
160 mPressedOrFocusedBackground = createGlowingOutline(
161 mTempCanvas, mFocusedGlowColor, mFocusedOutlineColor);
162 }
Michael Jurkaddd62e92011-02-16 17:49:14 -0800163 mStayPressed = false;
164 invalidatePressedOrFocusedBackground();
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800165 }
166 final boolean backgroundEmptyNow = mPressedOrFocusedBackground == null;
167 if (!backgroundEmptyBefore && backgroundEmptyNow) {
Michael Jurkaddd62e92011-02-16 17:49:14 -0800168 invalidatePressedOrFocusedBackground();
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800169 }
170 }
171
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800172 Drawable d = mBackground;
173 if (d != null && d.isStateful()) {
174 d.setState(getDrawableState());
175 }
176 super.drawableStateChanged();
177 }
178
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800179 /**
Patrick Dubroycd953712011-02-28 15:16:42 -0800180 * Draw this BubbleTextView into the given Canvas.
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800181 *
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800182 * @param destCanvas the canvas to draw on
183 * @param padding the horizontal and vertical padding to use when drawing
184 */
185 private void drawWithPadding(Canvas destCanvas, int padding) {
186 final Rect clipRect = mTempRect;
187 getDrawingRect(clipRect);
188
189 // adjust the clip rect so that we don't include the text label
190 clipRect.bottom =
191 getExtendedPaddingTop() - (int) BubbleTextView.PADDING_V + getLayout().getLineTop(0);
192
193 // Draw the View into the bitmap.
194 // The translate of scrollX and scrollY is necessary when drawing TextViews, because
195 // they set scrollX and scrollY to large values to achieve centered text
196 destCanvas.save();
197 destCanvas.translate(-getScrollX() + padding / 2, -getScrollY() + padding / 2);
198 destCanvas.clipRect(clipRect, Op.REPLACE);
Patrick Dubroya017c032011-03-09 15:58:32 -0800199 drawImpl(destCanvas, true);
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800200 destCanvas.restore();
201 }
202
203 /**
204 * Returns a new bitmap to be used as the object outline, e.g. to visualize the drop location.
205 * Responsibility for the bitmap is transferred to the caller.
206 */
207 private Bitmap createGlowingOutline(Canvas canvas, int outlineColor, int glowColor) {
208 final int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS;
209 final Bitmap b = Bitmap.createBitmap(
210 getWidth() + padding, getHeight() + padding, Bitmap.Config.ARGB_8888);
211
212 canvas.setBitmap(b);
213 drawWithPadding(canvas, padding);
214 mOutlineHelper.applyExtraThickExpensiveOutlineWithBlur(b, canvas, glowColor, outlineColor);
215
216 return b;
217 }
218
219 @Override
220 public boolean onTouchEvent(MotionEvent event) {
221 // Call the superclass onTouchEvent first, because sometimes it changes the state to
222 // isPressed() on an ACTION_UP
223 boolean result = super.onTouchEvent(event);
224
225 switch (event.getAction()) {
226 case MotionEvent.ACTION_DOWN:
227 // So that the pressed outline is visible immediately when isPressed() is true,
228 // we pre-create it on ACTION_DOWN (it takes a small but perceptible amount of time
229 // to create it)
230 if (mPressedOrFocusedBackground == null) {
231 mPressedOrFocusedBackground = createGlowingOutline(
232 mTempCanvas, mPressedGlowColor, mPressedOutlineColor);
233 }
234 // Invalidate so the pressed state is visible, or set a flag so we know that we
235 // have to call invalidate as soon as the state is "pressed"
236 if (isPressed()) {
237 mDidInvalidateForPressedState = true;
238 invalidate();
239 } else {
240 mDidInvalidateForPressedState = false;
241 }
242 break;
243 case MotionEvent.ACTION_CANCEL:
244 case MotionEvent.ACTION_UP:
245 // If we've touched down and up on an item, and it's still not "pressed", then
246 // destroy the pressed outline
247 if (!isPressed()) {
248 mPressedOrFocusedBackground = null;
249 }
250 break;
251 }
252 return result;
253 }
254
Michael Jurka99b6a5b2011-01-07 15:37:17 -0800255 public void setVisibilityChangedListener(VisibilityChangedListener listener) {
256 mOnVisibilityChangedListener = listener;
257 }
258
259 @Override
260 protected void onVisibilityChanged(View changedView, int visibility) {
261 if (mOnVisibilityChangedListener != null) {
262 mOnVisibilityChangedListener.receiveVisibilityChangedMessage(this);
263 }
264 super.onVisibilityChanged(changedView, visibility);
265 }
266
Michael Jurkaddd62e92011-02-16 17:49:14 -0800267 void setStayPressed(boolean stayPressed) {
268 mStayPressed = stayPressed;
269 if (!stayPressed) {
270 mPressedOrFocusedBackground = null;
271 }
272 invalidatePressedOrFocusedBackground();
273 }
Patrick Dubroya017c032011-03-09 15:58:32 -0800274
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800275 @Override
276 public void draw(Canvas canvas) {
Patrick Dubroya017c032011-03-09 15:58:32 -0800277 drawImpl(canvas, false);
278 }
279
280 private void drawImpl(Canvas canvas, boolean preventRecursion) {
281 // If the View is focused but the focused background hasn't been created yet, create it now
282 if (!preventRecursion && isFocused() && mPressedOrFocusedBackground == null) {
283 mPressedOrFocusedBackground = createGlowingOutline(
284 mTempCanvas, mFocusedGlowColor, mFocusedOutlineColor);
285 }
286
Michael Jurkaddd62e92011-02-16 17:49:14 -0800287 if (mPressedOrFocusedBackground != null && (isPressed() || isFocused() || mStayPressed)) {
Michael Jurka137142e2011-01-05 20:57:04 -0800288 // The blue glow can extend outside of our clip region, so we first temporarily expand
289 // the canvas's clip region
290 canvas.save(Canvas.CLIP_SAVE_FLAG);
291 int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS / 2;
292 canvas.clipRect(-padding + mScrollX, -padding + mScrollY,
293 getWidth() + padding + mScrollX, getHeight() + padding + mScrollY,
294 Region.Op.REPLACE);
295 // draw blue glow
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800296 canvas.drawBitmap(mPressedOrFocusedBackground,
Michael Jurka137142e2011-01-05 20:57:04 -0800297 mScrollX - padding, mScrollY - padding, mTempPaint);
298 canvas.restore();
Michael Jurka38b4f7c2010-12-14 16:46:39 -0800299 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800300
Michael Jurkabdb5c532011-02-01 15:05:06 -0800301 final Drawable background = mBackground;
302 if (background != null) {
303 final int scrollX = mScrollX;
304 final int scrollY = mScrollY;
Winson Chung88127032010-12-13 12:11:33 -0800305
Michael Jurkabdb5c532011-02-01 15:05:06 -0800306 if (mBackgroundSizeChanged) {
307 background.setBounds(0, 0, mRight - mLeft, mBottom - mTop);
308 mBackgroundSizeChanged = false;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800309 }
Michael Jurkabdb5c532011-02-01 15:05:06 -0800310
311 if ((scrollX | scrollY) == 0) {
312 background.draw(canvas);
313 } else {
314 canvas.translate(scrollX, scrollY);
315 background.draw(canvas);
316 canvas.translate(-scrollX, -scrollY);
317 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800318 }
Michael Jurkabdb5c532011-02-01 15:05:06 -0800319 // We enhance the shadow by drawing the shadow twice
Michael Jurkae7e3f6c2011-02-01 21:08:29 -0800320 getPaint().setShadowLayer(SHADOW_LARGE_RADIUS, 0.0f, SHADOW_Y_OFFSET, SHADOW_LARGE_COLOUR);
Michael Jurkabdb5c532011-02-01 15:05:06 -0800321 super.draw(canvas);
322 canvas.save(Canvas.CLIP_SAVE_FLAG);
323 canvas.clipRect(mScrollX, mScrollY + getExtendedPaddingTop(), mScrollX + getWidth(),
Winson Chung09693002011-02-03 23:56:47 -0800324 mScrollY + getHeight(), Region.Op.INTERSECT);
Michael Jurkae7e3f6c2011-02-01 21:08:29 -0800325 getPaint().setShadowLayer(SHADOW_SMALL_RADIUS, 0.0f, 0.0f, SHADOW_SMALL_COLOUR);
Michael Jurkabdb5c532011-02-01 15:05:06 -0800326 super.draw(canvas);
327 canvas.restore();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800328 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400329
330 @Override
331 protected void onAttachedToWindow() {
332 super.onAttachedToWindow();
Winson Chung656d11c2010-11-29 17:15:47 -0800333 if (mBackground != null) mBackground.setCallback(this);
Joe Onorato9c1289c2009-08-17 11:03:03 -0400334 }
335
336 @Override
337 protected void onDetachedFromWindow() {
338 super.onDetachedFromWindow();
Winson Chung656d11c2010-11-29 17:15:47 -0800339 if (mBackground != null) mBackground.setCallback(null);
Joe Onorato9c1289c2009-08-17 11:03:03 -0400340 }
Winson Chungaffd7b42010-08-20 15:11:56 -0700341
342 @Override
343 protected boolean onSetAlpha(int alpha) {
Winson Chunge22a8e92010-11-12 13:40:58 -0800344 if (mPrevAlpha != alpha) {
345 mPrevAlpha = alpha;
Winson Chung656d11c2010-11-29 17:15:47 -0800346 mPaint.setAlpha((int) (alpha * mBubbleColorAlpha));
Winson Chunge22a8e92010-11-12 13:40:58 -0800347 super.onSetAlpha(alpha);
348 }
349 return true;
Winson Chungaffd7b42010-08-20 15:11:56 -0700350 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800351}