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 相关文章推荐
JVM入门之类加载与字节码技术(类加载与类的加载器)
Jun 15 Java/Android
浅谈自定义校验注解ConstraintValidator
Jun 30 Java/Android
实体类或对象序列化时,忽略为空属性的操作
Jun 30 Java/Android
mybatis 解决从列名到属性名的自动映射失败问题
Jun 30 Java/Android
Spring Boot 整合 Apache Dubbo的示例代码
Jul 04 Java/Android
使用springMVC所需要的pom配置
Sep 15 Java/Android
springboot 自定义配置 解决Boolean属性不生效
Mar 18 Java/Android
详解Flutter和Dart取消Future的三种方法
Apr 07 Java/Android
Java工作中实用的代码优化技巧分享
Apr 21 Java/Android
Flutter Navigator 实现路由传递参数
Apr 22 Java/Android
带你了解Java中的ForkJoin
Apr 28 Java/Android
Qt数据库应用之实现图片转pdf
Jun 01 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计算加权平均数的方法
2015/07/16 PHP
jQuery EasyUI API 中文文档 - Panel面板
2011/09/30 Javascript
关于div自适应高度/左右高度自适应一致的js代码
2013/03/22 Javascript
js获取IP和PcName(IE)在vs中可用
2013/08/02 Javascript
js阻止默认事件与js阻止事件冒泡示例分享 js阻止冒泡事件
2014/01/27 Javascript
IE浏览器不支持getElementsByClassName的解决方法
2014/08/27 Javascript
node.js中的buffer.slice方法使用说明
2014/12/10 Javascript
js实现宇宙星空背景效果的方法
2015/03/03 Javascript
JavaScript 封装一个tab效果源码分享
2015/09/15 Javascript
js中flexible.js实现淘宝弹性布局方案
2020/06/23 Javascript
文件上传,iframe跨域数据提交的实现
2016/11/18 Javascript
原生JS+Canvas实现五子棋游戏实例
2017/06/19 Javascript
JavaScript实现封闭区域布尔运算的示例代码
2018/06/25 Javascript
如何在 JavaScript 中更好地利用数组
2018/09/27 Javascript
AutoJs实现刷宝短视频的思路详解
2020/05/22 Javascript
[01:28]一分钟告诉你DOTA2 TI9不朽宝藏Ⅱ中有什么!
2019/07/09 DOTA
Python 闭包的使用方法
2017/09/07 Python
利用python编写一个图片主色转换的脚本
2017/12/07 Python
Python爬取附近餐馆信息代码示例
2017/12/09 Python
Keras框架中的epoch、bacth、batch size、iteration使用介绍
2020/06/10 Python
使用html5 canvas 画时钟代码实例分享
2015/11/11 HTML / CSS
Bally美国官网:经典瑞士鞋履、手袋及配饰奢侈品牌
2018/05/18 全球购物
怎样在程序里获得一个空指针
2015/01/24 面试题
管理站站长岗位职责
2013/11/27 职场文书
大学生学业生涯规划
2014/01/05 职场文书
新学期校长寄语
2014/01/18 职场文书
《雨霖铃》听课反思
2014/02/13 职场文书
逃课上网检讨书
2014/02/20 职场文书
《美丽的公鸡》教学反思
2014/02/25 职场文书
群众路线学习心得体会范文
2014/11/05 职场文书
作文评语集锦
2014/12/25 职场文书
学雷锋倡议书
2015/01/19 职场文书
台风停课通知
2015/04/24 职场文书
2015年前台接待工作总结
2015/05/04 职场文书
fastdfs+nginx集群搭建的实现
2021/03/31 Servers
Java基础之this关键字的使用
2021/06/30 Java/Android