Android下拉刷新框架实现代码实例

Android下拉刷新框架实现代码实例,第1张

概述前段时间项目中用到了下拉刷新功能,之前在网上也找到过类似的demo,但这些demo的质量参差不齐,用户体验也不好,接口设计也不行。最张没办法,终于忍不了了,自己就写了一个下拉刷新的框架,这个框架是一个通用的框

前段时间项目中用到了下拉刷新功能,之前在网上也找到过类似的demo,但这些demo的质量参差不齐,用户体验也不好,接口设计也不行。最张没办法,终于忍不了了,自己就写了一个下拉刷新的框架,这个框架是一个通用的框架,效果和设计感觉都还不错,现在分享给各位看官。

一. 关于下拉刷新

下拉刷新这种用户交互最早由twitter创始人洛伦•布里切特(Loren Brichter)发明,有理论认为,下拉刷新是一种适用于按照从新到旧的时间顺序排列Feeds的应用,在这种应用场景中看完旧的内容时,用户会很自然地下拉查找更新的内容,因此下拉刷新就显得非常合理。大家可以参考这篇文章:有趣的下拉刷新,下面我贴出一个有趣的下拉刷新的案例。

图一、有趣的下拉刷新案例(一)

图二、有趣的下拉刷新案例(二)

二. 实现原理

上面这些例子,外观做得再好看,他的本质上都一样,那就是一个下拉刷新控件通常由以下几部分组成:

【1】header

header通常有下拉箭头,文字,进度条等元素,根据下拉的距离来改变它的状态,从而显示不同的样式

【2】Content

这部分是内容区域,网上有很多例子都是直接在ListVIEw里面添加header,但这就有局限性,因为好多情况下并不一定是用ListVIEw来显示数据。我们把要显示内容的VIEw放置在我们的一个容器中,如果你想实现一个用ListVIEw显示数据的下拉刷新,你需要创建一个ListVIEw旋转到我的容器中。我们处理这个容器的事件(down,move,up),如果向下拉,则把整个布局向下滑动,从而把header显示出来。

【3】Footer

Footer可以用来显示向上拉的箭头,自动加载更多的进度条等。

以上三部分总结的说来,就是如下图所示的这种布局结构:


图三,下拉刷新的布局结构

关于上图,需要说明几点:

1、这个布局扩展于linearLayout,垂直排列

2、从上到下的顺序是:header,Content,Footer

3、Content填充满父控件,通过设置top,bottom的padding来使header和Footer不可见,也就是让它超出屏幕外

4、下拉时,调用scrollTo方法来将整个布局向下滑动,从而把header显示出来,上拉正好与下拉相反。

5、派生类需要实现的是:将Content VIEw填充到父容器中,比如,如果你要使用的话,那么你需要把ListVIEw,ScrollVIEw,WebVIEw等添加到容器中。

6、上图中的红色区域就是屏的大小(严格来说,这里说屏幕大小并不准确,应该说成内容区域更加准确)

三. 具体实现

明白了实现原理与过程,我们尝试来具体实现,首先,为了以后更好地扩展,设计更加合理,我们把下拉刷新的功能抽象成一个接口:
1、IPullToRefresh<T extends VIEw>

它具体的定义方法如下:

public interface IPullToRefresh<T extends VIEw> {   public voID setPullRefreshEnabled(boolean pullRefreshEnabled);   public voID setPullLoadEnabled(boolean pullLoadEnabled);   public voID setScrollLoadEnabled(boolean scrollLoadEnabled);   public boolean isPullRefreshEnabled();   public boolean isPullLoadEnabled();   public boolean isScrollLoadEnabled();   public voID setonRefreshListener(OnRefreshListener<T> refreshListener);   public voID onPullDownRefreshComplete();   public voID onPullUpRefreshComplete();   public T getRefreshableVIEw();   public LoadingLayout getheaderLoadingLayout();   public LoadingLayout getFooterLoadingLayout();   public voID setLastUpdatedLabel(CharSequence label); } 

这个接口是一个泛型的,它接受VIEw的派生类,因为要放到我们的容器中的不就是一个VIEw吗?

2、PullToRefreshBase<T extends VIEw>

这个类实现了IPullToRefresh接口,它是从linearLayout继承过来,作为下拉刷新的一个抽象基类,如果你想实现ListVIEw的下拉刷新,只需要扩展这个类,实现一些必要的方法就可以了。这个类的职责主要有以下几点:

处理onIntercepttouchEvent()和ontouchEvent()中的事件:当内容的VIEw(比如ListVIEw)正如处于最顶部,此时再向下拉,我们必须截断事件,然后move事件就会把后续的事件传递到ontouchEvent()方法中,然后再在这个方法中,我们根据move的距离再进行scroll整个VIEw。 负责创建header、Footer和Content VIEw:在构造方法中调用方法去创建这三个部分的VIEw,派生类可以重写这些方法,以提供不同式样的header和Footer,它会调用createheaderLoadingLayout和createFooterLoadingLayout方法来创建header和Footer创建Content VIEw的方法是一个抽象方法,必须让派生类来实现,返回一个非null的VIEw,然后容器再把这个VIEw添加到自己里面。 设置各种状态:这里面有很多状态,如下拉、上拉、刷新、加载中、释放等,它会根据用户拉动的距离来更改状态,状态的改变,它也会把header和Footer的状态改变,然后header和Footer会根据状态去显示相应的界面式样。

3、PullToRefreshBase<T extends VIEw>继承关系

这里我实现了三个下拉刷新的派生类,分别是ListVIEw、ScrollVIEw、WebVIEw三个,它们的继承关系如下:


图四、PullToRefreshBase类的继承关系

关于PullToRefreshBase类及其派和类,有几点需要说明:

对于ListVIEw,ScrollVIEw,WebVIEw这三种情况,他们是否滑动到最顶部或是最底部的实现是不一样的,所以,在PullToRefreshBase类中需要调用两个抽象方法来判断当前的位置是否在顶部或底部,而其派生类必须要实现这两个方法。比如对于ListVIEw,它滑动到最顶部的条件就是第一个child完全可见并且first postion是0。这两个抽象方法是:

/**  * 判断刷新的VIEw是否滑动到顶部  *  * @return true表示已经滑动到顶部,否则false  */ protected abstract boolean isReadyForPullDown();  /**  * 判断刷新的VIEw是否滑动到底  *  * @return true表示已经滑动到底部,否则false  */ protected abstract boolean isReadyForPullUp(); 

创建可下拉刷新的VIEw(也就是content vIEw)的抽象方法是

/**  * 创建可以刷新的VIEw  *  * @param context context  * @param attrs 属性  * @return VIEw  */ protected abstract T createRefreshableVIEw(Context context,AttributeSet attrs); 

4、LoadingLayout

LoadingLayout是刷新Layout的一个抽象,它是一个抽象基类。header和Footer都扩展于这个类。这类抽象类,提供了两个抽象方法:

getContentSize

这个方法返回当前这个刷新Layout的大小,通常返回的是布局的高度,为了以后可以扩展为水平拉动,所以方法名字没有取成getLayoutHeight()之类的,这个返回值,将会作为松手后是否可以刷新的临界值,如果下拉的偏移值大于这个值,就认为可以刷新,否则不刷新,这个方法必须由派生类来实现。

setState

这个方法用来设置当前刷新Layout的状态,PullToRefreshBase类会调用这个方法,当进入下拉,松手等动作时,都会调用这个方法,派生类里面只需要根据这些状态实现不同的界面显示,如下拉状态时,就显示出箭头,刷新状态时,就显示loading的图标。

可能的状态值有:reset,PulL_TO_REFRESH,RELEASE_TO_REFRESH,REFRESHING,NO_MORE_DATA

LoadingLayout及其派生类的继承关系如下图所示:


图五、LoadingLayout及其派生类的类图

我们可以随意地制定自己的header和Footer,我们也可以实现如图一和图二中显示的各种下拉刷新案例中的header和Footer,只要重写上述两个方法getContentSize()和setState()就行了。headerLoadingLayout,它默认是显示箭头式样的布局,而RotateLoadingLayout则是显示一个旋转图标的式样。

5、事件处理

我们必须重写PullToRefreshBase类的两个事件相关的方法onIntercepttouchEvent()和ontouchEvent()方法。由于ListVIEw,ScrollVIEw,WebVIEw它们是放到PullToRefreshBase内部的,所在事件先是传递到PullToRefreshBase#onIntercepttouchEvent()方法中,所以我们应该在这个方法中去处理ACTION_MOVE事件,判断如果当前ListVIEw,ScrollVIEw,WebVIEw是否在最顶部或最底部,如果是,则开始截断事件,一旦事件被截断,后续的事件就会传递到PullToRefreshBase#onIntercepttouchEvent()方法中,我们再在ACTION_MOVE事件中去移动整个布局,从而实现下拉或上拉动作。

6、滚动布局(scrollTo)

如图三的布局结构可知,默认情况下header和Footer是放置在Content VIEw的最上面和最下面,通过设置padding来让他跑到屏幕外面去了,如果我们将整个布局向下滚动(scrollTo)一定距离,那么header就会被显示出来,基于这种情况,所以在我的实现中,最终我是调用scrollTo来实现下拉动作的。

总的说来,实现的重要的点就这些,具体的一些细节在实现在会碰到很多,可以参考代码。

四. 如何使用

使用下拉刷新的代码如下

@OverrIDe   public voID onCreate(Bundle savedInstanceState) {     super.onCreate(savedInstanceState);          mpullListVIEw = new PullToRefreshListVIEw(this);     setContentVIEw(mpullListVIEw);          // 上拉加载不可用     mpullListVIEw.setPullLoadEnabled(false);     // 滚动到底自动加载可用     mpullListVIEw.setScrollLoadEnabled(true);          mCurIndex = mloadDataCount;     mListItems = new linkedList<String>();     mListItems.addAll(Arrays.asList(mStrings).subList(0,mCurIndex));     mAdapter = new ArrayAdapter<String>(this,androID.R.layout.simple_List_item_1,mListItems);          // 得到实际的ListVIEw     mListVIEw = mpullListVIEw.getRefreshableVIEw();     // 绑定数据     mListVIEw.setAdapter(mAdapter);         // 设置下拉刷新的Listener     mpullListVIEw.setonRefreshListener(new OnRefreshListener<ListVIEw>() {       @OverrIDe       public voID onPullDownToRefresh(PullToRefreshBase<ListVIEw> refreshVIEw) {         mIsstart = true;         new GetDataTask().execute();       }        @OverrIDe       public voID onPullUpToRefresh(PullToRefreshBase<ListVIEw> refreshVIEw) {         mIsstart = false;         new GetDataTask().execute();       }     });     setLastUpdateTime();          // 自动刷新     mpullListVIEw.doPullRefreshing(true,500);   } 

这是初始化一个下拉刷新的布局,并且调用setContentVIEw来设置到Activity中。

在下拉刷新完成后,我们可以调用onPullDownRefreshComplete()和onPullUpRefreshComplete()方法来停止刷新和加载

五. 运行效果
这里列出了demo的运行效果图。


图六、ListVIEw下拉刷新,注意header和Footer的样式

六. BUG修复

已知BUG修复情况如下,发现了代码BUG的看官也可以给我反馈,谢谢~~~

1,对于ListVIEw的下拉刷新,当启用滚动到底自动加载时,如果footer由隐藏变为显示时,出现显示异常的情况
这个问题已经修复了,修正的代码如下:

PullToRefreshListVIEw#setScrollLoadEnabled方法,修正后的代码如下:@OverrIDe public voID setScrollLoadEnabled(boolean scrollLoadEnabled) {   if (isScrollLoadEnabled() == scrollLoadEnabled) {     return;   }      super.setScrollLoadEnabled(scrollLoadEnabled);      if (scrollLoadEnabled) {     // 设置Footer     if (null == mloadMoreFooterLayout) {       mloadMoreFooterLayout = new FooterLoadingLayout(getContext());       mListVIEw.addFooterVIEw(mloadMoreFooterLayout,null,false);     }          mloadMoreFooterLayout.show(true);   } else {     if (null != mloadMoreFooterLayout) {       mloadMoreFooterLayout.show(false);     }   } } 

LoadingLayout#show方法,修正后的代码如下:

/**  * 显示或隐藏这个布局  *  * @param show flag  */ public voID show(boolean show) {   // If is showing,do nothing.   if (show == (VIEw.VISIBLE == getVisibility())) {     return;   }      VIEwGroup.LayoutParams params = mContainer.getLayoutParams();   if (null != params) {     if (show) {       params.height = VIEwGroup.LayoutParams.WRAP_CONTENT;     } else {       params.height = 0;     }          requestLayout();     setVisibility(show ? VIEw.VISIBLE : VIEw.INVISIBLE);   } } 

在更改LayoutParameter后,调用requestLayout()方法。

图片旋转兼容2.x系统

我之前想的是这个只需要兼容3.x以上的系统,但发现有很多网友在使用过程中遇到过兼容性问题,这次抽空将这个兼容性一并实现了。

onPull的修改如下:
    

  @OverrIDe public voID onPull(float scale) {   if (null == mRotationHelper) {     mRotationHelper = new ImageVIEwRotationHelper(mArrowImageVIEw);   }      float angle = scale * 180f; // SUPPRESS CHECKSTYLE   mRotationHelper.setRotation(angle); } 

ImageVIEwRotationHelper主要的作用就是实现了ImageVIEw的旋转功能,内部作了版本的区分,实现代码如下:

/**    * The image vIEw rotation helper    *    * @author lihong06    * @since 2014-5-2    */   static class ImageVIEwRotationHelper {     /** The imagevIEw */     private final ImageVIEw mImageVIEw;     /** The matrix */     private Matrix mMatrix;     /** Pivot X */     private float mRotationPivotX;     /** Pivot Y */     private float mRotationPivotY;          /**      * The constructor method.      *      * @param imageVIEw the image vIEw      */     public ImageVIEwRotationHelper(ImageVIEw imageVIEw) {       mImageVIEw = imageVIEw;     }          /**      * Sets the degrees that the vIEw is rotated around the pivot point. Increasing values      * result in clockwise rotation.      *      * @param rotation The degrees of rotation.      *      * @see #getRotation()      * @see #getPivotX()      * @see #getPivotY()      * @see #setRotationX(float)      * @see #setRotationY(float)      *      * @attr ref androID.R.styleable#VIEw_rotation      */     public voID setRotation(float rotation) {       if (APIUtils.hasHoneycomb()) {         mImageVIEw.setRotation(rotation);       } else {         if (null == mMatrix) {           mMatrix = new Matrix();                      // 计算旋转的中心点           Drawable imageDrawable = mImageVIEw.getDrawable();           if (null != imageDrawable) {             mRotationPivotX = Math.round(imageDrawable.getIntrinsicWIDth() / 2f);             mRotationPivotY = Math.round(imageDrawable.getIntrinsicHeight() / 2f);           }         }                  mMatrix.setRotate(rotation,mRotationPivotX,mRotationPivotY);         mImageVIEw.setimageMatrix(mMatrix);       }     }   } 

最核心的就是,如果在2.x的版本上,旋转ImageVIEw使用Matrix。

PullToRefreshBase构造方法兼容2.x

在三个参数的构造方法声明如下标注:

@Suppresslint("NewAPI")

@TargetAPI(Build.VERSION_CODES.HONEYCOMB)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。

总结

以上是内存溢出为你收集整理的Android下拉刷新框架实现代码实例全部内容,希望文章能够帮你解决Android下拉刷新框架实现代码实例所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1141365.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存