pulltorefreshviewlistview中有没有方法自动刷新

Android中,刷新ListView中的item导致ListView闪烁的解决方法_百度知道
Android中,刷新ListView中的item导致ListView闪烁的解决方法
如题,因为下载列表使用ListView实现的,所以必须实时更新ListView中的数据。我采用的是Service+BroadcastReceiver的搭配。更新数据在Activity里面再用notifySetDataChange()的方法对ListView进行动态刷新但是发现一个很严重的问题。刷新的时候ListView有明显的闪烁。求解决方法
发现了原因所在。用Universal-Image-Loader来加载ListView中的图片就发生这种事情。
我有更好的答案
按默认排序
真机调试?另不要用broadcast,用handler,你刷新的时候的代码在看下。
Service和Activity之间的通讯可以用Handler?handler变成一个静态的?耦合性会变得很高吧,而且我也已经知道原因了。用Universal-Image-Loader来加载ListView中的图片就发生这种事情。
哦知道原因就行了,handler的问题不会造成耦合性问题的。详细的可以找个例子看下,或者自己定义接口也可以,模仿oclick就ok。
你好!你说的问题解决了吗?我现在也遇到了这样的问题!
Universal-Http-ImageLoader这个框架惹的祸。
那怎么搞啊!现现在要交项目就出现这个问题了!难道不用??
小图的话可以用SmartImageView这个框架。大图的话自己写咯。不用控件。
其他类似问题
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁26181人阅读
前段时间项目中用到了下拉刷新功能,之前在网上也找到过类似的demo,但这些demo的质量参差不齐,用户体验也不好,接口设计也不行。最张没办法,终于忍不了了,自己就写了一个下拉刷新的框架,这个框架是一个通用的框架,效果和设计感觉都还不错,现在分享给各位看官。
1. 感谢lk6233160同学提出的问题,旋转View时调用setRotation方法只能是在API Level11(3.0)以上才能用,这个问题的解决办法是给ImageView设置一个Matrix,把Matrix上面作用一个旋转矩阵,但是如果不是ImageView的话,可能实现起来比较麻烦,再次谢谢lk6233160同学。
2. 谢谢提出的问题,向下滑动后,再向上滑动到头,只能再松手后才能再次下拉。这个问题的回复请参考评论。
技术交流群:
QQ:(人员已满)
1. 关于下拉刷新
下拉刷新这种用户交互最早由twitter创始人洛伦o布里切特(Loren Brichter)发明,有理论认为,下拉刷新是一种适用于按照从新到旧的时间顺序排列feeds的应用,在这种应用场景中看完旧的内容时,用户会很自然地下拉查找更新的内容,因此下拉刷新就显得非常合理。大家可以参考这篇文章:,下面我贴出一个有趣的下拉刷新的案例。
图一、有趣的下拉刷新案例(一)
图一、有趣的下拉刷新案例(二)
2. 实现原理
上面这些例子,外观做得再好看,他的本质上都一样,那就是一个下拉刷新控件通常由以下几部分组成:
【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、上图中的红色区域就是屏的大小(严格来说,这里说屏幕大小并不准确,应该说成内容区域更加准确)
3. 具体实现
明白了实现原理与过程,我们尝试来具体实现,首先,为了以后更好地扩展,设计更加合理,我们把下拉刷新的功能抽象成一个接口:
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()之类的,这个返回值,将会作为松手后是否可以刷新的临界值,如果下拉的偏移值大于这个值,就认为可以刷新,否则不刷新,这个方法必须由派生类来实现。
这个方法用来设置当前刷新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来实现下拉动作的。
总的说来,实现的重要的点就这些,具体的一些细节在实现在会碰到很多,可以参考代码。
4. 如何使用
使用下拉刷新的代码如下
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mPullListView = new PullToRefreshListView(this);
setContentView(mPullListView);
// 上拉加载不可用
mPullListView.setPullLoadEnabled(false);
// 滚动到底自动加载可用
mPullListView.setScrollLoadEnabled(true);
mCurIndex = mLoadDataC
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&() {
public void onPullDownToRefresh(PullToRefreshBase&ListView& refreshView) {
mIsStart =
new GetDataTask().execute();
public void onPullUpToRefresh(PullToRefreshBase&ListView& refreshView) {
mIsStart =
new GetDataTask().execute();
setLastUpdateTime();
// 自动刷新
mPullListView.doPullRefreshing(true, 500);
}这是初始化一个下拉刷新的布局,并且调用setContentView来设置到Activity中。
在下拉刷新完成后,我们可以调用onPullDownRefreshComplete()和onPullUpRefreshComplete()方法来停止刷新和加载
5. 运行效果
这里列出了demo的运行效果图。
图六、ListView下拉刷新,注意Header和Footer的样式
图七、WebView和ScrollView的下拉刷新效果图
6. 源码下载
实现这个下拉刷新的框架,并不是我的原创,我也是参考了很多开源的,把我认为比较好的东西借鉴过来,从而形成我的东西,我主要是参考了下面这个demo:
&这个demo写得不错,不过他这个太复杂了,我们都知道,一旦复杂了,万一我们要添加一些需要,自然也要费劲一些,我其实就是把他的简化再简化,以满足我们自己的需要。
转载请说明出处
谢谢!!!
7. Bug修复
已知bug修复情况如下,发现了代码bug的看官也可以给我反馈,谢谢~~~
1,对于ListView的下拉刷新,当启用滚动到底自动加载时,如果footer由隐藏变为显示时,出现显示异常的情况
这个问题已经修复了,修正的代码如下:
PullToRefreshListView#setScrollLoadEnabled方法,修正后的代码如下:
public void setScrollLoadEnabled(boolean scrollLoadEnabled) {
if (isScrollLoadEnabled() == scrollLoadEnabled) {
super.setScrollLoadEnabled(scrollLoadEnabled);
if (scrollLoadEnabled) {
// 设置Footer
if (null == mLoadMoreFooterLayout) {
mLoadMoreFooterLayout = new FooterLoadingLayout(getContext());
mListView.addFooterView(mLoadMoreFooterLayout, null, false);
mLoadMoreFooterLayout.show(true);
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())) {
ViewGroup.LayoutParams params = mContainer.getLayoutParams();
if (null != params) {
if (show) {
params.height = ViewGroup.LayoutParams.WRAP_CONTENT;
params.height = 0;
requestLayout();
setVisibility(show ? View.VISIBLE : View.INVISIBLE);
}在更改LayoutParameter后,调用requestLayout()方法。
图片旋转兼容2.x系统
我之前想的是这个只需要兼容3.x以上的系统,但发现有很多网友在使用过程中遇到过兼容性问题,这次抽空将这个兼容性一并实现了。
& & & &onPull的修改如下:
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
static class ImageViewRotationHelper {
/** The imageview */
private final ImageView mImageV
/** The matrix */
private Matrix mM
/** Pivot X */
private float mRotationPivotX;
/** Pivot Y */
private float mRotationPivotY;
* The constructor method.
* @param imageView the image view
public ImageViewRotationHelper(ImageView imageView) {
mImageView = imageV
* 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);
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)
大家如果还有什么问题,欢迎留言~~~
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:200910次
积分:3208
积分:3208
排名:第3708名
原创:102篇
评论:423条
(1)(2)(1)(1)(1)(21)(1)(58)(1)(3)(13)14030人阅读
Android实现上拉加载更多功能以及下拉刷新功能,
采用了目前比较火的PullToRefresh,他是目前实现比较好的下拉刷新的类库。
目前他支持的控件有:ListView,&ExpandableListView,GridView,WebView等。
下载地址:/chrisbanes/Android-PullToRefresh
首先第一步当然是导入libriay到咱们的项目了,具体导入方式,这里不再赘述。
下面是个例子采用的是ListView,当然其余的和这个类似
1、布局文件activity_main.xml
&RelativeLayout&xmlns:android=&&
&&&&xmlns:tools=&&
&&&&android:layout_width=&match_parent&
&&&&android:layout_height=&match_parent&
&&&&tools:context=&.MainActivity&&&
&&&&&&&com.handmark.pulltorefresh.library.PullToRefreshListView
&&&&&&&&android:id=&@+id/pull_refresh_list&
&&&&&&&&android:layout_width=&fill_parent&
&&&&&&&&android:layout_height=&fill_parent&&/&
&/RelativeLayout&
2、要实现下拉刷新的功能很简单,只需要实现OnRefreshListener的OnRefresh方法即可。这里说一下如何实现上拉和下拉分别执行不同的操作。
这里参考了http://blog.csdn.net/ueryueryuery/article/details/
原理是:根据下拉和上拉显示的布局的可见状态类区分上拉还是下拉,然后执行相应操作。
在PullToRefresh的类库的com.handmark.pulltorefresh.library包下,打开PullToRefreshBase.java,在这个类的最后面添加如下代码:
public&boolean&isHeaderShown()
&&&&return&getHeaderLayout().isShown();
public&boolean&isFooterShown()
&&&&return&getFooterLayout().isShown();
3、在Activity的代码如下:
public&class&MainActivity&extends&ListActivity
&&&&private&PullToRefreshListView
mPullToRefreshListV
&&&&private&LinkedList&String&
&&&&private&ArrayAdapter&String&
&&&&private&Context
&&&&@Override
&&&&protected&void&onCreate(Bundle
savedInstanceState) {
&&&&&&&&super.onCreate(savedInstanceState);
&&&&&&&&setContentView(R.layout.activity_main);
&&&&&&&&context
&&&&&&&&initData();
&&&&&&&&adapter
=&new&ArrayAdapter&String&(this,
android.R.layout.simple_list_item_1, mItemList);
&&&&&&&&mPullToRefreshListView
= (PullToRefreshListView)findViewById(R.id.pull_refresh_list);
&&&&&&&&ListView
mListView = mPullToRefreshListView.getRefreshableView();
&&&&&&&&mListView.setAdapter(adapter);
&&&&&&&&mPullToRefreshListView.setMode(Mode.BOTH);
&&&&&&&&mPullToRefreshListView.setOnRefreshListener(new&OnRefreshListener&ListView&()
&&&&&&&&&&&&@Override
&&&&&&&&&&&&public&void&onRefresh(PullToRefreshBase&ListView&
refreshView) {
&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&&&&&if&(refreshView.isHeaderShown()){
&&&&&&&&&&&&&&&&&&&&Toast.makeText(context,&&下拉刷新&,Toast.LENGTH_SHORT).show();
&&&&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&&&&&}else&{
&&&&&&&&&&&&&&&&&&&&Toast.makeText(context,&&上拉加载更多&,Toast.LENGTH_SHORT).show();
&&&&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&&&&&}
&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&}
&&&&&&&&});
&&&&private&void&initData(){
&&&&&&&&mItemList
=&new&LinkedList&String&();
&&&&&&&&mItemList.addAll(Arrays.asList(data));
&&&&private&String[]
data& =&new&String[]{&data1&,&data2&,&data3&,&data4&,&data5&,&data6&,
&&&&&&&&&&&&&data1&,&data2&,&data3&,&data4&,&data5&,&data6&};
如上代码所示,在OnRefresh的实现代码中,用以区分上拉还是下拉,关键代码如下:
if&(refreshView.isHeaderShown()){
&&&&&&&&&&&&&&&&&&&&Toast.makeText(context,&&下拉刷新&,Toast.LENGTH_SHORT).show();
&&&&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&&&&&}else&{
&&&&&&&&&&&&&&&&&&&&Toast.makeText(context,&&上拉加载更多&,Toast.LENGTH_SHORT).show();
&&&&&&&&&&&&&&&&&&&&
&&&&&&&&&&&&&&&&}
至此,运行项目,可以得到演示结果了
原文地址&:&
我的博客:
参考网址:http://blog.csdn.net/ueryueryuery/article/details/
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
访问:49705次
排名:千里之外
原创:29篇
评论:25条
(2)(1)(4)(1)(1)(1)(1)(8)(10)503 Service Temporarily Unavailable
503 Service Temporarily Unavailable
ASERVER/1.2.9-3

我要回帖

更多关于 pullrefresh 的文章

 

随机推荐