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 相关文章推荐
ConstraintValidator类如何实现自定义注解校验前端传参
Jun 18 Java/Android
springboot利用redis、Redisson处理并发问题的操作
Jun 18 Java/Android
Java框架入门之简单介绍SpringBoot框架
Jun 18 Java/Android
解决ObjectMapper.convertValue() 遇到的一些问题
Jun 30 Java/Android
解决MultipartFile.transferTo(dest) 报FileNotFoundExcep的问题
Jul 01 Java/Android
java基础——多线程
Jul 03 Java/Android
MyBatis自定义SQL拦截器示例详解
Oct 24 Java/Android
Java中try catch处理异常示例
Dec 06 Java/Android
Android 界面一键变灰 深色主题工具类
Apr 28 Java/Android
Android Studio实现带三角函数对数运算功能的高级计算器
May 20 Java/Android
Java异常体系非正常停止和分类
Jun 14 Java/Android
Android基础入门之dataBinding的简单使用教程
Jun 21 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
Can't create/write to file 'C:\WINDOWS\TEMP\...MYSQL报错解决方法
2011/06/30 PHP
PHP实现生成唯一会员卡号
2015/08/24 PHP
php+jQuery+Ajax实现点赞效果的方法(附源码下载)
2020/07/21 PHP
ThinkPHP设置禁止百度等搜索引擎转码(简单实用)
2016/02/15 PHP
PHP实现通过CURL上传文件功能示例
2018/05/30 PHP
PHP中Session ID的实现原理实例分析
2019/08/17 PHP
JavaScript方法和技巧大全
2006/12/27 Javascript
js动态添加onload、onresize、onscroll事件(另类方法)
2012/12/26 Javascript
jQuery()方法的第二个参数详解
2015/04/29 Javascript
js实现延迟加载的方法
2015/06/24 Javascript
基于jquery步骤进度条源码分享
2015/11/12 Javascript
Javascript类型系统之undefined和null浅析
2016/07/13 Javascript
AngularJS ng-template寄宿方式用法分析
2016/11/07 Javascript
jQuery基于xml格式数据实现模糊查询及分页功能的方法
2016/12/25 Javascript
BootStrap select2 动态改变值的方法
2017/02/10 Javascript
JS逻辑运算符短路操作实例分析
2018/07/09 Javascript
Vue+ElementUI table实现表格分页
2019/12/14 Javascript
JavaScript JSON使用原理及注意事项
2020/07/30 Javascript
python解析发往本机的数据包示例 (解析数据包)
2014/01/16 Python
Python捕捉和模拟鼠标事件的方法
2015/06/03 Python
python脚本替换指定行实现步骤
2017/07/11 Python
python 筛选数据集中列中value长度大于20的数据集方法
2018/06/14 Python
解决Python3 被PHP程序调用执行返回乱码的问题
2019/02/16 Python
Python用类实现扑克牌发牌的示例代码
2020/06/01 Python
python转化excel数字日期为标准日期操作
2020/07/14 Python
详解如何用HTML5 Canvas API控制图片的缩放变换
2016/03/22 HTML / CSS
HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题
2021/01/19 HTML / CSS
Overload和Override的区别
2012/09/02 面试题
电信专业应届生自荐信
2013/09/28 职场文书
教师自我鉴定范文
2013/11/10 职场文书
上班离岗检讨书
2014/09/10 职场文书
厉行节约工作总结
2015/08/12 职场文书
小学学习委员竞选稿
2015/11/20 职场文书
2016年教代会开幕词
2016/03/04 职场文书
如何设计高效合理的MySQL查询语句
2021/05/26 MySQL
使用CSS自定义属性实现骨架屏效果
2022/06/21 HTML / CSS