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 11 Java/Android
Java内存模型之happens-before概念详解
Jun 13 Java/Android
ConstraintValidator类如何实现自定义注解校验前端传参
Jun 18 Java/Android
Java并发编程之详解CyclicBarrier线程同步
Jun 23 Java/Android
Java 中的 Unsafe 魔法类的作用大全
Jun 26 Java/Android
SpringRetry重试框架的具体使用
Jul 25 Java/Android
mybatis中注解与xml配置的对应关系和对比分析
Aug 04 Java/Android
Java中CyclicBarrier和CountDownLatch的用法与区别
Aug 23 Java/Android
使用Spring处理x-www-form-urlencoded方式
Nov 02 Java/Android
OpenCV实现反阈值二值化
Nov 17 Java/Android
Java练习之潜艇小游戏的实现
Mar 16 Java/Android
Java基于Dijkstra算法实现校园导游程序
Mar 17 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
德劲1103二次变频版的打磨
2021/03/02 无线电
完美解决thinkphp唯一索引重复时出错的问题
2017/03/31 PHP
PHP实现cookie跨域session共享的方法分析
2019/08/23 PHP
js 强制弹出窗口代码研究-又一款代码
2010/03/20 Javascript
一个基于jquery的文本框记数器
2012/09/19 Javascript
JavaScript中“基本类型”之争小结
2013/01/03 Javascript
如何使用jQuery来处理图片坏链具体实现步骤
2013/05/02 Javascript
jquery实现的一个导航滚动效果具体代码
2013/05/27 Javascript
js操作输入框中选择内容兼容IE及其他主流浏览器
2014/04/22 Javascript
一个JavaScript操作元素定位元素的实例
2014/10/29 Javascript
使用javascript将时间转换成今天,昨天,前天等格式
2015/06/25 Javascript
JS+CSS实现经典的左侧竖向滑动菜单效果
2015/09/23 Javascript
原生js和jquery分别实现横向导航菜单效果
2016/05/13 Javascript
Bootstrap表单使用方法详解
2017/02/17 Javascript
jQuery实现文章图片弹出放大效果
2017/04/06 jQuery
Node.JS使用Sequelize操作MySQL的示例代码
2017/10/09 Javascript
elementUI 设置input的只读或禁用的方法
2018/10/30 Javascript
关于ES6尾调用优化的使用
2020/09/11 Javascript
Python中文件操作简明介绍
2015/04/13 Python
python之文件的读写和文件目录以及文件夹的操作实现代码
2016/08/28 Python
python写一个md5解密器示例
2018/02/23 Python
pandas数据框,统计某列数据对应的个数方法
2018/04/11 Python
Python实现时钟显示效果思路详解
2018/04/11 Python
python tornado微信开发入门代码
2018/08/24 Python
python利用tkinter实现屏保
2019/07/30 Python
tensorflow将图片保存为tfrecord和tfrecord的读取方式
2020/02/17 Python
Selenium Webdriver元素定位的八种常用方式(小结)
2021/01/13 Python
HTML5实现锚点时请使用id取代name
2013/09/06 HTML / CSS
html5使用canvas画三角形
2014/12/15 HTML / CSS
找到不普通的东西:Bonanza
2016/10/20 全球购物
微软澳洲官方网站:Microsoft Australia
2017/01/10 全球购物
KTV的创业计划书范文
2014/02/02 职场文书
政府门卫岗位职责
2014/04/29 职场文书
2015年求职自荐信范文
2015/03/04 职场文书
自我推荐信怎么写
2015/03/24 职场文书
浅谈MySQL 亿级数据分页的优化
2021/06/15 MySQL