瀑布流ListView
前言
终于忙完了一段时间,现在前段时间写的一个瀑布流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的更多相关文章
- 飞流直下的精彩 -- 淘宝UWP中瀑布流列表的实现
在淘宝UWP中,搜索结果列表是用户了解宝贝的重要一环,其中的图片效果对吸引用户点击搜索结果,查看宝贝详情有比较大的影响.为此手机淘宝特意在搜索结果列表上采用了2种表现方式:一种就是普通的列表模式,而另 ...
- RecyclerView 制作瀑布流图片
这是瀑布流的效果图 RecyclerView(ListView的升级版) 加载图片用的是第三方图片加载框架 ------------------------------------- 在Xml中 &l ...
- android 瀑布流
我们还是来看一款示例: 看起来很像我们的gridview吧,不过又不像,因为item大小不固定的,看起来是不是别有一番风味,确实如此.就如我们的方角图形,斯通见惯后也就出现了圆角.下面我简单介绍下实现 ...
- android自定义viewgroup之我也玩瀑布流
先看效果图吧, 继上一篇<android自定义viewgroup实现等分格子布局>中实现的布局效果,这里稍微有些区别,每个格子的高度不规则,就是传说的瀑布流布局,一般实现这种效果,要么用第 ...
- 【转】android中最好的瀑布流控件PinterestLikeAdapterView
[源地址]http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/0919/1696.html 之前我们介绍过一个开源的瀑布流控件Stag ...
- android瀑布流效果(仿蘑菇街)
Android 转载分享(10) 我们还是来看一款示例:(蘑菇街) 看起来很像我们的gridview吧,不过又不像,因为item大小不固定的,看起来是不是别有一番风味,确实如此. ...
- Android RecyclerView(瀑布流)水平/垂直方向分割线
Android RecyclerView(瀑布流)水平/垂直方向分割线 Android RecyclerView不像过去的ListView那样随意的设置水平方向的分割线,如果要实现Recycle ...
- Android 高级UI设计笔记10:瀑布流控件PinterestLikeAdapterView的使用
1. 首先我们看看瀑布流的效果,如下: 2. 今天要介绍的瀑布流控件是:PinterestLikeAdapterView 项目地址:https://github.com/GDG-Korea/Pinte ...
- 使用RecyclerView实现瀑布流的效果
主函数: public class MainActivity extends AppCompatActivity { private RecyclerView recyclerView; privat ...
随机推荐
- jQuery.loadTemplate客户端模板
jQuery.Template虽然用起来没有Mustache简洁和方便,还是学习了解一下,做个笔记. 模板可以定义在页面script标签,如下 <script type="text/h ...
- 收缩sql server2008 数据库
USE DATABASENAME; GO -- Truncate the log by changing the database recovery model to SIMPLE. ALTER DA ...
- 山东省第三届ACM省赛
Solved ID PID Title Accepted Submit A 2407 Impasse (+) 0 0 B 2415 Chess 0 0 C 2414 An interest ...
- [SQL]分组排练进行更新
--方法(一):分组排练进行更新 ----------------------------------------------------------------------------------- ...
- oracle 数据恢复,只有oradata文件夹里的文件,没有备份文件的数据库恢复,重装系统后,oracle 10g数据库恢复
格式化重装系统后,才想起来oracle 10g 数据库没有做备份,开始以为很麻烦,没想到数据库恢复的还挺顺利的 恢复方法: 1,把原来的数据库文件备份,(D:\oracle\product\10.2. ...
- uva 11488 Hyper Prefix Sets(狂水)
题意: 获得集合中最长前缀长度*有该前缀个数的最大值 Prefix goodness of a set string is length of longest common prefix*number ...
- HttpClient + ASP.NET Web API, WCF之外的另一个选择
WCF的野心造成了它的庞大复杂,HTTP的单纯造就了它的简单优美.为了实现分布式Web应用,我们不得不将两者凑合在一起 —— WCF服务以HTTP绑定宿主于IIS. 于是有了让人晕头转向的配置.让人郁 ...
- 使用virtualbox安装centos虚拟机,以及VirtualBox无法安装64位Linux CentOS的解决办法
之前一直用vmware的虚拟机,好吧,其实一直盗版挺不好的,然后想用centos搭点东西,结果在vmare上安装centos总是有些问题,看了人给的建议换用virtualbox,虽然virtualbo ...
- jmeter随笔(1)-在csv中数据为json格式的数据不完整
昨天同事在使用jmeter遇到问题,在csv中数据为json格式的数据,在jmeter中无法完整的取值,小怪我看了下,给出解决办法,其实很简单,我们一起看看,看完了记得分享给你的朋友. 问题现象: 1 ...
- C语言学习笔记(1):Hello World!
#include <stdio.h> void main() { printf("Hello World!\n"); } 几乎学习任何语言接触到的第一个语言都是Hell ...