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 相关文章推荐
Java如何实现树的同构?
Jun 22 Java/Android
总结一下关于在Java8中使用stream流踩过的一些坑
Jun 24 Java/Android
SpringBoot工程下使用OpenFeign的坑及解决
Jul 02 Java/Android
使用logback实现按自己的需求打印日志到自定义的文件里
Aug 30 Java/Android
Spring Security中用JWT退出登录时遇到的坑
Oct 16 Java/Android
springboot中的pom文件 project报错问题
Jan 18 Java/Android
关于ObjectUtils.isEmpty() 和 null 的区别
Feb 28 Java/Android
spring cloud eureka 服务启动失败的原因分析及解决方法
Mar 17 Java/Android
详解Flutter和Dart取消Future的三种方法
Apr 07 Java/Android
Android存储中最基本的文件存储方式
Apr 30 Java/Android
Java处理延时任务的常用几种解决方案
Jun 01 Java/Android
Spring Boot优化后启动速度快到飞起技巧示例
Jul 23 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
php 引用(&)详解
2009/11/20 PHP
PHP通过iconv将字符串从GBK转换为UTF8字符集
2011/07/18 PHP
CodeIgniter钩子用法实例详解
2016/01/20 PHP
在Laravel 的 Blade 模版中实现定义变量
2019/10/14 PHP
javascript两段代码,两个小技巧
2010/02/04 Javascript
给html超链接设置事件不使用href来完成跳
2014/04/20 Javascript
jquery实现翻动fadeIn显示的方法
2015/03/05 Javascript
jQuery图片特效插件Revealing实现拉伸放大
2015/04/22 Javascript
angular.js指令中的controller、compile与link函数的不同之处
2017/05/10 Javascript
vue init webpack myproject构建项目 ip不能访问的解决方法
2018/03/20 Javascript
vue 详情跳转至列表页实现列表页缓存
2019/03/27 Javascript
js实现随机8位验证码
2020/07/24 Javascript
js+html实现点名系统功能
2019/11/05 Javascript
jQuery实现简单评论区功能
2020/10/26 jQuery
微信小程序实现购物车小功能
2020/12/30 Javascript
初步理解Python进程的信号通讯
2015/04/09 Python
Python实现可获取网易页面所有文本信息的网易网络爬虫功能示例
2018/01/15 Python
python如何去除字符串中不想要的字符
2020/07/05 Python
Tensorflow中的placeholder和feed_dict的使用
2018/07/09 Python
Python实现的IP端口扫描工具类示例
2019/02/15 Python
python3.6下Numpy库下载与安装图文教程
2019/04/02 Python
Python 通过打码平台实现验证码的实现
2019/05/13 Python
Django中信号signals的简单使用方法
2019/07/04 Python
Windows10+anacond+GPU+pytorch安装详细过程
2020/03/24 Python
如何用Java判断一个文件或目录是否存在
2012/11/19 面试题
自我鉴定的范文
2013/10/03 职场文书
计算机应用与科学个人的自我评价
2013/11/15 职场文书
化学专业大学生职业生涯规划范文
2014/09/13 职场文书
分居协议书范本(律师见证版)
2014/11/26 职场文书
情人节活动总结范文
2015/02/05 职场文书
教师节校长致辞
2015/07/31 职场文书
教师教育教学随笔
2015/08/15 职场文书
vue-cropper插件实现图片截取上传组件封装
2021/05/27 Vue.js
python之PySide2安装使用及QT Designer UI设计案例教程
2021/07/26 Python
漫画「狩龙人拉格纳」公开TV动画预告图
2022/03/22 日漫
TV动画《八十龟酱观察日记》第四季宣传PV公布
2022/04/06 日漫