Android自定义ScrollView实现阻尼回弹


Posted in Java/Android onApril 01, 2022

Android开发中,当一个页面存放的控件超出屏幕时,通常需要使用ScrollView来包裹布局。这样用户可以通过手指的滑动来查看超出屏幕的部分。然而当ScrollView滑动到边界时,继续滑动只会显示一个阴影效果。iOS自带的控件却可以实现边界的阻尼回弹效果,这种阻尼回弹效果会让用户有更好的使用体验。这里给出一个Android上的实现方案

解决思路:

ScrollView使用时要求内部有且仅一个子View。当ScrollView滑动到边界时,让子View在ScrollView中随着手指按一定的规则进行平移,模拟出拉伸效果。当手指松开时,再让子View恢复拉伸前的位置,模拟出回弹效果。

Android自定义ScrollView实现阻尼回弹

完整的代码如下,详细的原理见注释即可

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;
    }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Java/Android 相关文章推荐
Feign调用传输文件异常的解决
Jun 24 Java/Android
Jackson 反序列化时实现大小写不敏感设置
Jun 29 Java/Android
分析并发编程之LongAdder原理
Jun 29 Java/Android
logback 实现给变量指定默认值
Aug 30 Java/Android
Spring-cloud Config Server的3种配置方式
Sep 25 Java/Android
Java中Quartz高可用定时任务快速入门
Apr 03 Java/Android
Java Spring Boot 正确读取配置文件中的属性的值
Apr 20 Java/Android
Spring Boot 使用 Spring-Retry 进行重试框架
Apr 24 Java/Android
springboot读取nacos配置文件
May 20 Java/Android
多线程Spring通过@Scheduled实现定时任务
May 25 Java/Android
SpringBoot接入钉钉自定义机器人预警通知
Jul 15 Java/Android
spring 项目实现限流方法示例
Jul 15 Java/Android
Java实战之课程信息管理系统的实现
Android超详细讲解组件ScrollView的使用
Spring Boot DevTools 全局配置学习指南
Spring事务管理下synchronized锁失效问题的解决方法
Mar 31 #Java/Android
Spring依赖注入多种类型数据的示例代码
Mar 31 #Java/Android
springboot layui hutool Excel导入的实现
spring注解 @PropertySource配置数据源全流程
Mar 25 #Java/Android
You might like
一个程序下载的管理程序(四)
2006/10/09 PHP
关于javascript function对象那些迷惑分析
2011/10/24 Javascript
JavaScript中的Math.sin()方法使用详解
2015/06/15 Javascript
每天一篇javascript学习小结(Array数组)
2015/11/11 Javascript
jQuery 获取多选框的值及多选框中文的函数
2016/05/16 Javascript
JS+CSS3模拟溢出滚动效果
2016/08/12 Javascript
JS实现禁止高频率连续点击的方法【基于ES6语法】
2017/04/25 Javascript
vue实现未登录跳转到登录页面的方法
2018/07/17 Javascript
Vue作用域插槽slot-scope实例代码
2018/09/05 Javascript
js replace替换字符串同时替换多个方法
2018/11/27 Javascript
详解jQuery中的getAll()和cleanData()
2019/04/15 jQuery
自定义Vue中的v-module双向绑定的实现
2019/04/17 Javascript
利用Vue实现一个markdown编辑器实例代码
2019/05/19 Javascript
[00:44]华丽开场!DOTA2勇士令状带来全新对阵画面
2019/05/15 DOTA
Python 12306抢火车票脚本
2018/02/07 Python
对python for 文件指定行读写操作详解
2018/12/29 Python
Python字符串匹配之6种方法的使用详解
2019/04/08 Python
使用python list 查找所有匹配元素的位置实例
2019/06/11 Python
Django中使用MySQL5.5的教程
2019/12/18 Python
python实现五子棋程序
2020/04/24 Python
PyTorch中torch.tensor与torch.Tensor的区别详解
2020/05/18 Python
Windows 下更改 jupyterlab 默认启动位置的教程详解
2020/05/18 Python
关于keras中keras.layers.merge的用法说明
2020/05/23 Python
python rsa-oaep加密的示例代码
2020/09/23 Python
Python WebSocket长连接心跳与短连接的示例
2020/11/24 Python
Html5中的桌面通知Notification的实现
2018/09/25 HTML / CSS
班干部演讲稿
2014/04/24 职场文书
2014机关党员干部“正风肃纪”思想汇报
2014/09/15 职场文书
小学生推普周国旗下讲话稿
2014/09/21 职场文书
学生旷课检讨书500字
2014/10/28 职场文书
小学教师教育随笔
2015/08/14 职场文书
2016年学校安全教育月活动总结
2016/04/06 职场文书
六年级作文之家庭作文
2019/12/12 职场文书
浅谈Python协程asyncio
2021/06/20 Python
使用CSS3实现按钮悬停闪烁动态特效代码
2021/08/30 HTML / CSS
LyScript实现绕过反调试保护的示例详解
2022/08/14 Python