实现思路
- 控件拉伸回弹,可通过缩放画布来达到,我们只要计算出控件在拉伸时的缩放比例即可
- 缩放比例可通过手指移动距离来计算,当控件滑动到边界时,手指继续滑动,额外滑动距离/控件总高度,即可作为拉伸比例
- 控件状态可分为三类,正常滑动状态,手越界滑动时的拉伸状态,手松开时的回弹状态,第一种状态我们使用RecyclerView默认的滑动处理即可
- 手松开时,我们可以通过一个渐变值动画,来让额外滑动距离逐渐减少到0
- 回弹过程中欧冠,如果手指重新按下,我们则立刻取消渐变值动画,并让控件重新进入到拉伸状态
- 控件滑动的临界值,可以通过computeVerticalScrollRange、computeVerticalScrollExtent、computeVerticalScrollOffset这三个方法来计算
核心代码
package com.android.architecture;
import android.content.Context;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.Animation;
import android.view.animation.Interpolator;
import android.view.animation.Transformation;
import androidx.core.view.MotionEventCompat;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
//实现橡皮筋拉伸回弹效果
//支持多指操控,一只手指滑到屏幕边界时,可以通过其它手指继续滑动
@SuppressWarnings("all")
public class ElasticListView extends RecyclerView {
//正常状态
private static final int STATE_NORMAL = 0;
//沿顶部拉伸
private static final int STATE_STRETCHING_TOP = 1;
//沿底部拉伸
private static final int STATE_STRETCHING_BOTTOM = 2;
//回弹
private static final int STATE_BOUNCING_BACK = 3;
//滑动状态
int state = STATE_NORMAL;
//当前起作用的手指
int activePointerId = -1;
//记录上次手指位置
float preY;
//记录手指松开时的拉伸距离
float maxOffset;
//记录当前的拉伸距离,用于回弹动画
float nowOffset;
//回弹动画
Animation bounceBackAnimation;
Interpolator bounceBackInterpolator;
public ElasticListView(Context context) {
this(context, null);
}
public ElasticListView(Context context, AttributeSet attributeSet) {
super(context, attributeSet);
init(context, attributeSet);
}
protected void init(Context context, AttributeSet attributeSet) {
//设置竖向布局
LinearLayoutManager layoutManager = new LinearLayoutManager(context);
layoutManager.setOrientation(RecyclerView.VERTICAL);
setLayoutManager(layoutManager);
//启用默认的越界滚动效果,即滚动到边界时,继续滚动会产生一个水纹效果
//手动滚动到边界时,使用橡皮筋拉伸效果,通过惯性自动滚动到边界时,仍然使用默认的水纹效果
setOverScrollMode(View.OVER_SCROLL_ALWAYS);
//创建回弹动画
bounceBackAnimation = new Animation() {
@Override
protected void applyTransformation(float progress, Transformation transformation) {
nowOffset = maxOffset * progress;
if (hasEnded()) {
nowOffset = 0;
state = STATE_NORMAL;
}
invalidate();
}
};
//设置动画插值器
bounceBackInterpolator = new Interpolator() {
@Override
public float getInterpolation(float timePercent) {
float progress = (float) Math.cos(Math.PI * timePercent / 2);
return progress;
}
};
bounceBackAnimation.setInterpolator(bounceBackInterpolator);
bounceBackAnimation.setDuration(300);
}
@Override
public void draw(Canvas canvas) {
//普通状态下,正常绘制
if (state == STATE_NORMAL) {
super.draw(canvas);
return;
}
//拉伸或回弹状态下,拉伸画布
int saveCount = canvas.save();
int height = getHeight();
float scale = 1 + Math.abs(nowOffset) / height * 0.3F;
canvas.scale(1, scale, 0, nowOffset >= 0 ? 0 : height);
super.draw(canvas);
canvas.restoreToCount(saveCount);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent e) {
if (onInterceptTouchEventInternal(e))
return true;
return super.onInterceptTouchEvent(e);
}
protected boolean onInterceptTouchEventInternal(MotionEvent e) {
int action = MotionEventCompat.getActionMasked(e);
switch (action) {
case MotionEvent.ACTION_DOWN: {
preY = e.getY();
activePointerId = e.getPointerId(0);
//暂停回弹状态,或恢复到正常状态
if (state == STATE_BOUNCING_BACK) {
//回弹动画已结束,则恢复到正常状态
if (nowOffset == 0) {
state = STATE_NORMAL;
invalidate();
break;
}
//回弹动画尚未结束,切换到拉伸状态
//如果此时立刻将手松开,会开始一个新的回弹动画,从上次位置继续回弹
clearAnimation();
state = nowOffset > 0 ? STATE_STRETCHING_TOP : STATE_STRETCHING_BOTTOM;
invalidate();
}
break;
}
}
//如果处于拉伸状态,则自己处理该事件,不分发给子节点
//如果处于普通状态,则调用基类方法去处理
boolean stretching = isStretching();
return stretching;
}
@Override
public boolean onTouchEvent(MotionEvent e) {
if (onTouchEventInternal(e))
return true;
return super.onTouchEvent(e);
}
protected boolean onTouchEventInternal(MotionEvent e) {
int action = MotionEventCompat.getActionMasked(e);
switch (action) {
case MotionEvent.ACTION_MOVE: {
int pointerIndex = e.findPointerIndex(activePointerId);
float nowY = e.getY(pointerIndex);
float deltaY = nowY - preY;
preY = nowY;
//非拉伸状态下,需要处理切换到拉伸状态的case
if (!isStretching()) {
//判断控件能不能滑动,有没有到达边界
boolean canScrollUp = false;
boolean canScrollDown = false;
//computeVerticalScrollRange返回控件全部内容的高度
//computeVerticalScrollExtent返回控件在屏幕上可展示区域的高度
//computeVerticalScrollOffset返回控件已滑过的距离
int range = computeVerticalScrollRange() - computeVerticalScrollExtent();
if (range == 0) {
//内容较少,已经全部展示,不需要滑动
canScrollUp = false;
canScrollDown = false;
} else {
int offset = computeVerticalScrollOffset();
canScrollUp = offset > 0;
canScrollDown = offset < range;
}
//未到达边界,上下都可以滑动
if (canScrollUp && canScrollDown)
break;
//达到边界时,再继续滑动,即进入越界滚动状态
if (!canScrollUp && deltaY > 0)
state = STATE_STRETCHING_TOP;
if (!canScrollDown && deltaY < 0)
state = STATE_STRETCHING_BOTTOM;
}
//拉伸状态下,需要处理重绘控件,和切换到非拉伸状态的case
if (isStretching()) {
nowOffset = nowOffset + deltaY;
if ((state == STATE_STRETCHING_TOP && nowOffset < 0) || (state == STATE_STRETCHING_BOTTOM && nowOffset > 0)) {
state = STATE_NORMAL;
nowOffset = 0;
}
invalidate();
}
break;
}
case MotionEventCompat.ACTION_POINTER_DOWN: {
int index = MotionEventCompat.getActionIndex(e);
preY = e.getY(index);
activePointerId = e.getPointerId(index);
break;
}
case MotionEventCompat.ACTION_POINTER_UP: {
onPointerUp(e);
int pointerIndex = e.findPointerIndex(activePointerId);
preY = e.getY(pointerIndex);
break;
}
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL: {
if (nowOffset != 0) {
maxOffset = nowOffset;
startAnimation(bounceBackAnimation);
state = STATE_BOUNCING_BACK;
}
}
}
//如果处于拉伸状态,则自己处理该事件,不分发给子节点
//如果处于普通状态,则调用基类方法去处理
boolean stretching = isStretching();
return stretching;
}
//多指触控时,有一只手指松开
protected void onPointerUp(MotionEvent e) {
int pointerIndex = e.getActionIndex();
int pointerId = e.getPointerId(pointerIndex);
if (pointerId == activePointerId) {
//选取一个新的有效手指
int newPointerIndex = pointerIndex == 0 ? 1 : 0;
activePointerId = e.getPointerId(newPointerIndex);
}
}
//判断控件是否处于拉伸状态
protected boolean isStretching() {
return state == STATE_STRETCHING_TOP || state == STATE_STRETCHING_BOTTOM;
}
}