Android新姿势:可以下拉/上拉回弹的ListView原理(续)

原创
2014/11/18 10:20
阅读数 5.2K

上一次自己根据原理写了个下拉ListView,结果现在才发现,其实谷歌的码神早已经做了相关的实现代码了!!


可是为什么直接用ListView看不到下拉效果呢?


其实这是因为版权的问题,这个下拉的效果是苹果先出的,大家也都知道苹果跟谷歌的版权纠纷,所以谷歌虽然也能做到下拉,但没有明确给出来,只是做了一个动画效果,就是拉到顶部或底部后继续拉会有个亮亮的光出现(注意是2.3之后的系统才有)。


接下来看看谷歌给我们提供了什么。


首先是View类内的overScrollBy方法。

    protected boolean overScrollBy(int deltaX, int deltaY,

            int scrollX, int scrollY,

            int scrollRangeX, int scrollRangeY,

            int maxOverScrollX, int maxOverScrollY,

            boolean isTouchEvent) {

        final int overScrollMode = mOverScrollMode;

        final boolean canScrollHorizontal =

                computeHorizontalScrollRange() > computeHorizontalScrollExtent();

        final boolean canScrollVertical =

                computeVerticalScrollRange() > computeVerticalScrollExtent();

        final boolean overScrollHorizontal = overScrollMode == OVER_SCROLL_ALWAYS ||

                (overScrollMode == OVER_SCROLL_IF_CONTENT_SCROLLS && canScrollHorizontal);

        final boolean overScrollVertical = overScrollMode == OVER_SCROLL_ALWAYS ||

                (overScrollMode == OVER_SCROLL_IF_CONTENT_SCROLLS && canScrollVertical);


        int newScrollX = scrollX + deltaX;

        if (!overScrollHorizontal) {

            maxOverScrollX = 0;

        }


        int newScrollY = scrollY + deltaY;

        if (!overScrollVertical) {

            maxOverScrollY = 0;

        }


        // Clamp values if at the limits and record

        final int left = -maxOverScrollX;

        final int right = maxOverScrollX + scrollRangeX;

        final int top = -maxOverScrollY;

        final int bottom = maxOverScrollY + scrollRangeY;


        boolean clampedX = false;

        if (newScrollX > right) {

            newScrollX = right;

            clampedX = true;

        } else if (newScrollX < left) {

            newScrollX = left;

            clampedX = true;

        }


        boolean clampedY = false;

        if (newScrollY > bottom) {

            newScrollY = bottom;

            clampedY = true;

        } else if (newScrollY < top) {

            newScrollY = top;

            clampedY = true;

        }


        onOverScrolled(newScrollX, newScrollY, clampedX, clampedY);


        return clampedX || clampedY;

    } 


其实View的这个方法已经帮我们实现了下拉、上拉,甚至左拉右拉都实现了!!

可是为什么ListView继承了View却没有实现下拉呢?


我们接下来到ListView中的overScrollBy看一下。

protected boolean overScrollBy(int deltaX, int deltaY, int scrollX,

int scrollY, int scrollRangeX, int scrollRangeY,

int maxOverScrollX, int maxOverScrollY, boolean isTouchEvent) {

L.d("deltaX=" + deltaX + ";deltaY=" + deltaY + ";scrollX=" + scrollX

+ ";scrollY=" + scrollY + ";scrollRangeX=" + scrollRangeX

+ ";scrollRangeY=" + scrollRangeY + ";maxOverScrollX="

+ maxOverScrollX + ";maxOverScrollY=" + maxOverScrollY);

return false;


根据我打印的结果,发现后面的maxOverScrollX、maxOverScrollY永远是0!!

原来如此,谷歌也真是机(jiao)智(hua),代码明明实现了,就是不明确摆出来,这样苹果也无话可说了。


那么知道了原因,要实现也下拉也简单,自定义一个类继承ListView,重写overScrollBy方法即可。

protected boolean overScrollBy(int deltaX, int deltaY, int scrollX,

int scrollY, int scrollRangeX, int scrollRangeY,

int maxOverScrollX, int maxOverScrollY, boolean isTouchEvent) {

return super.overScrollBy(deltaX, deltaY, scrollX, scrollY,

scrollRangeX, scrollRangeY, maxOverScrollX,

500, isTouchEvent);

上面的代码只是给maxOverScrollY硬性的给了个500的值,然后就可以下拉上拉了,很简单吧~~


PS:这种方法做出来的下拉ListView效果一般,也没有下拉刷新功能,如果要用到项目中的话,还是去找开源的下拉列表吧~~ 


展开阅读全文
加载中
点击引领话题📣 发布并加入讨论🔥
打赏
0 评论
2 收藏
0
分享
返回顶部
顶部