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同步系列之synchronized解析
Jun 28 Java/Android
Java基础-封装和继承
Jul 02 Java/Android
Java使用jmeter进行压力测试
Jul 09 Java/Android
Java SSM配置文件案例详解
Aug 30 Java/Android
springboot 多数据源配置不生效遇到的坑及解决
Nov 17 Java/Android
Java 超详细讲解hashCode方法
Apr 07 Java/Android
SpringBoot 集成短信和邮件 以阿里云短信服务为例
Apr 22 Java/Android
Java由浅入深通关抽象类与接口(上篇)
Apr 26 Java/Android
Java8 CompletableFuture 异步回调
Apr 28 Java/Android
Java+swing实现抖音上的表白程序详解
Jun 25 Java/Android
Java实现超大Excel文件解析(XSSF,SXSSF,easyExcel)
Jul 15 Java/Android
JDK8中String的intern()方法实例详细解读
Sep 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 构造函数construct的前下划线是双的_
2009/12/08 PHP
ECMall支持SSL连接邮件服务器的配置方法详解
2014/05/19 PHP
php实现四舍五入的方法小结
2015/03/03 PHP
PHP实现指定字段的多维数组排序函数分享
2015/03/09 PHP
php保存信息到当前Session的方法
2015/03/16 PHP
php实现的mongodb操作类
2015/05/28 PHP
thinkPHP删除前弹出确认框的简单实现方法
2016/05/16 PHP
php使用CURL模拟GET与POST向微信接口提交及获取数据的方法
2016/09/23 PHP
php数据序列化测试实例详解
2017/08/12 PHP
浅谈PHP进程管理
2019/03/08 PHP
全面理解面向对象的 JavaScript(来自ibm)
2013/11/10 Javascript
jquery使用append(content)方法注意事项分享
2014/01/06 Javascript
手机平板等移动端适配跳转URL的js代码
2014/01/25 Javascript
js 获取页面高度和宽度兼容 ie firefox chrome等
2014/05/14 Javascript
jQuery实现强制cookie过期方法汇总
2015/05/22 Javascript
require.js+vue开发微信上传图片组件
2016/10/27 Javascript
Nodejs下用submit提交表单提示cannot post错误的解决方法
2016/11/21 NodeJs
AngularJs中Bootstrap3 datetimepicker使用实例
2016/12/13 Javascript
React-Router如何进行页面权限管理的方法
2017/12/06 Javascript
vue router 跳转后回到顶部的实例
2018/08/31 Javascript
讲解Python中的标识运算符
2015/05/14 Python
Python查询IP地址归属完整代码
2017/06/21 Python
深入理解Python分布式爬虫原理
2017/11/23 Python
浅析Git版本控制器使用
2017/12/10 Python
Python+OpenCV+pyQt5录制双目摄像头视频的实例
2019/06/28 Python
python3应用windows api对后台程序窗口及桌面截图并保存的方法
2019/08/27 Python
Python3运算符常见用法分析
2020/02/14 Python
css3的transition效果和transfor效果示例介绍
2013/10/30 HTML / CSS
css3加js做一个简单的3D行星运转效果实例代码
2017/01/18 HTML / CSS
HTML5中的websocket实现直播功能
2018/05/21 HTML / CSS
在html页面中取得session中的值的方法
2020/08/11 HTML / CSS
自我评价怎么写正确呢?
2013/12/02 职场文书
烹饪自我鉴定
2014/03/01 职场文书
追悼会答谢词
2015/01/05 职场文书
2015驻村干部工作总结
2015/04/07 职场文书
2019年中,最受大众欢迎的6本新书
2019/08/07 职场文书