实现原理
1.下拉刷新
通过onTouchEvent判断手势,来改变listview的header。header的状态共4种,自己定义为:
NONE(对应图1):初始状态
PULL(对应图2):下拉状态,此时松开会还原到状态NONE,并不进行刷新
RELEASE(对应图3):同样是下拉状态,但此刻松开会执行刷新,进入状态REFRESHING
REFRESHING(对应图4):正在执行刷新,刷新结束后进入状态NONE。
header在四种状态切换时不仅改变内部组件,同时改变自身的大小。改变内部组件的体现比如,箭头的朝上或者朝下,文字提示的变化,等待圆圈的显示与否。大小的改变其实就是高度的改变,NONE时header高度为0,RELEASE时header的高度由你下拉的程度决定。
2.加载更多
在listview滑动停止后,判断listview的最后一个item是否已经显示,如果显示说明listview已经滑动到了最底部,这时便触发加载更多的方法,方法结束根据结果改变footer。
3.回调方法
在类中定义了两个接口OnRefreshListener和OnLoadListener,用来定义和提供加载数据的方法,具体实现则交给它们的实现类去做。
/**
* 支持下拉刷新和上拉加载的列表
*
* @author Internet
*/
publicclass XListViewextends
ListView implements OnScrollListener {
private float mLastY
= -1; // save event y
private Scroller
mScroller; // used for scroll back
private OnScrollListener
mScrollListener; // user's scrolllistener
// the interface to trigger refreshand load more.
private XListViewListener
mListViewListener;
// -- header view
private XListViewHeader
mHeaderView;
// header view content, use it to calculatethe Header's height. And hide it
// when disable pull refresh.
private RelativeLayout
mHeaderViewContent;
private TextView mHeaderTimeView;
private int mHeaderViewHeight;
// header view'sheight
private boolean mEnablePullRefresh=
true;
private boolean mPullRefreshing
= false; // is refreashing.
// -- footer view
private XListViewFooter
mFooterView;
private boolean mEnablePullLoad
= false;
private boolean mPullLoading;
private boolean mIsFooterReady
= false;
// total list items, used to detect is atthe bottom of listview.
private int mTotalItemCount;
// for mScroller, scroll back from headeror footer.
private int mScrollBack;
private final static int SCROLLBACK_HEADER=
0;
private final static int SCROLLBACK_FOOTER=
1;
private final static int SCROLL_DURATION
= 400; // scroll back duration
private final static int
PULL_LOAD_MORE_DELTA=
50; // when pull up>= 50px
// at bottom, trigger
// load more.
private final static float
OFFSET_RADIO =
1.8f; // support iOS likepull
// feature.
public XListView(Context context) {
super(context);
initWithContext(context);
}
public XListView(Context context, AttributeSetattrs) {
super(context, attrs);
initWithContext(context);
}
public XListView(Context context, AttributeSetattrs,
int defStyle) {
super(context, attrs, defStyle);
initWithContext(context);
}
private void initWithContext(Context context) {
mScroller = new
Scroller(context, new DecelerateInterpolator());
// XListView need the scroll event, and itwill dispatch the event to
// user's listener (as a proxy).
super.setOnScrollListener(this);
// init header view
mHeaderView =
new XListViewHeader(context);
mHeaderViewContent = (RelativeLayout)mHeaderView
.findViewById(R.id.xlistview_header_content);
mHeaderTimeView = (TextView)
mHeaderView
.findViewById(R.id.xlistview_header_time);
addHeaderView(mHeaderView);
// init footer view
mFooterView =
new XListViewFooter(context);
// init header height
mHeaderView.getViewTreeObserver().addOnGlobalLayoutListener(
new OnGlobalLayoutListener(){
@Override
public void onGlobalLayout() {
mHeaderViewHeight=
mHeaderViewContent.getHeight();
getViewTreeObserver()
.removeGlobalOnLayoutListener(this);
}
});
}
@Override
public void setAdapter(ListAdapter adapter) {
// make sure XListViewFooter is the lastfooter view, and only add once.
if (mIsFooterReady
== false &&
mEnablePullLoad) {
mIsFooterReady =
true;
addFooterView(mFooterView);
}
super.setAdapter(adapter);
}
/**
* 设置底部显示的文本提示
*
* @param show
显示更多
* @param show_touch
松开载入更多
*/
public void setFooterText(String show, Stringshow_touch) {
mFooterView.setFooterText(show, show_touch);
}
/**
* enable or disable pull downrefresh feature.
*
* @param enable
*/
public void setPullRefreshEnable(boolean
enable) {
mEnablePullRefresh = enable;
if (!mEnablePullRefresh) {
// disable, hide the content
mHeaderViewContent.setVisibility(View.INVISIBLE);
} else {
mHeaderViewContent.setVisibility(View.VISIBLE);
}
}
/**
* enable or disable pull up loadmore feature.
*
* @param enable
*/
public void setPullLoadEnable(boolean
enable) {
mEnablePullLoad = enable;
mHeaderView.setOnClickListener(null);// no touching.
if (!mEnablePullLoad) {
mFooterView.hide();
mFooterView.setOnClickListener(null);
} else {
mPullLoading =
false;
mFooterView.show();
mFooterView.setState(XListViewFooter.STATE_NORMAL);
// both "pull up" and"click" will invoke load more.
mFooterView.setOnClickListener(new
OnClickListener(){
@Override
public void onClick(View v) {
startLoadMore();
}
});
}
}
/**
* 设置为弹性列表(前提是未启用footer)
*/
public void setSpring() {
//尾部处理
if (mEnablePullLoad)
return;
mEnablePullLoad =
true;
mFooterView.show();
mFooterView.setFooterSpring();
//头部处理
if (mEnablePullRefresh)
return;
mEnablePullRefresh =
true;
mHeaderViewContent.setVisibility(View.GONE);
}
/**
* 仅设置为弹性列表
* (就是给版本低,项目多的一个单独的优化,项目多必须超过一屏幕)
*
* @param low_and_more
系统版本低,并且项目多(会启用上滑,否则低版本是禁用上滑的)
*/
public void setSpringOnly(boolean
low_and_more) {
setPullRefreshEnable(false);
if (android.os.Build.VERSION.SDK_INT
>= 19) {//受到系统版本影响
setFooterDividersEnabled(false);
setSpring();
} else {
setSpring();
//项目多时,注释此行即可(项目过少会出现双线),所以通过禁用上滑来隐藏
if (!low_and_more)
setPullLoadEnable(false);
}
setXListViewListener(new XListViewListener(){
public void onRefresh() {
stopRefresh();
}
public void onLoadMore() {
stopLoadMore();
}
});
}
/**
* 仅设置为弹性列表(默认)
*/
public void setSpringOnly() {
setSpringOnly(false);
}
/**
* stop refresh, reset header view.
*/
public void stopRefresh() {
if (mPullRefreshing
== true) {
mPullRefreshing =
false;
resetHeaderHeight();
}
}
/**
* stop load more, reset footer view.
*/
public void stopLoadMore() {
if (mPullLoading
== true) {
mPullLoading =
false;
mFooterView.setState(XListViewFooter.STATE_NORMAL);
}
}
/**
* set last refresh time
*
* @param time
*/
//我把布局中的gone了
public void setRefreshTime(String time) {
mHeaderTimeView.setText(time);
}
private void invokeOnScrolling() {
if (mScrollListener
instanceof OnXScrollListener){
OnXScrollListener l =(OnXScrollListener) mScrollListener;
l.onXScrolling(this);
}
}
private void updateHeaderHeight(float
delta) {
if (mEnablePullRefresh
&& !mPullRefreshing) {
mHeaderView.setVisiableHeight((int) delta
+ mHeaderView.getVisiableHeight());
if (mHeaderView.getVisiableHeight() >
mHeaderViewHeight) {
mHeaderView.setState(XListViewHeader.STATE_READY);
} else {
mHeaderView.setState(XListViewHeader.STATE_NORMAL);
}
}
setSelection(0); // scroll to topeach time
}
/**
* reset header view's height.
*/
private void resetHeaderHeight() {
int height = mHeaderView.getVisiableHeight();
if (height ==
0) // not visible.
return;
// refreshing and header isn't shown fully.do nothing.
if (mPullRefreshing
&& height<= mHeaderViewHeight) {
return;
}
int finalHeight =
0; // default: scroll back to dismiss header.
// is refreshing, just scrollback to show all the header.
if (mPullRefreshing
&& height> mHeaderViewHeight) {
finalHeight = mHeaderViewHeight;
}
mScrollBack =
SCROLLBACK_HEADER;
mScroller.startScroll(0, height,
0, finalHeight - height,
SCROLL_DURATION);
// trigger computeScroll
invalidate();
}
private void updateFooterHeight(float
delta) {
int height = mFooterView.getBottomMargin() + (int) delta;
if (mEnablePullLoad
&& !mPullLoading) {
if (height >
PULL_LOAD_MORE_DELTA) {
// height enough toinvoke load
// more.
mFooterView.setState(XListViewFooter.STATE_READY);
} else {
mFooterView.setState(XListViewFooter.STATE_NORMAL);
}
}
mFooterView.setBottomMargin(height);
// setSelection(mTotalItemCount - 1); //scroll to bottom
}
private void resetFooterHeight() {
int bottomMargin =
mFooterView.getBottomMargin();
if (bottomMargin >
0) {
mScrollBack =
SCROLLBACK_FOOTER;
mScroller.startScroll(0, bottomMargin,
0, -bottomMargin,
SCROLL_DURATION);
invalidate();
}
}
private void startLoadMore() {
mPullLoading =
true;
mFooterView.setState(XListViewFooter.STATE_LOADING);
if (mListViewListener
!= null) {
mListViewListener.onLoadMore();
}
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
if (mLastY
== -1) {
mLastY = ev.getRawY();
}
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
mLastY = ev.getRawY();
break;
case MotionEvent.ACTION_MOVE:
final float deltaY =ev.getRawY() -
mLastY;
mLastY = ev.getRawY();
if (getFirstVisiblePosition()==
0
&& (mHeaderView.getVisiableHeight() >
0 || deltaY > 0)) {
// the first itemis showing, header has shown or pull down.
updateHeaderHeight(deltaY/ OFFSET_RADIO);
invokeOnScrolling();
} else if (getLastVisiblePosition()==
mTotalItemCount-
1
&& (mFooterView.getBottomMargin()>
0 || deltaY < 0)) {
// last item,already pulled up or want to pull up.
updateFooterHeight(-deltaY/ OFFSET_RADIO);
}
break;
default:
mLastY = -1;
// reset
if (getFirstVisiblePosition()==
0) {
// invoke refresh
if (mEnablePullRefresh
&& mHeaderView.getVisiableHeight()>
mHeaderViewHeight) {
mPullRefreshing
= true;
mHeaderView.setState(XListViewHeader.STATE_REFRESHING);
if (mListViewListener
!= null) {
mListViewListener.onRefresh();
}
}
resetHeaderHeight();
}
if (getLastVisiblePosition()==
mTotalItemCount-
1) {
// invoke loadmore.
if (mEnablePullLoad
&& mFooterView.getBottomMargin()>
PULL_LOAD_MORE_DELTA) {
startLoadMore();
}
resetFooterHeight();
}
break;
}
return super.onTouchEvent(ev);
}
@Override
public void computeScroll() {
if (mScroller.computeScrollOffset()) {
if (mScrollBack
== SCROLLBACK_HEADER) {
mHeaderView.setVisiableHeight(mScroller.getCurrY());
} else {
mFooterView.setBottomMargin(mScroller.getCurrY());
}
postInvalidate();
invokeOnScrolling();
}
super.computeScroll();
}
@Override
public void setOnScrollListener(OnScrollListener l) {
mScrollListener = l;
}
@Override
public void onScrollStateChanged(AbsListView view,
int scrollState) {
if (mScrollListener
!= null) {
mScrollListener.onScrollStateChanged(view,scrollState);
}
}
/**
* @param view
视图
* @param firstVisibleItem目前可见的第一条是列表中的第几条
* @param visibleItemCount目前可见的条数
* @param totalItemCount
列表总的条数
*/
@Override
public void onScroll(AbsListView view,
int firstVisibleItem,
int visibleItemCount,
int totalItemCount) {
// send to user's listener
mTotalItemCount = totalItemCount;
if (mScrollListener
!= null) {
mScrollListener.onScroll(view,firstVisibleItem, visibleItemCount,
totalItemCount);
}
}
public void setXListViewListener(XListViewListener l){
mListViewListener = l;
}
/**
* you can listenListView.OnScrollListener or this one. it will invoke
* onXScrolling when header/footerscroll back.
*/
public interface OnXScrollListener
extends OnScrollListener {
public void onXScrolling(View view);
}
/**
* implements this interface to getrefresh/load more event.
*/
public interface XListViewListener{
public void onRefresh();
public void onLoadMore();
}
}