Android自定义ScrollView实现阻尼回弹
作者:YX_BB 发布时间:2021-11-23 07:20:19
标签:Android,ScrollView,阻尼回弹
Android开发中,当一个页面存放的控件超出屏幕时,通常需要使用ScrollView来包裹布局。这样用户可以通过手指的滑动来查看超出屏幕的部分。然而当ScrollView滑动到边界时,继续滑动只会显示一个阴影效果。iOS自带的控件却可以实现边界的阻尼回弹效果,这种阻尼回弹效果会让用户有更好的使用体验。这里给出一个Android上的实现方案
解决思路:
ScrollView使用时要求内部有且仅一个子View。当ScrollView滑动到边界时,让子View在ScrollView中随着手指按一定的规则进行平移,模拟出拉伸效果。当手指松开时,再让子View恢复拉伸前的位置,模拟出回弹效果。
完整的代码如下,详细的原理见注释即可
public class StretchScrollView extends NestedScrollView {
// 子View
private View innerView;
// 上次手势事件的y坐标
private float mLastY;
// 记录子View的正常位置
private Rect normal = new Rect();
public StretchScrollView(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
protected void onFinishInflate() {
initView();
super.onFinishInflate();
}
/**
* 获取ScrollView的子布局
*/
private void initView() {
// 去除原本ScrollView滚动到边界时的阴影效果
setOverScrollMode(OVER_SCROLL_NEVER);
if (getChildAt(0) != null) {
innerView = getChildAt(0);
}
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
switch (ev.getAction()) {
case MotionEvent.ACTION_UP:
// 手指松开恢复
if (!normal.isEmpty()) {
planAnimation();
normal.setEmpty();
mLastY = 0;
}
break;
case MotionEvent.ACTION_MOVE:
float currentY = ev.getY();
// 滑动距离
int distanceY = (int) (mLastY - currentY);
// 处理Y轴的滚动事件,当滚动到最上或者最下时需要移动布局
// 手指刚触及屏幕时,也会触发此事件,此时mLastY的值还是0,会立即触发一个比较大的移动。这里过滤掉这种情况
if (isNeedTranslate() && mLastY != 0) {
if (normal.isEmpty()) {
// 保存正常的布局位置
normal.set(innerView.getLeft(), innerView.getTop(), innerView.getRight(), innerView.getBottom());
}
// 移动布局, 使distance / 2 防止平移过快
innerView.layout(innerView.getLeft(), innerView.getTop() - distanceY / 2,
innerView.getRight(), innerView.getBottom() - distanceY / 2);
}
mLastY = currentY;
break;
}
return super.onTouchEvent(ev);
}
/**
* 回缩动画
*/
public void planAnimation() {
// 开启移动动画
TranslateAnimation animation = new TranslateAnimation(0, 0, innerView.getTop(), normal.top);
animation.setDuration(200);
innerView.startAnimation(animation);
// 补间动画并不会真正修改innerView的位置,这里需要设置使得innerView回到正常的布局位置
innerView.layout(normal.left, normal.top, normal.right, normal.bottom);
}
/**
* 是否需要Y移动布局
*/
public boolean isNeedTranslate() {
int offset = innerView.getMeasuredHeight() - getHeight();
int scrollY = getScrollY();
// 顶部或者底部
return scrollY == 0 || scrollY == offset;
}
}
来源:https://blog.csdn.net/YX_BB/article/details/104698970


猜你喜欢
- 公钥加密算法,也就是 非对称加密算法,这种算法加密和解密的密码不一样,一个是公钥,另一个是私钥:公钥和私钥成对出现公开的密钥叫公钥,只有自己
- 新建SL4 应用程序,在MainPage下添加代码:<Button x:Name="btnThread1" Cli
- 前言之前介绍了 Animatable 动画以及其 animateTo和 snapTo两个开启动画 api 的使用,实际上 Animatabl
- 通常C#使用基于XML的配置文件,不过如果有需要的话,比如要兼顾较老的系统,可能还是要用到INI文件。但C#本身并不具备读写INI文件的AP
- 本文实例为大家分享了Java实现串口通信的具体代码,供大家参考,具体内容如下1.介绍使用Java实现的串口通信程序,支持十六进制数据的发送与
- 在日常Android手机的使用过程中,根据电话号码获得联系人头像,是经常会碰到的问题。本文即以实例形式讲述了Android根据电话号码获得联
- Spring中添加计时器的时候根据业务需求可能会需要动态处理触发时间;import org.slf4j.Logger; import org
- 源程序揭秘杨辉三角形性质: 每行数字左右对称,由 1 开始逐渐变大,然后变小,回到 1。 第 n 行的数字个数为 n 个。 第 n 行数字和
- 1、什么是线程及线程池线程是操作系统进行时序调度的基本单元。线程池可以理解为一个存在线程的池子,就是一个容器,这个容器只能存在线程。这个容器
- 本文实例讲述了C#实现字符串与图片的Base64编码转换操作。分享给大家供大家参考,具体如下:using System;using Syst
- 方法一:1.在pom.xml文件下添加依赖包<dependency><groupId>com.alibaba<
- 最近在开发一个项目,需要写一个后管系统,Bootstrap是美国Twitter公司的设计师Mark Otto和Jacob Thornton合
- 一、树形结构树是一种非线性的数据结构,它是由n(n>=0)个有限结点组成一个具有层次关系的集合。把它叫做树是因为它看起来像一棵倒挂的树
- 这篇文章主要介绍了Spring boot @RequestBody数据传递过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有
- 委托的Invoke方法用来进行同步调用。同步调用也可以叫阻塞调用,它将阻塞当前线程,然后执行调用,调用完毕后再继续向下进行。同步调用的例子:
- 前面文章已经详细介绍了Android界面的入门技术,相信大家在看完和跟着练习之后,会对于常用的Layout和View都会有一定的了解了,接下
- 本文实例为大家分享了Java实现聊天机器人完善版的具体代码,供大家参考,具体内容如下Client代码:package GUISocket.c
- 编写C#程序的时候,我们都遇到过配置文件,而如今绝大多数的配置文件都是用XML写的。所以在处理的时候就需要操作XML文件。那么C#如何操作X
- 本文以实例形式讲述了C#命令模式的实现方法,分享给大家供大家参考。具体实现方法如下:现假设想让遥控器控制电灯的开关、电视机的开关和切换,该如
- 概述RocketMQ 支持发送延迟消息,但不支持任意时间的延迟消息的设置,仅支持内置预设值的延迟时间间隔的延迟消息;预设值的延迟时间间隔为: