`
亚当爱上java
  • 浏览: 708976 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

ListView中重写overScrollBy()方法 实现拉出回弹效果

 
阅读更多
           第一部分  原理分析
      IOS上的bounce功能给人的感觉很爽,当一个可以滚动的区域被拖到边界时,它允许用户将内容拖过界,放手后再弹回来,以一种非常棒的方式提示了用户边界的存在,是IOS的一大特色。android2.3新增了overscroll功能,听名字就知道应该是bounce功能的翻版,但也许是出于专利方面的考虑,google的默认实现跟IOS有所不同,它只是在list拖到边界处时做了一个发光的动画,个人觉得体验比IOS差远了。而且这个黄色的发光在黑色背景下虽然效果不错,在其它背景下可就难说了,因此很多人想要关掉它。

      日前google上搜索“android overscroll”,对此效果的介绍很多,但关于其具体使用方式和实现,则很少涉及,偶有提及,也经常答非所问或似是而非,反而误导了别人。于是我查阅了android相关源码,并做了一些测试,在此讲讲我的理解。

      首先是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,而且我没找到任何可以影响其结果的设置。

      真悲催,绕了半天,android的默认实现里面根本没有给出overscroll功能,它只是提供了实现机制,要想用起来还得应用程序自己显式重写相关控件,估计还有一层隐含的意思,法律风险自负。在我的系统中一试,果然一个像素都不能拉出界。但那个闪光是怎么回事呢?

      在处理笔点消息处,overScrollBy后面不远处有一段mEdgeGlowTop的操作代码,看名字就像,用它一搜,相关机制就全明白了。mEdgeGlowTop在setOverScrollMode函数时创建,它使用的图片都是系统中固有的,甚至不能通过theme改变。它的实现原理也很简单,仅仅是两张png图片的合成,通过透明度的变化制造闪光的效果。更无语的是它既不能被应用程序访问,也不受任何控制,要关闭它的唯一办法是setOverScrollMode(View.OVER_SCROLL_NEVER)。否则就重写onTouchEvent函数吧,想干啥都可以,只是得自己做。

      谈到overScroll,很多文章都提到了ListView的setOverscrollHeader和setOverscrollFooter,很多人想通过这个来控制那个闪光效果。这两玩意不但可以通过函数设置,也可以在xml中指定,相当方便。但最后很多人发现没有任何作用,百思不得其解。其实这两张图片是用来作为overScroll拖过界时的背景的,默认系统不能拖过界,自然永远都看不到,有些定制的系统中能拖出界几个像素,但也很难看清。

                第二部分  代码实现
     1. 在View中增加了overSrollBy方法,用于记录x, y 轴上滚动。

     2. 在AbsListView的onTouchEvent中判断是否到达边界(顶部 或 底部) ,然后调用view.overScrollBy ,传入 mScrollY等参数

     3. overScrollBy 最终赋值给View的mScrollX, mScrollY 两个变量

     4. 在AbsListView中调用完overScrollBy之后,调用invalidate重绘



自定义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 size
         
        final DisplayMetrics metrics = mContext.getResources().getDisplayMetrics();
        final float density = metrics.density;
         
        mMaxYOverscrollDistance = (int) (density * MAX_Y_OVERSCROLL_DISTANCE);
    }
     
    @Override
    protected 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);  
    }
     
}



    public class MainActivity extends Activity {  
      
        @Override  
        public void onCreate(Bundle savedInstanceState) {  
            super.onCreate(savedInstanceState);  
              
            LinearLayout linearLayout = new LinearLayout(this);  
            linearLayout.setLayoutParams(new LayoutParams(LayoutParams.FILL_PARENT, LayoutParams.FILL_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);  
        }  
          
          
          
    }  

分享到:
评论

相关推荐

    ScrollView ListView 普通布局的回弹效果的实现

    本篇文章将深入探讨如何在ScrollView、ListView以及普通的布局中实现这种回弹效果。 首先,我们来看ScrollView的回弹效果。ScrollView是Android内置的一个可以垂直滚动的单向容器,它允许其内部的组件超出屏幕的...

    ListView回弹效果

    本篇文章将详细探讨如何实现ListView的回弹效果,以及其在源码中的实现原理。 “回弹效果”,又称为下拉刷新或滑动加载更多,是一种常见的交互设计,用户通过手势向下拉动ListView,列表会呈现出一个回弹动作,然后...

    android listview 下拉回弹刷新效果

    总之,实现Android ListView的下拉回弹刷新效果,需要结合SwipeRefreshLayout组件、事件监听、自定义动画和性能优化等多个方面。通过这些技术,你可以为你的APP增添类似QQ空间和新浪微博的互动体验,提升用户的使用...

    Android实现ListView可伸展并回弹功能

    在该方法中,我们需要检测用户的滑动行为,并计算出滑动的距离,以便于在手指离开屏幕时模拟回弹效果。 在处理触摸事件时,我们可以使用MotionEvent的ACTION_DOWN、ACTION_MOVE和ACTION_UP三个关键动作。ACTION_...

    实现ListView头部和尾部可伸展并回弹功能

    然而,为了增强用户体验,有时我们需要实现ListView的头部和尾部可伸展并回弹的效果,这种功能通常被称为“Pull to Refresh”(下拉刷新)和“Load More”(上拉加载更多)。本篇文章将详细介绍如何在Android中实现...

    c# 多个ListView重写控件

    在C#编程中,ListView控件是一个非常常用且功能强大的组件,它允许开发者在Windows Forms或WPF应用程序中展示列表...在实践中,我们应该根据项目的需求,选择合适的方法和工具,以实现高效、美观、易用的ListView控件。

    仿小米lListView回弹效果

    4. **自定义滚动监听**: 可以通过重写`OnScrollListener`,在`onScroll`和`onScrollStateChanged`方法中判断是否到达边界,并执行回弹动画。 在"test_Android3"这个压缩包文件中,可能包含了实现这一效果的示例代码...

    Android ListView 下拉刷新、上拉加载

    2. 自定义滚动效果:通过重写ListView的onScroll()和onScrollStateChanged()方法,可以实现自定义的滚动行为。 3. 自定义触摸事件处理:通过覆写onInterceptTouchEvent()和onTouchEvent()方法,可以控制ListView的...

    Android实现ListView阻尼效果

    为了提供更好的用户体验,开发者经常会在ListView中实现一种类似物理运动的阻尼效果,即当用户快速滑动ListView到列表顶部或底部时,ListView会有一种“弹跳”或者“回弹”的效果,这种效果被称为“阻尼效果”。...

    安卓Android源码——重写listview实现下拉刷新.zip

    本压缩包"安卓Android源码——重写listview实现下拉刷新.zip"就是针对这一需求提供的一种解决方案。 首先,我们来理解下拉刷新这一概念。下拉刷新是一种常见的移动应用设计模式,用户通过在ListView顶部向下拉动,...

    Android 重写Listview实现左滑删除功能

    本项目"Android 重写Listview实现左滑删除功能"旨在教你如何在ListView中添加滑动删除的交互效果。 首先,我们需要理解ListView的工作原理。ListView通过复用View(也称为ViewHolder模式)来优化性能,减少内存消耗...

    AndroidListView反弹效果源码.zip

    重写`onTouchEvent()`方法,捕获用户的滑动事件,然后根据滑动方向和速度来控制OverScroller,实现个性化的反弹效果。 5. **布局管理器**:对于更复杂的布局,例如网格布局(GridView)或瀑布流布局...

    android应用源码Android重写listview实现下拉刷新.zip

    【标题】"Android应用源码Android重写ListView实现下拉刷新.zip" 提供了一个关于在Android平台上使用自定义ListView实现下拉刷新功能的实例。在Android开发中,下拉刷新是一种常见的用户交互模式,用于更新列表数据...

    Android 实现ListView 3D效果 - 1

    本篇将详细讲解如何在Android中实现ListView的3D翻转效果,参照的是CSDN博主"love_world_"的一篇文章。 首先,要实现3D效果,我们需要对ListView的每个项视图(Item View)进行处理。这通常通过自定义Adapter来完成...

    高仿android-QQ空间下拉背景图拉伸回弹效果

    本项目主要探讨如何实现“高仿Android-QQ空间下拉背景图拉伸回弹效果”,这是一个自定义ListView控件的实例,旨在模拟QQ空间主页面下拉时背景图片的动态变化,即图片随着手指滑动而拉伸,松手后又会回弹到原始状态。...

    ListView反弹效果实现

    ListView的反弹效果,也称为回弹或Over Scroll效果,是指用户在滑动到ListView的顶部或底部时,继续滑动,ListView会有一个自然回弹的效果,使得用户体验更加流畅和自然。本篇文章将深入探讨如何在Android中实现...

    Android之用PopupWindow实现弹出listview形式的菜单

    在Android开发中,`PopupWindow`是一个非常实用的组件,它可以用来实现各种形式的弹出窗口,如下拉菜单、提示框等。本教程将详细讲解如何使用`PopupWindow`来创建一个以`ListView`形式展示的菜单。首先,我们需要...

    Android 实现ListView 3D效果 - 2 - 弹性滚动,Fling

    本篇将深入探讨如何在Android中实现ListView的3D弹性滚动,即Fling效果。 首先,要理解3D效果的本质,它通常是通过改变ListView项的透明度、大小或位置来模拟深度感。在Android中,我们可以通过自定义Adapter和...

Global site tag (gtag.js) - Google Analytics