前言  

  终于忙完了一段时间,现在前段时间写的一个瀑布流ListView到想法分享下,这个东西是扩展自Listview,当列表内容拉到最后后触发刷新操作,以便抓取更多到数据。

先贴下整个代码,先有个整体到概念,然后再进一步细聊。代码如下

 package cn.bitlove.waterfalllistview.widget;

 import cn.bitlove.waterfalllistview.R;
import android.content.Context;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AbsListView;
import android.widget.FrameLayout;
import android.widget.AbsListView.OnScrollListener;
import android.widget.ListView; /**
* 瀑布流Listview
* */
public class WaterfallListview extends ListView implements OnScrollListener { final private String tag="WaterfallListView";
private Context mContext;
private LayoutInflater mInflater;
final int REFRESH_IDEL=0; //空闲中
final int REFRESH_PREPARE=1; //准备刷新
final int REFRESH_ING=2; //刷新中
final int REFRESH_End=3; //刷新完成
private int mRefreshState = REFRESH_IDEL; //刷新状态
private int mScrollState; //滚动状态 private FrameLayout mFooterLayout; //ListView 的footer
private View mListFoot; //底部刷新区域 private IOnRefresh mRefreshListener = null; //刷新监听器
public WaterfallListview(Context context) {
super(context);
}
public WaterfallListview(Context context, AttributeSet attrs) {
super(context,attrs);
init();
}
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
mScrollState = scrollState;
} @Override
public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
if(totalItemCount==0 || mScrollState==SCROLL_STATE_IDLE){
return;
}
//最后一个item
int lastItemCount = firstVisibleItem+visibleItemCount;
if(lastItemCount==totalItemCount && mRefreshState==REFRESH_IDEL){
prepareRefresh();
doRefresh();
}
} private void init(){
mContext = getContext();
mInflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
initFooter();
setOnScrollListener(this);
}
/**
* 设置刷新器
* */
public void setRefreshListener(IOnRefresh refresher){
mRefreshListener = refresher;
} /**
* 初始化底部刷新内容
* */
private void initFooter(){
mListFoot = mInflater.inflate(R.layout.foot_waterfall_listview, null);
mFooterLayout = new FrameLayout(mContext);
addFooterView(mFooterLayout);
}
/**
* 准备刷新
* */
private void prepareRefresh(){
//Log.i(tag,"prepareRefresh");
mRefreshState = REFRESH_PREPARE;
ViewGroup vg = (ViewGroup) mListFoot.getParent();
if(vg!=null){
vg.removeView(mListFoot);
} mFooterLayout.addView(mListFoot); if(mRefreshListener!=null){
mRefreshListener.beforeRefresh();
}
}
/**
* 执行刷新
* */
private void doRefresh(){
//Log.i(tag,"doRefresh");
mRefreshState=REFRESH_ING; if(mRefreshListener!=null){
mRefreshListener.doRefresh();
} }
/**
* 完成刷新
* */
public void completeRefresh(){
mRefreshState = REFRESH_IDEL;
mFooterLayout.removeAllViews();
} public interface IOnRefresh{
public void beforeRefresh();
/**
* 执行完doRefresh操作后,必须调用completeRefresh方法来清理刷新状态
* */
public void doRefresh();
}
}

底部视图  

  为了完成滑动到底部后触发刷新事件,为此给Listview增加了一个底部view,代码如下

 <?xml version="1.0" encoding="utf-8"?>
<!-- waterfallListview 底部刷新区域 -->
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
> <TextView
android:layout_height="40dp"
android:layout_width="match_parent"
android:text="加载更多数据..."
android:gravity="center"
/>
</RelativeLayout>

很简单,只是用来展示一个获取状态。

接口

   除此之外,还增加了一个内部接口,IOnRefresh,表示相关事件,代码如下所示

 public interface IOnRefresh{
public void beforeRefresh();
/**
* 执行完doRefresh操作后,必须调用completeRefresh方法来清理刷新状态
* */
public void doRefresh();
}

这两个方法的含义根据名字来看就很清楚了,beforeRefresh,在刷新之前调用,让client有机会做一些事情;doRefresh,真正执行刷新到部分,另外需要注意到时候,当执行完刷新到时候,要掉用下widget的completeRefresh方法来清理刷新状态,标示真正执行完了刷新操作

completeRefresh方法

很简单到一个方法,有两个主要功能,一是置状态,二是取消listview底部的view,代码如下

 public void completeRefresh(){
mRefreshState = REFRESH_IDEL;
mFooterLayout.removeAllViews();
}

设计思路

  大体到内容就是以上那些,下面来看下实现的思路。
  为了完成这个功能,增加了一个刷新状态mRefreshState,这个状态用来表示当前控件所处到刷新状态,以便在不同状态下做不同到处理,其有以下四个状态

final int REFRESH_IDEL=0;        //空闲中
final int REFRESH_PREPARE=1;    //准备刷新
final int REFRESH_ING=2;        //刷新中
final int REFRESH_End=3;        //刷新完成

然后整个widget到逻辑就是控制这四个状态,然后在不同到状态执行不同到操作,比如在执行beforeRefresh函数时,此时状态处于REFRESH_PREPARE,
当执行doRefresh时状态处于REFRESH_ING,当我们调用completeRefresh后,状态改为REFRESH_IDEL,其实当初设计的时候还希望有一个REFRESH_End状态,但是实现过程中没有发现此状态的用处,暂时为保留状态,以便以后扩充用。

控制状态

控制状态,主要是在onScroll函数中操作的,代码如下

 @Override
public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
if(totalItemCount==0 || mScrollState==SCROLL_STATE_IDLE){
return;
}
//最后一个item
int lastItemCount = firstVisibleItem+visibleItemCount;
if(lastItemCount==totalItemCount && mRefreshState==REFRESH_IDEL){
prepareRefresh();
doRefresh();
}
}

主要实现思路就是监测最后一个item是否出现。

  好了,到此为止,主要到思路就完了。

后记

  这个控件到Demo程序下载地址:https://github.com/xiaoai-opensource/WaterfallListview

  有需要到朋友可以前去下载查看

瀑布流ListView的更多相关文章

  1. 飞流直下的精彩 -- 淘宝UWP中瀑布流列表的实现

    在淘宝UWP中,搜索结果列表是用户了解宝贝的重要一环,其中的图片效果对吸引用户点击搜索结果,查看宝贝详情有比较大的影响.为此手机淘宝特意在搜索结果列表上采用了2种表现方式:一种就是普通的列表模式,而另 ...

  2. RecyclerView 制作瀑布流图片

    这是瀑布流的效果图 RecyclerView(ListView的升级版) 加载图片用的是第三方图片加载框架 ------------------------------------- 在Xml中 &l ...

  3. android 瀑布流

    我们还是来看一款示例: 看起来很像我们的gridview吧,不过又不像,因为item大小不固定的,看起来是不是别有一番风味,确实如此.就如我们的方角图形,斯通见惯后也就出现了圆角.下面我简单介绍下实现 ...

  4. android自定义viewgroup之我也玩瀑布流

    先看效果图吧, 继上一篇<android自定义viewgroup实现等分格子布局>中实现的布局效果,这里稍微有些区别,每个格子的高度不规则,就是传说的瀑布流布局,一般实现这种效果,要么用第 ...

  5. 【转】android中最好的瀑布流控件PinterestLikeAdapterView

    [源地址]http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/0919/1696.html 之前我们介绍过一个开源的瀑布流控件Stag ...

  6. android瀑布流效果(仿蘑菇街)

    Android 转载分享(10)  我们还是来看一款示例:(蘑菇街)           看起来很像我们的gridview吧,不过又不像,因为item大小不固定的,看起来是不是别有一番风味,确实如此. ...

  7. Android RecyclerView(瀑布流)水平/垂直方向分割线

     Android RecyclerView(瀑布流)水平/垂直方向分割线 Android RecyclerView不像过去的ListView那样随意的设置水平方向的分割线,如果要实现Recycle ...

  8. Android 高级UI设计笔记10:瀑布流控件PinterestLikeAdapterView的使用

    1. 首先我们看看瀑布流的效果,如下: 2. 今天要介绍的瀑布流控件是:PinterestLikeAdapterView 项目地址:https://github.com/GDG-Korea/Pinte ...

  9. 使用RecyclerView实现瀑布流的效果

    主函数: public class MainActivity extends AppCompatActivity { private RecyclerView recyclerView; privat ...

随机推荐

  1. 算法库:基础线性代数子程序库(Basic Linear Algebra Subprograms,BLAS)介绍

    调试DeepFlow光流算法,由于作者给出的算法是基于Linux系统的,所以要在Windows上运行,不得不做大量的修改工作.移植到Windows平台,除了一些头文件找不到外,还有一些函数也找不到.这 ...

  2. NYOJ 55-懒省事的小明

    点击打开链接 懒省事的小明 时间限制:3000 ms  |  内存限制:65535 KB 难度:3 描述       小明很想吃果子,正好果园果子熟了.在果园里,小明已经将所有的果子打了下来,而且按果 ...

  3. ios7 Cocos2dx 隐藏状态栏设置

    环境: cocos2d-x 2.1.5 ios7.0     在info.plist 添加 UIViewControllerBasedStatusBarAppearance(View controll ...

  4. MapReduce从HBase读写数据简单示例

    就用单词计数这个例子,需要统计的单词存在HBase中的word表,MapReduce执行的时候从word表读取数据,统计结束后将结果写入到HBase的stat表中. 1.在eclipse中建立一个ha ...

  5. XMLHttpRequest的五步使用方法

    <html> <head> <title>Demo</title> <style> body,input,button,select,h1{ ...

  6. Linux-Apache+Mysql+PHP+PHPWind(重点Apache+PHP集成环境)

    整理Apache+Mysql+PHP+PHPWind(Apache+PHP集成环境) 一.情况简述: 1.虚拟机VM上面CentOS 2.全部yum安装(yum安装与源码安装的安装路径不同) 二.操作 ...

  7. 慧自文档:代替 Everything 来快速查找文件的,实现文件显示在文件夹的层次结构中

    1. 搜索功能和Everything一样快和强大 具有 Everything 搜索快.搜索功能强等优点, 解决了不能方便选择搜索哪个文件夹, 解决了不能同一个画面进行预览等问题 2.文件直接显示到文件 ...

  8. com学习(一)GUID 和 接口

    话说在 doc(Word) 复合文件中,已经解决了保存 xls(Excel) 数据的问题了.那么,接下来又要解决另一个问题:当 WORD 程序读取复合文件,遇到了 xls 数据的时候,它该如何启动 E ...

  9. jquery是如何清除ajax缓存的

    大家都知道万恶的IE在ajax中往往只读取第一次ajax请求时候的数据,其余时候都是从cache提取数据,(太懒了T_T).原生的JS清除ajax缓存的方法多,但是终觉有点繁琐,如果是用jquery的 ...

  10. AD采样问题总结

    说明:来源http://bbs.csdn.net/topics/390899032论坛讨论 一个100HZ的正弦波,我用300HZ的采样率去采样,那么根据香农定律是不是一秒钟就采集到300个点,因为这 ...