blob: a835d9969ed53743b8e8bf0b88330f3b2094d0b2 [file] [log] [blame]
/*
* Copyright (C) 2011 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.
*/
package com.android.launcher3;
import android.animation.Animator;
import android.animation.PropertyValuesHolder;
import android.content.Context;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.util.Pair;
import android.view.View;
import com.android.launcher3.util.Thunk;
public class FocusIndicatorView extends View implements View.OnFocusChangeListener {
// It can be any number >0. The view is resized using scaleX and scaleY.
static final int DEFAULT_LAYOUT_SIZE = 100;
private static final float MIN_VISIBLE_ALPHA = 0.2f;
private static final long ANIM_DURATION = 150;
private final int[] mIndicatorPos = new int[2];
private final int[] mTargetViewPos = new int[2];
private Animator mCurrentAnimation;
private ViewAnimState mTargetState;
private View mLastFocusedView;
private boolean mInitiated;
private final OnFocusChangeListener mHideIndicatorOnFocusListener;
private Pair<View, Boolean> mPendingCall;
public FocusIndicatorView(Context context) {
this(context, null);
}
public FocusIndicatorView(Context context, AttributeSet attrs) {
super(context, attrs);
setAlpha(0);
setBackgroundColor(getResources().getColor(R.color.focused_background));
mHideIndicatorOnFocusListener = new OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus) {
endCurrentAnimation();
setAlpha(0);
}
}
};
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
// Redraw if it is already showing. This avoids a bug where the height changes by a small
// amount on connecting/disconnecting a bluetooth keyboard.
if (mLastFocusedView != null) {
mPendingCall = Pair.create(mLastFocusedView, Boolean.TRUE);
invalidate();
}
}
/**
* Sets the alpha of this FocusIndicatorView to 0 when a view with this listener receives focus.
*/
public View.OnFocusChangeListener getHideIndicatorOnFocusListener() {
return mHideIndicatorOnFocusListener;
}
@Override
public void onFocusChange(View v, boolean hasFocus) {
mPendingCall = null;
if (!mInitiated && (getWidth() == 0)) {
// View not yet laid out. Wait until the view is ready to be drawn, so that be can
// get the location on screen.
mPendingCall = Pair.create(v, hasFocus);
invalidate();
return;
}
if (!mInitiated) {
// The parent view should always the a parent of the target view.
computeLocationRelativeToParent(this, (View) getParent(), mIndicatorPos);
mInitiated = true;
}
if (hasFocus) {
int indicatorWidth = getWidth();
int indicatorHeight = getHeight();
endCurrentAnimation();
ViewAnimState nextState = new ViewAnimState();
nextState.scaleX = v.getScaleX() * v.getWidth() / indicatorWidth;
nextState.scaleY = v.getScaleY() * v.getHeight() / indicatorHeight;
computeLocationRelativeToParent(v, (View) getParent(), mTargetViewPos);
nextState.x = mTargetViewPos[0] - mIndicatorPos[0] - (1 - nextState.scaleX) * indicatorWidth / 2;
nextState.y = mTargetViewPos[1] - mIndicatorPos[1] - (1 - nextState.scaleY) * indicatorHeight / 2;
if (getAlpha() > MIN_VISIBLE_ALPHA) {
mTargetState = nextState;
mCurrentAnimation = new LauncherViewPropertyAnimator(this)
.alpha(1)
.translationX(mTargetState.x)
.translationY(mTargetState.y)
.scaleX(mTargetState.scaleX)
.scaleY(mTargetState.scaleY);
} else {
applyState(nextState);
mCurrentAnimation = LauncherAnimUtils.ofPropertyValuesHolder(this,
PropertyValuesHolder.ofFloat(View.ALPHA, 1));
}
mLastFocusedView = v;
} else {
if (mLastFocusedView == v) {
mLastFocusedView = null;
endCurrentAnimation();
mCurrentAnimation = LauncherAnimUtils.ofPropertyValuesHolder(this,
PropertyValuesHolder.ofFloat(View.ALPHA, 0));
}
}
if (mCurrentAnimation != null) {
mCurrentAnimation.setDuration(ANIM_DURATION).start();
}
}
private void endCurrentAnimation() {
if (mCurrentAnimation != null) {
mCurrentAnimation.cancel();
mCurrentAnimation = null;
}
if (mTargetState != null) {
applyState(mTargetState);
mTargetState = null;
}
}
private void applyState(ViewAnimState state) {
setTranslationX(state.x);
setTranslationY(state.y);
setScaleX(state.scaleX);
setScaleY(state.scaleY);
}
@Override
protected void onDraw(Canvas canvas) {
if (mPendingCall != null) {
onFocusChange(mPendingCall.first, mPendingCall.second);
}
}
/**
* Computes the location of a view relative to {@param parent}, off-setting
* any shift due to page view scroll.
* @param pos an array of two integers in which to hold the coordinates
*/
private static void computeLocationRelativeToParent(View v, View parent, int[] pos) {
pos[0] = pos[1] = 0;
computeLocationRelativeToParentHelper(v, parent, pos);
// If a view is scaled, its position will also shift accordingly. For optimization, only
// consider this for the last node.
pos[0] += (1 - v.getScaleX()) * v.getWidth() / 2;
pos[1] += (1 - v.getScaleY()) * v.getHeight() / 2;
}
private static void computeLocationRelativeToParentHelper(View child,
View commonParent, int[] shift) {
View parent = (View) child.getParent();
shift[0] += child.getLeft();
shift[1] += child.getTop();
if (parent instanceof PagedView) {
PagedView page = (PagedView) parent;
shift[0] -= page.getScrollForPage(page.indexOfChild(child));
}
if (parent != commonParent) {
computeLocationRelativeToParentHelper(parent, commonParent, shift);
}
}
@Thunk static final class ViewAnimState {
float x, y, scaleX, scaleY;
}
}