471e2178f8
In workspace, the page indicator dots end up in the bottom because we were missing a setInsets method (WorkspacePageInidcator.java uses this method to set the page indicator to the right height). Bug: 249773534 Test: manual Change-Id: I03636e28244f329b249453a6d22eb0ec7d85c45f
496 lines
18 KiB
Java
496 lines
18 KiB
Java
/*
|
|
* Copyright (C) 2016 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.pageindicators;
|
|
|
|
import static com.android.launcher3.config.FeatureFlags.SHOW_DELIGHTFUL_PAGINATION;
|
|
|
|
import android.animation.Animator;
|
|
import android.animation.AnimatorListenerAdapter;
|
|
import android.animation.AnimatorSet;
|
|
import android.animation.ObjectAnimator;
|
|
import android.animation.ValueAnimator;
|
|
import android.animation.ValueAnimator.AnimatorUpdateListener;
|
|
import android.content.Context;
|
|
import android.graphics.Canvas;
|
|
import android.graphics.Outline;
|
|
import android.graphics.Paint;
|
|
import android.graphics.Paint.Style;
|
|
import android.graphics.Rect;
|
|
import android.graphics.RectF;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.util.AttributeSet;
|
|
import android.util.Property;
|
|
import android.view.View;
|
|
import android.view.ViewOutlineProvider;
|
|
import android.view.animation.Interpolator;
|
|
import android.view.animation.OvershootInterpolator;
|
|
|
|
import com.android.launcher3.Insettable;
|
|
import com.android.launcher3.R;
|
|
import com.android.launcher3.Utilities;
|
|
import com.android.launcher3.anim.Interpolators;
|
|
import com.android.launcher3.util.Themes;
|
|
|
|
/**
|
|
* {@link PageIndicator} which shows dots per page. The active page is shown with the current
|
|
* accent color.
|
|
*/
|
|
public class PageIndicatorDots extends View implements Insettable, PageIndicator {
|
|
|
|
private static final float SHIFT_PER_ANIMATION = 0.5f;
|
|
private static final float SHIFT_THRESHOLD = 0.1f;
|
|
private static final long ANIMATION_DURATION = 150;
|
|
|
|
private static final int ENTER_ANIMATION_START_DELAY = 300;
|
|
private static final int ENTER_ANIMATION_STAGGERED_DELAY = 150;
|
|
private static final int ENTER_ANIMATION_DURATION = 400;
|
|
|
|
private static final int PAGE_INDICATOR_ALPHA = 255;
|
|
private static final int DOT_ALPHA = 128;
|
|
private static final int DOT_GAP_FACTOR = 3;
|
|
private static final float DOT_GAP_FACTOR_FLOAT = 3.8f;
|
|
|
|
// This value approximately overshoots to 1.5 times the original size.
|
|
private static final float ENTER_ANIMATION_OVERSHOOT_TENSION = 4.9f;
|
|
|
|
private static final float INDICATOR_ROTATION = 180f;
|
|
|
|
private static final RectF sTempRect = new RectF();
|
|
|
|
private static final Property<PageIndicatorDots, Float> CURRENT_POSITION
|
|
= new Property<PageIndicatorDots, Float>(float.class, "current_position") {
|
|
@Override
|
|
public Float get(PageIndicatorDots obj) {
|
|
return obj.mCurrentPosition;
|
|
}
|
|
|
|
@Override
|
|
public void set(PageIndicatorDots obj, Float pos) {
|
|
obj.mCurrentPosition = pos;
|
|
obj.invalidate();
|
|
obj.invalidateOutline();
|
|
}
|
|
};
|
|
|
|
private final Paint mPaginationPaint;
|
|
private final Drawable mPageIndicatorDrawable;
|
|
private final float mDotRadius;
|
|
private final float mCircleGap;
|
|
private final float mPageIndicatorSize;
|
|
private final float mPageIndicatorRadius;
|
|
private final boolean mIsRtl;
|
|
|
|
private int mNumPages;
|
|
private int mActivePage;
|
|
private int mCurrentScroll;
|
|
private int mTotalScroll;
|
|
|
|
/**
|
|
* The current position of the active dot including the animation progress.
|
|
* For ex:
|
|
* 0.0 => Active dot is at position 0
|
|
* 0.33 => Active dot is at position 0 and is moving towards 1
|
|
* 0.50 => Active dot is at position [0, 1]
|
|
* 0.77 => Active dot has left position 0 and is collapsing towards position 1
|
|
* 1.0 => Active dot is at position 1
|
|
*/
|
|
private float mCurrentPosition;
|
|
private float mFinalPosition;
|
|
private ObjectAnimator mAnimator;
|
|
|
|
private float[] mEntryAnimationRadiusFactors;
|
|
|
|
public PageIndicatorDots(Context context) {
|
|
this(context, null);
|
|
}
|
|
|
|
public PageIndicatorDots(Context context, AttributeSet attrs) {
|
|
this(context, attrs, 0);
|
|
}
|
|
|
|
public PageIndicatorDots(Context context, AttributeSet attrs, int defStyleAttr) {
|
|
super(context, attrs, defStyleAttr);
|
|
|
|
mPaginationPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
|
|
mPaginationPaint.setStyle(Style.FILL);
|
|
mPaginationPaint.setColor(Themes.getAttrColor(context, R.attr.folderPaginationColor));
|
|
mDotRadius = getResources().getDimension(R.dimen.page_indicator_dot_size) / 2;
|
|
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
mPageIndicatorSize = getResources().getDimension(
|
|
R.dimen.page_indicator_size);
|
|
mPageIndicatorRadius = mPageIndicatorSize / 2;
|
|
mPageIndicatorDrawable = context.getDrawable(R.drawable.page_indicator);
|
|
mPageIndicatorDrawable.setBounds(0, 0, (int) mPageIndicatorSize,
|
|
(int) mPageIndicatorSize);
|
|
mCircleGap = DOT_GAP_FACTOR_FLOAT * mDotRadius;
|
|
|
|
} else {
|
|
mPageIndicatorSize = 0;
|
|
mPageIndicatorRadius = 0;
|
|
mPageIndicatorDrawable = null;
|
|
mCircleGap = DOT_GAP_FACTOR * mDotRadius;
|
|
}
|
|
if (!SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
setOutlineProvider(new MyOutlineProver());
|
|
}
|
|
mIsRtl = Utilities.isRtl(getResources());
|
|
}
|
|
|
|
@Override
|
|
public void setScroll(int currentScroll, int totalScroll) {
|
|
if (mNumPages <= 1) {
|
|
mCurrentScroll = 0;
|
|
return;
|
|
}
|
|
|
|
if (mIsRtl) {
|
|
currentScroll = totalScroll - currentScroll;
|
|
}
|
|
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
mCurrentScroll = currentScroll;
|
|
mTotalScroll = totalScroll;
|
|
invalidate();
|
|
return;
|
|
}
|
|
|
|
int scrollPerPage = totalScroll / (mNumPages - 1);
|
|
int pageToLeft = currentScroll / scrollPerPage;
|
|
int pageToLeftScroll = pageToLeft * scrollPerPage;
|
|
int pageToRightScroll = pageToLeftScroll + scrollPerPage;
|
|
|
|
float scrollThreshold = SHIFT_THRESHOLD * scrollPerPage;
|
|
if (currentScroll < pageToLeftScroll + scrollThreshold) {
|
|
// scroll is within the left page's threshold
|
|
animateToPosition(pageToLeft);
|
|
} else if (currentScroll > pageToRightScroll - scrollThreshold) {
|
|
// scroll is far enough from left page to go to the right page
|
|
animateToPosition(pageToLeft + 1);
|
|
} else {
|
|
// scroll is between left and right page
|
|
animateToPosition(pageToLeft + SHIFT_PER_ANIMATION);
|
|
}
|
|
}
|
|
|
|
private void animateToPosition(float position) {
|
|
mFinalPosition = position;
|
|
if (Math.abs(mCurrentPosition - mFinalPosition) < SHIFT_THRESHOLD) {
|
|
mCurrentPosition = mFinalPosition;
|
|
}
|
|
if (mAnimator == null && Float.compare(mCurrentPosition, mFinalPosition) != 0) {
|
|
float positionForThisAnim = mCurrentPosition > mFinalPosition ?
|
|
mCurrentPosition - SHIFT_PER_ANIMATION : mCurrentPosition + SHIFT_PER_ANIMATION;
|
|
mAnimator = ObjectAnimator.ofFloat(this, CURRENT_POSITION, positionForThisAnim);
|
|
mAnimator.addListener(new AnimationCycleListener());
|
|
mAnimator.setDuration(ANIMATION_DURATION);
|
|
mAnimator.start();
|
|
}
|
|
}
|
|
|
|
public void stopAllAnimations() {
|
|
if (mAnimator != null) {
|
|
mAnimator.cancel();
|
|
mAnimator = null;
|
|
}
|
|
mFinalPosition = mActivePage;
|
|
CURRENT_POSITION.set(this, mFinalPosition);
|
|
}
|
|
|
|
/**
|
|
* Sets up up the page indicator to play the entry animation.
|
|
* {@link #playEntryAnimation()} must be called after this.
|
|
*/
|
|
public void prepareEntryAnimation() {
|
|
mEntryAnimationRadiusFactors = new float[mNumPages];
|
|
invalidate();
|
|
}
|
|
|
|
public void playEntryAnimation() {
|
|
int count = mEntryAnimationRadiusFactors.length;
|
|
if (count == 0) {
|
|
mEntryAnimationRadiusFactors = null;
|
|
invalidate();
|
|
return;
|
|
}
|
|
|
|
Interpolator interpolator = new OvershootInterpolator(ENTER_ANIMATION_OVERSHOOT_TENSION);
|
|
AnimatorSet animSet = new AnimatorSet();
|
|
for (int i = 0; i < count; i++) {
|
|
ValueAnimator anim = ValueAnimator.ofFloat(0, 1).setDuration(ENTER_ANIMATION_DURATION);
|
|
final int index = i;
|
|
anim.addUpdateListener(new AnimatorUpdateListener() {
|
|
@Override
|
|
public void onAnimationUpdate(ValueAnimator animation) {
|
|
mEntryAnimationRadiusFactors[index] = (Float) animation.getAnimatedValue();
|
|
invalidate();
|
|
}
|
|
});
|
|
anim.setInterpolator(interpolator);
|
|
anim.setStartDelay(ENTER_ANIMATION_START_DELAY + ENTER_ANIMATION_STAGGERED_DELAY * i);
|
|
animSet.play(anim);
|
|
}
|
|
|
|
animSet.addListener(new AnimatorListenerAdapter() {
|
|
|
|
@Override
|
|
public void onAnimationEnd(Animator animation) {
|
|
mEntryAnimationRadiusFactors = null;
|
|
invalidateOutline();
|
|
invalidate();
|
|
}
|
|
});
|
|
animSet.start();
|
|
}
|
|
|
|
@Override
|
|
public void setActiveMarker(int activePage) {
|
|
if (mActivePage != activePage) {
|
|
mActivePage = activePage;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void setMarkersCount(int numMarkers) {
|
|
mNumPages = numMarkers;
|
|
requestLayout();
|
|
}
|
|
|
|
@Override
|
|
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
|
|
// Add extra spacing of mDotRadius on all sides so than entry animation could be run.
|
|
int width = MeasureSpec.getMode(widthMeasureSpec) == MeasureSpec.EXACTLY ?
|
|
MeasureSpec.getSize(widthMeasureSpec) : (int) ((mNumPages * 3 + 2) * mDotRadius);
|
|
int height = MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.EXACTLY
|
|
? MeasureSpec.getSize(heightMeasureSpec) : (int) (4 * mDotRadius);
|
|
setMeasuredDimension(width, height);
|
|
}
|
|
|
|
@Override
|
|
protected void onDraw(Canvas canvas) {
|
|
// Draw all page indicators;
|
|
float circleGap = mCircleGap;
|
|
float startX = (getWidth() - (mNumPages * circleGap) + mDotRadius) / 2;
|
|
|
|
float x = startX + mDotRadius;
|
|
float y = getHeight() / 2;
|
|
|
|
if (mEntryAnimationRadiusFactors != null) {
|
|
// During entry animation, only draw the circles
|
|
if (mIsRtl) {
|
|
x = getWidth() - x;
|
|
circleGap = -circleGap;
|
|
}
|
|
for (int i = 0; i < mEntryAnimationRadiusFactors.length; i++) {
|
|
mPaginationPaint.setAlpha(i == mActivePage ? PAGE_INDICATOR_ALPHA : DOT_ALPHA);
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
if (i != mActivePage) {
|
|
canvas.drawCircle(x, y, mDotRadius * mEntryAnimationRadiusFactors[i],
|
|
mPaginationPaint);
|
|
} else {
|
|
drawPageIndicator(canvas, mEntryAnimationRadiusFactors[i]);
|
|
}
|
|
} else {
|
|
canvas.drawCircle(x, y, mDotRadius * mEntryAnimationRadiusFactors[i],
|
|
mPaginationPaint);
|
|
}
|
|
x += circleGap;
|
|
}
|
|
} else {
|
|
// Here we draw the dots
|
|
mPaginationPaint.setAlpha(DOT_ALPHA);
|
|
for (int i = 0; i < mNumPages; i++) {
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
canvas.drawCircle(x, y, getRadius(x), mPaginationPaint);
|
|
} else {
|
|
canvas.drawCircle(x, y, mDotRadius, mPaginationPaint);
|
|
}
|
|
x += circleGap;
|
|
}
|
|
|
|
// Here we draw the current page indicator
|
|
mPaginationPaint.setAlpha(PAGE_INDICATOR_ALPHA);
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
drawPageIndicator(canvas, 1);
|
|
} else {
|
|
canvas.drawRoundRect(getActiveRect(), mDotRadius, mDotRadius, mPaginationPaint);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Draws the page indicator, denoting the currently selected page
|
|
*
|
|
* @param canvas is used to draw the page indicator and to rotate it as we scroll
|
|
* @param scale is used to set the scale of our canvas
|
|
*/
|
|
private void drawPageIndicator(Canvas canvas, float scale) {
|
|
RectF currRect = getActiveRect();
|
|
|
|
// saves the canvas so we can later restore it to its original scale
|
|
canvas.save();
|
|
|
|
// Moves the canvas to start at the top left corner of the page indicator
|
|
canvas.translate(currRect.left, currRect.top);
|
|
|
|
// Scales the canvas in place to animate the indicator on entry
|
|
canvas.scale(scale, scale, mPageIndicatorRadius, mPageIndicatorRadius);
|
|
|
|
int scrollPerPage = getScrollPerPage();
|
|
// This IF is to avoid division by 0
|
|
if (scrollPerPage != 0) {
|
|
int delta = mCurrentScroll % scrollPerPage;
|
|
canvas.rotate((INDICATOR_ROTATION * delta) / scrollPerPage,
|
|
mPageIndicatorRadius, mPageIndicatorRadius);
|
|
}
|
|
|
|
mPageIndicatorDrawable.draw(canvas);
|
|
canvas.restore();
|
|
}
|
|
|
|
/**
|
|
* Returns the radius of the circle based on how close the page indicator is to it
|
|
*
|
|
* @param dotPositionX is the position the dot is located at in the x-axis
|
|
*/
|
|
private float getRadius(float dotPositionX) {
|
|
|
|
float startXIndicator =
|
|
((getWidth() - (mNumPages * mCircleGap) + mDotRadius) / 2) - getOffset();
|
|
float indicatorPosition = startXIndicator + getIndicatorScrollDistance()
|
|
+ mPageIndicatorRadius;
|
|
|
|
// If the indicator gets close enough to a dot then we change the radius
|
|
// of the dot based on how close the indicator is to it.
|
|
float dotDistance = Math.abs(indicatorPosition - dotPositionX);
|
|
if (dotDistance <= mCircleGap) {
|
|
return Utilities.mapToRange(dotDistance, 0, mCircleGap, 0f, mDotRadius,
|
|
Interpolators.LINEAR);
|
|
}
|
|
return mDotRadius;
|
|
}
|
|
|
|
private RectF getActiveRect() {
|
|
float startCircle = (int) mCurrentPosition;
|
|
float delta = mCurrentPosition - startCircle;
|
|
float diameter = 2 * mDotRadius;
|
|
float startX;
|
|
|
|
if (SHOW_DELIGHTFUL_PAGINATION.get()) {
|
|
startX = ((getWidth() - (mNumPages * mCircleGap) + mDotRadius) / 2) - getOffset();
|
|
sTempRect.top = (getHeight() - mPageIndicatorSize) * 0.5f;
|
|
sTempRect.bottom = (getHeight() + mPageIndicatorSize) * 0.5f;
|
|
sTempRect.left = startX + getIndicatorScrollDistance();
|
|
sTempRect.right = sTempRect.left + mPageIndicatorSize;
|
|
} else {
|
|
startX = ((getWidth() - (mNumPages * mCircleGap) + mDotRadius) / 2);
|
|
sTempRect.top = (getHeight() * 0.5f) - mDotRadius;
|
|
sTempRect.bottom = (getHeight() * 0.5f) + mDotRadius;
|
|
sTempRect.left = startX + (startCircle * mCircleGap);
|
|
sTempRect.right = sTempRect.left + diameter;
|
|
|
|
if (delta < SHIFT_PER_ANIMATION) {
|
|
// dot is capturing the right circle.
|
|
sTempRect.right += delta * mCircleGap * 2;
|
|
} else {
|
|
// Dot is leaving the left circle.
|
|
sTempRect.right += mCircleGap;
|
|
|
|
delta -= SHIFT_PER_ANIMATION;
|
|
sTempRect.left += delta * mCircleGap * 2;
|
|
}
|
|
}
|
|
|
|
if (mIsRtl) {
|
|
float rectWidth = sTempRect.width();
|
|
sTempRect.right = getWidth() - sTempRect.left;
|
|
sTempRect.left = sTempRect.right - rectWidth;
|
|
}
|
|
|
|
return sTempRect;
|
|
}
|
|
|
|
/**
|
|
* The offset between the radius of the dot and the midpoint of the indicator so that
|
|
* the indicator is centered in with the indicator circles
|
|
*/
|
|
private float getOffset() {
|
|
return mPageIndicatorRadius - mDotRadius;
|
|
}
|
|
|
|
/**
|
|
* Returns an int that is the amount we need to scroll per page
|
|
*/
|
|
private int getScrollPerPage() {
|
|
return mNumPages > 1 ? mTotalScroll / (mNumPages - 1) : 0;
|
|
}
|
|
|
|
/**
|
|
* The current scroll adjusted for the distance the indicator needs to travel on the screen
|
|
*/
|
|
private float getIndicatorScrollDistance() {
|
|
int scrollPerPage = getScrollPerPage();
|
|
return scrollPerPage != 0 ? ((float) mCurrentScroll / scrollPerPage) * mCircleGap : 0;
|
|
}
|
|
|
|
private class MyOutlineProver extends ViewOutlineProvider {
|
|
|
|
@Override
|
|
public void getOutline(View view, Outline outline) {
|
|
if (mEntryAnimationRadiusFactors == null) {
|
|
RectF activeRect = getActiveRect();
|
|
outline.setRoundRect(
|
|
(int) activeRect.left,
|
|
(int) activeRect.top,
|
|
(int) activeRect.right,
|
|
(int) activeRect.bottom,
|
|
mDotRadius
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Listener for keep running the animation until the final state is reached.
|
|
*/
|
|
private class AnimationCycleListener extends AnimatorListenerAdapter {
|
|
|
|
private boolean mCancelled = false;
|
|
|
|
@Override
|
|
public void onAnimationCancel(Animator animation) {
|
|
mCancelled = true;
|
|
}
|
|
|
|
@Override
|
|
public void onAnimationEnd(Animator animation) {
|
|
if (!mCancelled) {
|
|
mAnimator = null;
|
|
animateToPosition(mFinalPosition);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* We need to override setInsets to prevent InsettableFrameLayout from applying different
|
|
* margins on the pagination.
|
|
*/
|
|
@Override
|
|
public void setInsets(Rect insets) {
|
|
}
|
|
}
|