android 实现拉出回弹效果通过自定义ListView重写overScrollBy()
来源:互联网 发布:js复选框取消选中事件 编辑:程序博客网 时间:2024/04/30 10:45
日前google上搜索“android overscroll”,对此效果的介绍很多,但关于其具体使用方式和实现,则很少涉及,偶有提及,也经常答非所问或似是而非,反而误导了别人。于是我查阅了android相关源码,并做了一些测试,在此讲讲我的理解。
首先是overscroll功能本身,在最顶层的View类提供了支持,可通过setOverScrollMode函数控制其出现条件。但其实View中并没有实现overscroll功能,它仅仅提供了一个辅助函数overScrollBy,该函数根据overScrollMode和内容是否需要滚动控制最大滚动范围,最后将计算结果传给onOverScrolled实现具体的overscroll功能,但此函数在View类中是全空的。
运行效果如下图:
首先是overscroll功能本身,在最顶层的View类提供了支持,可通过setOverScrollMode函数控制其出现条件。但其实View中并没有实现overscroll功能,它仅仅提供了一个辅助函数overScrollBy,该函数根据overScrollMode和内容是否需要滚动控制最大滚动范围,最后将计算结果传给onOverScrolled实现具体的overscroll功能,但此函数在View类中是全空的。
overscroll功能真正的实现分别在ScrollView、AbsListView、HorizontalScrollView和WebView中各有一份,代码基本一样。以ScrollView为例,它在处理笔点移动消息时调用overScrollBy来滚动视图,然后重载了overScrollBy函数来实现具体功能,其位置计算通过OverScroller类实现。OverScroller作为一个计算引擎,应该是一个独立的模块,具体滚动效果和范围都不可能通过它来设置,我觉得没有必要细看。但滚动位置最终是它给出的,那相关数据肯定要传递给它,回头看overScrollBy函数,它有两个控制overScroll出界范围的参数,几个实现里面都是取自ViewConfiguration.getScaledOverscrollDistance,而这个参数的值在我的源码中都是0,而且我没找到任何可以影响其结果的设置。
代码实现
1. 在View中增加了overSrollBy方法,用于记录x, y 轴上滚动。
2. 在AbsListView的onTouchEvent中判断是否到达边界(顶部 或 底部) ,然后调用view.overScrollBy ,传入 mScrollY等参数
3. overScrollBy 最终赋值给View的mScrollX, mScrollY 两个变量
4. 在AbsListView中调用完overScrollBy之后,调用invalidate重绘
自定义ListView.
package com.alex.diylistview;import android.content.Context;import android.util.AttributeSet;import android.util.DisplayMetrics;import android.widget.ListView;public class BounceListView extends ListView {private static final int MAX_Y_OVERSCROLL_DISTANCE = 200;private Context mContext;private int mMaxYOverscrollDistance;public BounceListView(Context context) {super(context);mContext = context;initBounceListView();}public BounceListView(Context context, AttributeSet attrs) {super(context, attrs);mContext = context;initBounceListView();}public BounceListView(Context context, AttributeSet attrs, int defStyle) {super(context, attrs, defStyle);mContext = context;initBounceListView();}private void initBounceListView() {// get the density of the screen and do some maths with it on the max// overscroll distance// variable so that you get similar behaviors no matter what the screen// sizefinal DisplayMetrics metrics = mContext.getResources().getDisplayMetrics();final float density = metrics.density;mMaxYOverscrollDistance = (int) (density * MAX_Y_OVERSCROLL_DISTANCE);}@Overrideprotected boolean overScrollBy(int deltaX, int deltaY, int scrollX,int scrollY, int scrollRangeX, int scrollRangeY,int maxOverScrollX, int maxOverScrollY, boolean isTouchEvent) {// This is where the magic happens, we have replaced the incoming// maxOverScrollY with our own custom variable mMaxYOverscrollDistance;return super.overScrollBy(deltaX, deltaY, scrollX, scrollY,scrollRangeX, scrollRangeY, maxOverScrollX,mMaxYOverscrollDistance, isTouchEvent);}}
package com.alex.diylistview;import android.app.Activity;import android.os.Bundle;import android.view.ViewGroup.LayoutParams;import android.widget.ArrayAdapter;import android.widget.LinearLayout;public class MainActivity extends Activity {@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);LinearLayout linearLayout = new LinearLayout(this);linearLayout.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));setContentView(linearLayout);BounceListView bounceListView = new BounceListView(this);String[] data = new String[30];for (int i = 0; i < data.length; i++) {data[i] = "回弹效果 " + i;}ArrayAdapter<String> arrayAdapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1, data);bounceListView.setAdapter(arrayAdapter);linearLayout.addView(bounceListView);}}
运行效果如下图:
1 0
- android 实现拉出回弹效果通过自定义ListView重写overScrollBy()
- Android使用最简单的方式实现ListView 拉出回弹效果,阻尼效果(一)
- 自定义控件:overScrollBy+动画实现 ListView 的头部回弹抖动
- ListView 拉出回弹效果 overscroll,阻尼效果
- Android ListView滑动回弹——overScrollBy
- listView 的回弹overScrollBy
- 通过overScrollBy实现下拉视差特效(阻尼效果)
- Android实现ListView阻尼式(下拉回弹)效果
- 【Android实战】ListView的回弹效果实现
- Android:自定义ImageView实现缩放,回弹效果
- Android自定义View--ScrollView实现回弹效果
- Android中自定义仿IOS回弹效果的ListView
- Android ListView的回弹效果
- 如何实现android ScrollView ListView的回弹效果
- Android超简单实现listview上下拉伸回弹动画效果
- Android 三种布局实现上下回弹效果(普通布局,ListView,ScrollView)
- Android 三种布局实现上下回弹效果(普通布局,ListView,ScrollView)
- android ScrollView ListView的回弹效果
- C++ 虚函数表解析
- 简单使用Ajax实现异步查询用户名是否被占用
- Shiro中自定义Realm的作用(FormAuthenticationFilter和PermissionAuthorizationFilter)以及源码解析
- Visual Studio控制台程序输出窗口一闪而过的解决方法
- ssm整合原来的版本
- android 实现拉出回弹效果通过自定义ListView重写overScrollBy()
- [BZOJ3510][启发式合并][LCT维护子树信息]首都
- 06 无线通信概念
- jggrid可见列文件引入不好用
- java常用的Date方法
- 单独安装 PHP 的扩展
- CentOS 6.5搭建Redis集群
- ECharts初体验
- SQL Server,MySQL,Oracle三者的区别