前言

本文针对自定义组件进行一些分析。还是那句老话“授之于鱼不如授之以渔”。今天要讲的是一个自定义的可以分页的ListView。

网上都讲了些ListView分页的方法,那么为什么我在这里还需要自己写一个呢?

①分页功能是很多时候都需要的;

②网上的很多代码和数据绑定在一起的,要使用的话还需要改些东西,更重要的是代码很啰嗦、很糟糕(当然只是个人风格问题,至少我是这么认为);

③或者是功能太强大而我们仅仅是需要分页功能,但是又不好分离出来。

所以写个分页的listview就很有必要了,尤其是可以直接使用的listview。

效果图

说得再多都是苍白无力的理论,先给个效果图看看吧。

分析

这个组件看上去简单,不过需要处理的细节还是挺多的。首先要明确这个组件是做什么的。简单地讲就是做分页显示或分页加载的。那么需要考虑这两个问题:

1、什么时候需要分页?

那么什么时候需要分页呢,这个答案是很明朗的。当数据数量不能填满一屏的时候那么就不用分页。在这个组件中我是这么来定分页的:首次加载的数量大于一屏显示数量时那么就认为有分页的可能,滑动到底部就需要“more”这个按钮。

2、何时加载分页信息?

至于何时进行分页,这是这个组件的关键。理论上是滑动到listview最底部的时候就需要显示一个“more”按钮,点击后进行分页,但是实际上我们需要做一定的小改动,也就是提前加载,及还没有滑动到最底部就开始加载。这样看来主要的问题就落在了如何判断是否滑动到了最底部。这问题在代码中讨论吧。

代码实现

MoreListView 组件代码:

 /***********************************************************
*@description : This class function is you can load more datas
*
* @create author : kwzhang
* @create date :2013-6-19
* @modify author :
* @modify date :
* @contact: vanezkw@163.com
*
**********************************************************/
package com.example.demo; import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.widget.AbsListView;
import android.widget.Button;
import android.widget.ListAdapter;
import android.widget.ListView; /**
* @author kwzhang
*
*/
public class MoreListView extends ListView { private View mFooter;
private LoadingListener mListener;
private boolean mShowMore = true; /**
* @param context
* @param attrs
*/
public MoreListView(Context context, AttributeSet attrs) {
super(context, attrs);
initFooter(context);
} /**
* @param mListener
* the mListener to set
*/
public void setListener(LoadingListener mListener) {
this.mListener = mListener;
setOnScrollListener(innerOnScrollListener);
} /**
* @return the mShowMore
*/
public boolean isShowMore() {
return mShowMore;
} /**
* @param mShowMore
* the mShowMore to set
*/
public void setShowMore(boolean mShowMore) {
this.mShowMore = mShowMore;
} /**
* @description :TODO
* @author : kwzhang
* @create :2013-6-19
* @param context
* @return :void
*/
protected void initFooter(Context context, View footer) {
if (null == footer) {
Button bt = new Button(context);
bt.setText("More ...");
mFooter = bt;
} else {
mFooter = footer;
}
mFooter.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (getFooterViewsCount() > 0) {
removeFooterView(mFooter);
}
if (null != mListener) {
mListener.loadingMore(MoreListView.this);
}
}
});
addFooterView(mFooter);
} private void initFooter(Context context) {
initFooter(context, null);
} @Override
public void setAdapter(ListAdapter adapter) {
super.setAdapter(adapter);
removeFooterView(mFooter);
} /**
* @param context
* @param attrs
* @param defStyle
*/
public MoreListView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
initFooter(context);
} /**
* @param context
*/
public MoreListView(Context context) {
super(context);
initFooter(context);
} private OnScrollListener innerOnScrollListener = new OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
// mLastVisibleItemIndex+1 == getAdapter().getCount()时说明滑动到最底端。
// mLastVisibleItemIndex+3 是为了在倒数第二个就开始预先加载。不过此方法是采样调用,不一定及时执行。
if (mShowMore) {
int count = getAdapter().getCount();
if ((mVisibleItemCount < count) && ((mLastVisibleItemIndex + 3) >= count) && (getFooterViewsCount() == 0)) {
addFooterView(mFooter);
}
}
} private int mLastVisibleItemIndex;
private int mVisibleItemCount = -1; @Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
mLastVisibleItemIndex = firstVisibleItem + visibleItemCount - 1; mVisibleItemCount = visibleItemCount; }
}; public static interface LoadingListener {
public void loadingMore(View view);
}
}

测试Demo Activity:

 package com.example.demo;

 import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.ArrayAdapter; public class ActDemo extends Activity implements MoreListView.LoadingListener { private MoreListView listView;
private ArrayAdapter<String> mAdapter; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView = (MoreListView) findViewById(R.id.listview);
mAdapter = getAdapter();
listView.setAdapter(mAdapter);
listView.setListener(this);
} private ArrayAdapter<String> getAdapter() {
ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1);
for (int i = 0; i < 20; i++) {
adapter.add("测试数据" + i);
}
return adapter;
} @Override
public void loadingMore(View view) {
for (int i = 0; i < 10; i++) {
mAdapter.add("新数据" + i);
}
if (mAdapter.getCount() > 60) {
listView.setShowMore(false);
}
}
}

测试Demo XML布局:

<com.example.demo.MoreListView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/listview"
android:layout_width="fill_parent"
android:layout_height="fill_parent" />

从上面的测试demo中可以看出使用起来非常方便。

1、protected void initFooter(Context context, View footer)方法说明一下。这个方法是给外部提供传递view的,这个view就是显示“more”的那个view,传null的话就有个默认的Button。

2、public void setListener(LoadingListener mListener)需要分页的话就必须调用这个方法。点击“more”按钮的时候其实去调用了LoadingListener.loadingMore(View view)方法。你可以根据需要进行实现。

总结

这样的组件就比较简单实用,也没有太复杂的代码,一看就能懂。我们需要的不正是这样简单的代码吗。随便说一下如果你需要更强的的Listview的话可以去看看AmazingListView项目。下拉刷新你可以看看johannilsson-android-pulltorefresh项目

PS:对文中有不解之处欢迎交流,有什么好的建议也可以留言。留下个QQ学习群:196761677。

自定义组件之MoreListView的更多相关文章

  1. Android开发之自定义组件和接口回调

    说到自定义控件不得不提的就是接口回调,在Android开发中接口回调用的还是蛮多的.在这篇博客开始的时候呢,我想聊一下iOS的自定义控件.在iOS中自定义控件的思路是继承自UIView, 在UIVie ...

  2. Android自定义组件

    [参考的原文地址] http://blog.csdn.net/l1028386804/article/details/47101387效果图: 实现方式: 一:自定义一个含有EditText和Butt ...

  3. 自己写的几个android自定义组件

    http://www.see-source.com/androidwidget/list.html 多多指点,尤其是自定义组件的适配问题,希望能有更好的方法

  4. PhoneGap: Android 自定义组件

    Hello Core Demo Plugin Development(组件部署): http://docs.phonegap.com/en/2.0.0/guide_plugin-development ...

  5. android开发之自定义组件

    android开发之自定义组件 一:自定义组件: 我认为,自定义组件就是android给我们提供的的一个空白的可以编辑的图片,它帮助我们实现的我们想要的界面,也就是通过自定义组件我们可以把我们要登入的 ...

  6. HTML5 UI框架Kendo UI Web教程:创建自定义组件(三)

    Kendo UI Web包 含数百个创建HTML5 web app的必备元素,包括UI组件.数据源.验证.一个MVVM框架.主题.模板等.在前面的2篇文章<HTML5 Web app开发工具Ke ...

  7. HTML5 UI框架Kendo UI Web中如何创建自定义组件(二)

    在前面的文章<HTML5 UI框架Kendo UI Web自定义组件(一)>中,对在Kendo UI Web中如何创建自定义组件作出了一些基础讲解,下面将继续前面的内容. 使用一个数据源 ...

  8. HTML5 UI框架Kendo UI Web自定义组件(一)

    Kendo UI Web包含数百个创建HTML5 web app的必备元素,包括UI组件.数据源.验证.一个MVVM框架.主题.模板等.在Kendo UI Web中如何创建自定义组件呢,在下面的文章中 ...

  9. axure复用-自定义组件,母版(模板)

    组件(控件)是用于设计线框图的用户界面元素.在组件(控件)面板中包含有常用的控件库,如按钮.图片.文本框等.从组件面板中拖动一个控件到线框图区域中,就可以添加一个组件.组件可以从一个线框图中被拷贝(C ...

随机推荐

  1. jQuery的XX如何实现?——1.框架

    源码链接:内附实例代码 jQuery使用许久了,但是有一些API的实现实在想不通.于是抽空看了jQuery源码,现在把学习过程中发现的一些彩蛋介绍给大家(⊙0⊙). 下面将使用简化的代码来介绍,主要关 ...

  2. 让VS默认以管理员身份运行

    In Windows 8 & 10, you have to right-click devenv.exe and select "Troubleshoot compatibilit ...

  3. [BTS]The join order has been enforced because a local join hint is used.;Duplicate key was ignored.".

    在一个客户的BizTalk Server 2013 R2环境中会报如下的ERROR,查找相关资料后,先试试停掉所有Trace. Log Name:      ApplicationSource:    ...

  4. 利用nodejs模块缓存机制创建“全局变量”

    在<深入浅出nodejs>有这样一段(有部分增减): 1.nodejs引入模块分四个步骤 路径分析 文件定位 编译执行 加入内存 2.核心模块部分在node源代码的编译过程中就编译成了二级 ...

  5. windows下安装PhpDocumentor(phpdoc)笔记

    PhpDocumentor简介 PHPDocumentor是一个用PHP写的工具,对于有规范注释的php程序,它能够快速生成具有相互参照,索引等功能的API文档.老的版本是phpdoc,从1.3.0开 ...

  6. Moses在Ubuntu14.04平台的安装过程

    平台环境:在windows 7中建立VMware虚拟机,操作系统为Ubuntu_14.04_amd_64 1.安装GIZA++ 安装步骤如下: wget http://giza-pp.googleco ...

  7. XXXXXXXX系统失败总结

    2013年底因同事离职接手了他负责的<XXXXXX病例系统>,当时由3个刚毕业的同事在做,框架使用Spring+Hibernate+SpringMVC+EasyUI,因为当时的主力开发人员 ...

  8. (转)数据库获得当前时间getdate()

    CONVERT(nvarchar(10),count_time,121): CONVERT为日期转换函数,一般就是在时间类型 (datetime,smalldatetime)与字符串类型(nchar, ...

  9. 配置editplus,讓其支持代碼自動格式化功能.

    使用editplus已經好多年了,累積了不少的東西,想換IDE比較麻煩,所以就研究了一下用editplus搭配gofmt.exe配置go語言代碼自動格式化的功能.還好功夫不負有心人,終於被我搞懂了,不 ...

  10. java string转为xml

    一.使用最原始的javax.xml.parsers,标准的jdk api // 字符串转XML String xmlStr = \"......\"; StringReader s ...