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 相关文章推荐
启动Tomcat时出现大量乱码的解决方法
Jun 21 Java/Android
Java中常用解析工具jackson及fastjson的使用
Jun 28 Java/Android
java基础——多线程
Jul 03 Java/Android
springboot中的pom文件 project报错问题
Jan 18 Java/Android
SpringBoot中HttpSessionListener的简单使用方式
Mar 17 Java/Android
SpringBoot2零基础到精通之异常处理与web原生组件注入
Mar 22 Java/Android
Android Flutter实现图片滑动切换效果
Apr 07 Java/Android
Java中生成微信小程序太阳码的实现方案
Jun 01 Java/Android
Java完整实现记事本代码
Jun 16 Java/Android
Android学习之BottomSheetDialog组件的使用
Jun 21 Java/Android
Spring中bean集合注入的方法详解
Jul 07 Java/Android
AndroidStudio图片压缩工具ImgCompressPlugin使用实例
Aug 05 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/02/22 PHP
php面向对象全攻略 (十) final static const关键字的使用
2009/09/30 PHP
PHP多进程之pcntl_fork的实例详解
2017/10/15 PHP
跨浏览器开发经验总结(四) 怎么写入剪贴板
2010/05/13 Javascript
jQuery EasyUI API 中文文档 - Form表单
2011/10/06 Javascript
JS截取字符串常用方法整理及使用示例
2013/10/18 Javascript
append和appendTo的区别以及appendChild用法
2013/12/24 Javascript
javascript中不提供sleep功能如何实现这个功能
2014/05/27 Javascript
jQuery获得子元素个数的方法
2015/04/14 Javascript
javascript生成不重复的随机数
2015/07/17 Javascript
javascript的变量、传值、传址、参数之间关系
2015/07/26 Javascript
不想让浏览器运行javascript脚本的方法
2015/11/20 Javascript
javascript定义类和类的实现实例详解
2015/12/01 Javascript
AngularJS入门教程之AngularJS模型
2016/04/18 Javascript
JS解析url查询参数的简单代码
2017/08/06 Javascript
JS分页的实现(同步与异步)
2017/09/16 Javascript
解析Angular 2+ 样式绑定方式
2018/01/15 Javascript
原生JS实现循环Nodelist Dom列表的4种方式示例
2018/02/11 Javascript
在微信小程序中渲染HTML内容的方法示例
2018/09/28 Javascript
js中arguments对象的深入理解
2019/05/14 Javascript
js实现小时钟效果
2020/03/25 Javascript
react结合bootstrap实现评论功能
2020/05/30 Javascript
Vue axios获取token临时令牌封装案例
2020/09/11 Javascript
python编写暴力破解FTP密码小工具
2014/11/19 Python
Python中的os.path路径模块中的操作方法总结
2016/07/07 Python
Django数据库操作的实例(增删改查)
2017/09/04 Python
python 限制函数调用次数的实例讲解
2018/04/21 Python
HTML5新增元素如何兼容旧浏览器有哪些方法
2014/05/09 HTML / CSS
美国首屈一指的高品质珠宝设计师和零售商:Allurez
2018/01/23 全球购物
Eagle Eyes Optics鹰眼光学:高性能太阳镜
2018/12/07 全球购物
AssertionError 跟一下那个类是 “is – a”的关系
2012/02/21 面试题
高中班长自我鉴定
2013/12/20 职场文书
迎新晚会邀请函
2014/02/01 职场文书
环境保护与污染治理求职信
2014/07/16 职场文书
试用期员工工作自我评价
2014/09/10 职场文书
领导干部群众路线对照检查材料
2014/11/05 职场文书