瀑布流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 ...
随机推荐
- Java中Map的用法
Map的一般用法 1.声明一个Map : Map map = new HashMap(); 2 .向map中放值 ,注意: map是key-value的形式存放的,如: map.put("s ...
- struts2访问servlet API
搭建环境: 引入jar包,src下建立struts.xml文件 项目配置文件web.xml. web.xml: <?xml version="1.0" encoding=&q ...
- Codeforces 626D Jerry's Protest 「数学组合」「数学概率」
题意: 一个袋子里装了n个球,每个球都有编号.甲乙二人从每次随机得从袋子里不放回的取出一个球,如果甲取出的球比乙取出的球编号大则甲胜,否则乙胜.保证球的编号xi各不相同.每轮比赛完了之后把取出的两球放 ...
- CodeForces 604D 【离散数学 置换群】
题意: 给你一个方程,方程太变态不打,给你一个p一个k,p保证是大于等于3的质数,k保证在0~p-1之间的整数.要求对应函数的定义域在0~p-1值域为0~p-1的子集,求这样的函数有多少个... 分析 ...
- 用HTML代码加载Unity内容
Unity内容在浏览器通过Unity网络播放器插件加载.HTML代码与这个插件通常不直接通信,而是通过UnityObject的脚本帮助.其主要任务是Unity的内容嵌入一个非常简单的任务,通过从各种浏 ...
- OC基础(21)
Foundation框架介绍 NSString基本概念 字符串读写 字符串比较 字符串搜索 字符串截取 字符串替换 字符串与路径 字符串与基本数据类型转换 *:first-child { margin ...
- 初探接口测试框架--python系列2
点击标题下「蓝色微信名」可快速关注 坚持的是分享,搬运的是知识,图的是大家的进步,没有收费的培训,没有虚度的吹水,喜欢就关注.转发(免费帮助更多伙伴)等来交流,想了解的知识请留言,给你带来更多价值,是 ...
- Xcode entitlement 问题定位和解决指南
背景故事 前两天,本来一个运行正常.打包测试都没问题的XCode工程突然爆出各种奇怪的Entitlement错误: 什么签名的内容跟配置文件的不一致. 又或者 无法安装,因为签名或者配置文件的配置错误 ...
- conpot_usage简要说明
conpot是一个ICS(工业控制系统)蜜罐, 旨在收集攻击者针对工业控制系统的攻击方法和动机. 这篇文章主要用来说明conpot的用户定制相关的一些配置. (英文原文详见: https://gith ...
- pyhton标准库 json
使用loads方法即可将json字符串转换成python对象,对应关系如下: #JSON Python object dict array list string unicode number (in ...