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.
|
|
|
|
|
*/
|
|
|
|
|
|
2009-07-30 13:37:37 -07:00
|
|
|
package com.android.launcher2;
|
2009-03-03 19:32:27 -08:00
|
|
|
|
2010-12-14 16:46:39 -08:00
|
|
|
import com.android.launcher.R;
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.content.Context;
|
2010-11-29 17:15:47 -08:00
|
|
|
import android.content.res.Resources;
|
2010-11-17 12:33:46 -08:00
|
|
|
import android.graphics.Bitmap;
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.graphics.Canvas;
|
2010-11-29 17:15:47 -08:00
|
|
|
import android.graphics.Color;
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.graphics.Paint;
|
2010-12-14 16:46:39 -08:00
|
|
|
import android.graphics.Rect;
|
2011-01-05 20:57:04 -08:00
|
|
|
import android.graphics.Region;
|
2010-12-14 16:46:39 -08:00
|
|
|
import android.graphics.Region.Op;
|
2009-03-03 19:32:27 -08:00
|
|
|
import android.graphics.drawable.Drawable;
|
2010-11-29 17:15:47 -08:00
|
|
|
import android.util.AttributeSet;
|
2010-12-14 16:46:39 -08:00
|
|
|
import android.view.MotionEvent;
|
2011-01-07 15:37:17 -08:00
|
|
|
import android.view.View;
|
2011-02-01 15:05:06 -08:00
|
|
|
import android.widget.TextView;
|
2010-03-04 13:03:17 -08:00
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
/**
|
|
|
|
|
* TextView that draws a bubble behind the text. We cannot use a LineBackgroundSpan
|
|
|
|
|
* because we want to make the bubble taller than the text and TextView's clip is
|
|
|
|
|
* too aggressive.
|
|
|
|
|
*/
|
2011-02-01 15:05:06 -08:00
|
|
|
public class BubbleTextView extends TextView implements VisibilityChangedBroadcaster {
|
2010-11-29 17:15:47 -08:00
|
|
|
static final float CORNER_RADIUS = 4.0f;
|
2010-12-13 12:11:33 -08:00
|
|
|
static final float SHADOW_LARGE_RADIUS = 4.0f;
|
|
|
|
|
static final float SHADOW_SMALL_RADIUS = 1.75f;
|
|
|
|
|
static final float SHADOW_Y_OFFSET = 2.0f;
|
|
|
|
|
static final int SHADOW_LARGE_COLOUR = 0xCC000000;
|
|
|
|
|
static final int SHADOW_SMALL_COLOUR = 0xBB000000;
|
2010-11-29 17:15:47 -08:00
|
|
|
static final float PADDING_H = 8.0f;
|
|
|
|
|
static final float PADDING_V = 3.0f;
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
private Paint mPaint;
|
2010-11-29 17:15:47 -08:00
|
|
|
private float mBubbleColorAlpha;
|
2010-11-12 13:40:58 -08:00
|
|
|
private int mPrevAlpha = -1;
|
2009-03-03 19:32:27 -08:00
|
|
|
|
2010-12-14 16:46:39 -08:00
|
|
|
private final HolographicOutlineHelper mOutlineHelper = new HolographicOutlineHelper();
|
|
|
|
|
private final Canvas mTempCanvas = new Canvas();
|
|
|
|
|
private final Rect mTempRect = new Rect();
|
|
|
|
|
private final Paint mTempPaint = new Paint();
|
|
|
|
|
private boolean mDidInvalidateForPressedState;
|
|
|
|
|
private Bitmap mPressedOrFocusedBackground;
|
|
|
|
|
private int mFocusedOutlineColor;
|
|
|
|
|
private int mFocusedGlowColor;
|
|
|
|
|
private int mPressedOutlineColor;
|
|
|
|
|
private int mPressedGlowColor;
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
private boolean mBackgroundSizeChanged;
|
|
|
|
|
private Drawable mBackground;
|
|
|
|
|
|
2011-02-16 17:49:14 -08:00
|
|
|
private boolean mStayPressed;
|
|
|
|
|
|
2011-01-07 15:37:17 -08:00
|
|
|
private VisibilityChangedListener mOnVisibilityChangedListener;
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
public BubbleTextView(Context context) {
|
|
|
|
|
super(context);
|
|
|
|
|
init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public BubbleTextView(Context context, AttributeSet attrs) {
|
|
|
|
|
super(context, attrs);
|
|
|
|
|
init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public BubbleTextView(Context context, AttributeSet attrs, int defStyle) {
|
|
|
|
|
super(context, attrs, defStyle);
|
|
|
|
|
init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void init() {
|
|
|
|
|
mBackground = getBackground();
|
2010-11-29 17:15:47 -08:00
|
|
|
setFocusable(true);
|
2009-03-03 19:32:27 -08:00
|
|
|
setBackgroundDrawable(null);
|
|
|
|
|
|
2010-11-29 17:15:47 -08:00
|
|
|
final Resources res = getContext().getResources();
|
|
|
|
|
int bubbleColor = res.getColor(R.color.bubble_dark_background);
|
2009-03-03 19:32:27 -08:00
|
|
|
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
|
2010-11-29 17:15:47 -08:00
|
|
|
mPaint.setColor(bubbleColor);
|
|
|
|
|
mBubbleColorAlpha = Color.alpha(bubbleColor) / 255.0f;
|
2010-12-21 11:31:54 -08:00
|
|
|
mFocusedOutlineColor = res.getColor(R.color.workspace_item_focused_outline_color);
|
|
|
|
|
mFocusedGlowColor = res.getColor(R.color.workspace_item_focused_glow_color);
|
|
|
|
|
mPressedOutlineColor = res.getColor(R.color.workspace_item_pressed_outline_color);
|
|
|
|
|
mPressedGlowColor = res.getColor(R.color.workspace_item_pressed_glow_color);
|
2011-02-01 21:08:29 -08:00
|
|
|
|
|
|
|
|
setShadowLayer(SHADOW_LARGE_RADIUS, 0.0f, SHADOW_Y_OFFSET, SHADOW_LARGE_COLOUR);
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
|
|
|
|
|
2010-11-17 12:33:46 -08:00
|
|
|
public void applyFromShortcutInfo(ShortcutInfo info, IconCache iconCache) {
|
|
|
|
|
Bitmap b = info.getIcon(iconCache);
|
|
|
|
|
|
|
|
|
|
setCompoundDrawablesWithIntrinsicBounds(null,
|
|
|
|
|
new FastBitmapDrawable(b),
|
|
|
|
|
null, null);
|
|
|
|
|
setText(info.title);
|
|
|
|
|
setTag(info);
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
@Override
|
|
|
|
|
protected boolean setFrame(int left, int top, int right, int bottom) {
|
|
|
|
|
if (mLeft != left || mRight != right || mTop != top || mBottom != bottom) {
|
|
|
|
|
mBackgroundSizeChanged = true;
|
|
|
|
|
}
|
|
|
|
|
return super.setFrame(left, top, right, bottom);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected boolean verifyDrawable(Drawable who) {
|
|
|
|
|
return who == mBackground || super.verifyDrawable(who);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-16 17:49:14 -08:00
|
|
|
private void invalidatePressedOrFocusedBackground() {
|
|
|
|
|
int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS / 2;
|
2011-02-23 16:21:57 -08:00
|
|
|
View parent = (View) getParent();
|
|
|
|
|
if (parent != null) {
|
|
|
|
|
parent.invalidate(getLeft() - padding, getTop() - padding,
|
|
|
|
|
getRight() + padding, getBottom() + padding);
|
|
|
|
|
}
|
2011-02-16 17:49:14 -08:00
|
|
|
invalidate();
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
@Override
|
|
|
|
|
protected void drawableStateChanged() {
|
2010-12-14 16:46:39 -08:00
|
|
|
if (isPressed()) {
|
|
|
|
|
// In this case, we have already created the pressed outline on ACTION_DOWN,
|
|
|
|
|
// so we just need to do an invalidate to trigger draw
|
|
|
|
|
if (!mDidInvalidateForPressedState) {
|
2011-02-16 17:49:14 -08:00
|
|
|
invalidatePressedOrFocusedBackground();
|
2010-12-14 16:46:39 -08:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Otherwise, either clear the pressed/focused background, or create a background
|
|
|
|
|
// for the focused state
|
|
|
|
|
final boolean backgroundEmptyBefore = mPressedOrFocusedBackground == null;
|
2011-02-16 17:49:14 -08:00
|
|
|
if (!mStayPressed) {
|
|
|
|
|
mPressedOrFocusedBackground = null;
|
|
|
|
|
}
|
2010-12-14 16:46:39 -08:00
|
|
|
if (isFocused()) {
|
|
|
|
|
mPressedOrFocusedBackground = createGlowingOutline(
|
|
|
|
|
mTempCanvas, mFocusedGlowColor, mFocusedOutlineColor);
|
2011-02-16 17:49:14 -08:00
|
|
|
mStayPressed = false;
|
|
|
|
|
invalidatePressedOrFocusedBackground();
|
2010-12-14 16:46:39 -08:00
|
|
|
}
|
|
|
|
|
final boolean backgroundEmptyNow = mPressedOrFocusedBackground == null;
|
|
|
|
|
if (!backgroundEmptyBefore && backgroundEmptyNow) {
|
2011-02-16 17:49:14 -08:00
|
|
|
invalidatePressedOrFocusedBackground();
|
2010-12-14 16:46:39 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-03 19:32:27 -08:00
|
|
|
Drawable d = mBackground;
|
|
|
|
|
if (d != null && d.isStateful()) {
|
|
|
|
|
d.setState(getDrawableState());
|
|
|
|
|
}
|
|
|
|
|
super.drawableStateChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 16:46:39 -08:00
|
|
|
/**
|
2011-02-28 15:16:42 -08:00
|
|
|
* Draw this BubbleTextView into the given Canvas.
|
2010-12-14 16:46:39 -08:00
|
|
|
*
|
|
|
|
|
* @param destCanvas the canvas to draw on
|
|
|
|
|
* @param padding the horizontal and vertical padding to use when drawing
|
|
|
|
|
*/
|
|
|
|
|
private void drawWithPadding(Canvas destCanvas, int padding) {
|
|
|
|
|
final Rect clipRect = mTempRect;
|
|
|
|
|
getDrawingRect(clipRect);
|
|
|
|
|
|
|
|
|
|
// adjust the clip rect so that we don't include the text label
|
|
|
|
|
clipRect.bottom =
|
|
|
|
|
getExtendedPaddingTop() - (int) BubbleTextView.PADDING_V + getLayout().getLineTop(0);
|
|
|
|
|
|
|
|
|
|
// Draw the View into the bitmap.
|
|
|
|
|
// The translate of scrollX and scrollY is necessary when drawing TextViews, because
|
|
|
|
|
// they set scrollX and scrollY to large values to achieve centered text
|
|
|
|
|
destCanvas.save();
|
|
|
|
|
destCanvas.translate(-getScrollX() + padding / 2, -getScrollY() + padding / 2);
|
|
|
|
|
destCanvas.clipRect(clipRect, Op.REPLACE);
|
|
|
|
|
draw(destCanvas);
|
|
|
|
|
destCanvas.restore();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a new bitmap to be used as the object outline, e.g. to visualize the drop location.
|
|
|
|
|
* Responsibility for the bitmap is transferred to the caller.
|
|
|
|
|
*/
|
|
|
|
|
private Bitmap createGlowingOutline(Canvas canvas, int outlineColor, int glowColor) {
|
|
|
|
|
final int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS;
|
|
|
|
|
final Bitmap b = Bitmap.createBitmap(
|
|
|
|
|
getWidth() + padding, getHeight() + padding, Bitmap.Config.ARGB_8888);
|
|
|
|
|
|
|
|
|
|
canvas.setBitmap(b);
|
|
|
|
|
drawWithPadding(canvas, padding);
|
|
|
|
|
mOutlineHelper.applyExtraThickExpensiveOutlineWithBlur(b, canvas, glowColor, outlineColor);
|
|
|
|
|
|
|
|
|
|
return b;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onTouchEvent(MotionEvent event) {
|
|
|
|
|
// Call the superclass onTouchEvent first, because sometimes it changes the state to
|
|
|
|
|
// isPressed() on an ACTION_UP
|
|
|
|
|
boolean result = super.onTouchEvent(event);
|
|
|
|
|
|
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
|
// So that the pressed outline is visible immediately when isPressed() is true,
|
|
|
|
|
// we pre-create it on ACTION_DOWN (it takes a small but perceptible amount of time
|
|
|
|
|
// to create it)
|
|
|
|
|
if (mPressedOrFocusedBackground == null) {
|
|
|
|
|
mPressedOrFocusedBackground = createGlowingOutline(
|
|
|
|
|
mTempCanvas, mPressedGlowColor, mPressedOutlineColor);
|
|
|
|
|
}
|
|
|
|
|
// Invalidate so the pressed state is visible, or set a flag so we know that we
|
|
|
|
|
// have to call invalidate as soon as the state is "pressed"
|
|
|
|
|
if (isPressed()) {
|
|
|
|
|
mDidInvalidateForPressedState = true;
|
|
|
|
|
invalidate();
|
|
|
|
|
} else {
|
|
|
|
|
mDidInvalidateForPressedState = false;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case MotionEvent.ACTION_CANCEL:
|
|
|
|
|
case MotionEvent.ACTION_UP:
|
|
|
|
|
// If we've touched down and up on an item, and it's still not "pressed", then
|
|
|
|
|
// destroy the pressed outline
|
|
|
|
|
if (!isPressed()) {
|
|
|
|
|
mPressedOrFocusedBackground = null;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2011-01-07 15:37:17 -08:00
|
|
|
public void setVisibilityChangedListener(VisibilityChangedListener listener) {
|
|
|
|
|
mOnVisibilityChangedListener = listener;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onVisibilityChanged(View changedView, int visibility) {
|
|
|
|
|
if (mOnVisibilityChangedListener != null) {
|
|
|
|
|
mOnVisibilityChangedListener.receiveVisibilityChangedMessage(this);
|
|
|
|
|
}
|
|
|
|
|
super.onVisibilityChanged(changedView, visibility);
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-16 17:49:14 -08:00
|
|
|
void setStayPressed(boolean stayPressed) {
|
|
|
|
|
mStayPressed = stayPressed;
|
|
|
|
|
if (!stayPressed) {
|
|
|
|
|
mPressedOrFocusedBackground = null;
|
|
|
|
|
}
|
|
|
|
|
invalidatePressedOrFocusedBackground();
|
|
|
|
|
}
|
2009-03-03 19:32:27 -08:00
|
|
|
@Override
|
|
|
|
|
public void draw(Canvas canvas) {
|
2011-02-16 17:49:14 -08:00
|
|
|
if (mPressedOrFocusedBackground != null && (isPressed() || isFocused() || mStayPressed)) {
|
2011-01-05 20:57:04 -08:00
|
|
|
// The blue glow can extend outside of our clip region, so we first temporarily expand
|
|
|
|
|
// the canvas's clip region
|
|
|
|
|
canvas.save(Canvas.CLIP_SAVE_FLAG);
|
|
|
|
|
int padding = HolographicOutlineHelper.MAX_OUTER_BLUR_RADIUS / 2;
|
|
|
|
|
canvas.clipRect(-padding + mScrollX, -padding + mScrollY,
|
|
|
|
|
getWidth() + padding + mScrollX, getHeight() + padding + mScrollY,
|
|
|
|
|
Region.Op.REPLACE);
|
|
|
|
|
// draw blue glow
|
2010-12-14 16:46:39 -08:00
|
|
|
canvas.drawBitmap(mPressedOrFocusedBackground,
|
2011-01-05 20:57:04 -08:00
|
|
|
mScrollX - padding, mScrollY - padding, mTempPaint);
|
|
|
|
|
canvas.restore();
|
2010-12-14 16:46:39 -08:00
|
|
|
}
|
2010-12-13 12:11:33 -08:00
|
|
|
|
2011-02-01 15:05:06 -08:00
|
|
|
final Drawable background = mBackground;
|
|
|
|
|
if (background != null) {
|
|
|
|
|
final int scrollX = mScrollX;
|
|
|
|
|
final int scrollY = mScrollY;
|
|
|
|
|
|
|
|
|
|
if (mBackgroundSizeChanged) {
|
|
|
|
|
background.setBounds(0, 0, mRight - mLeft, mBottom - mTop);
|
|
|
|
|
mBackgroundSizeChanged = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((scrollX | scrollY) == 0) {
|
|
|
|
|
background.draw(canvas);
|
|
|
|
|
} else {
|
|
|
|
|
canvas.translate(scrollX, scrollY);
|
|
|
|
|
background.draw(canvas);
|
|
|
|
|
canvas.translate(-scrollX, -scrollY);
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
2010-12-04 18:11:57 -08:00
|
|
|
}
|
2011-02-01 15:05:06 -08:00
|
|
|
// We enhance the shadow by drawing the shadow twice
|
2011-02-01 21:08:29 -08:00
|
|
|
getPaint().setShadowLayer(SHADOW_LARGE_RADIUS, 0.0f, SHADOW_Y_OFFSET, SHADOW_LARGE_COLOUR);
|
2011-02-01 15:05:06 -08:00
|
|
|
super.draw(canvas);
|
|
|
|
|
canvas.save(Canvas.CLIP_SAVE_FLAG);
|
|
|
|
|
canvas.clipRect(mScrollX, mScrollY + getExtendedPaddingTop(), mScrollX + getWidth(),
|
2011-02-03 23:56:47 -08:00
|
|
|
mScrollY + getHeight(), Region.Op.INTERSECT);
|
2011-02-01 21:08:29 -08:00
|
|
|
getPaint().setShadowLayer(SHADOW_SMALL_RADIUS, 0.0f, 0.0f, SHADOW_SMALL_COLOUR);
|
2011-02-01 15:05:06 -08:00
|
|
|
super.draw(canvas);
|
|
|
|
|
canvas.restore();
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|
2009-08-17 11:03:03 -04:00
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onAttachedToWindow() {
|
|
|
|
|
super.onAttachedToWindow();
|
2010-11-29 17:15:47 -08:00
|
|
|
if (mBackground != null) mBackground.setCallback(this);
|
2009-08-17 11:03:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onDetachedFromWindow() {
|
|
|
|
|
super.onDetachedFromWindow();
|
2010-11-29 17:15:47 -08:00
|
|
|
if (mBackground != null) mBackground.setCallback(null);
|
2009-08-17 11:03:03 -04:00
|
|
|
}
|
2010-08-20 15:11:56 -07:00
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected boolean onSetAlpha(int alpha) {
|
2010-11-12 13:40:58 -08:00
|
|
|
if (mPrevAlpha != alpha) {
|
|
|
|
|
mPrevAlpha = alpha;
|
2010-11-29 17:15:47 -08:00
|
|
|
mPaint.setAlpha((int) (alpha * mBubbleColorAlpha));
|
2010-11-12 13:40:58 -08:00
|
|
|
super.onSetAlpha(alpha);
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2010-08-20 15:11:56 -07:00
|
|
|
}
|
2009-03-03 19:32:27 -08:00
|
|
|
}
|