文档章节

ListView下拉刷新,上拉加载更多,终结版

董家二少
 董家二少
发布于 2014/03/27 13:58
字数 3471
阅读 12424
收藏 13

话不多说直接代码

代码结构,主要是两个类,实现头部功能的封装,写出一个自定义控件

1、头部封装类

/**

 * 该类主要是完成 头部部分的功能封装

 * 

 * 一个可以监听ListView是否滚动到最顶部或最底部的自定义控件 只能监听由触摸产生的,如果是ListView本身Flying导致的,则不能监听</br>

 * 如果加以改进,可以实现监听scroll滚动的具体位置等

 * 

 * @author

 */


public class ScrollOverListView extends ListView implements OnScrollListener {


private int mLastY;


private int mBottomPosition;


private static final String TAG = "listview";


/** 松开更新 **/

private final static int RELEASE_To_REFRESH = 0;

/** 下拉更新 **/

private final static int PULL_To_REFRESH = 1;

/** 更新中 **/

private final static int REFRESHING = 2;

/** 无 **/

private final static int DONE = 3;

/** 加载中 **/

private final static int LOADING = 4;

/** 实际的padding的距离与界面上偏移距离的比例 **/

private final static int RATIO = 3;


private LayoutInflater inflater;

/** 头部刷新的布局 **/

private LinearLayout headView;

/** 头部显示下拉刷新等的控件 **/

private TextView tipsTextview;

/** 刷新控件 **/

private TextView lastUpdatedTextView;

/** 箭头图标 **/

private ImageView arrowImageView;

/** 头部滚动条 **/

private ProgressBar progressBar;

/** 显示动画 **/

private RotateAnimation animation;

/** 头部回退显示动画 **/

private RotateAnimation reverseAnimation;

/** 用于保证startY的值在一个完整的touch事件中只被记录一次 **/

private boolean isRecored;

/** 头部高度 **/

private int headContentHeight;

/** 开始的Y坐标 **/

private int startY;

/** 第一个item **/

private int firstItemIndex;

/** 状态 **/

private int state;


private boolean isBack;

/** 是否要使用下拉刷新功能 **/

public boolean showRefresh = true;


public static boolean canRefleash = true;


public ScrollOverListView(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

init(context);

}


public ScrollOverListView(Context context, AttributeSet attrs) {

super(context, attrs);

init(context);

}


public ScrollOverListView(Context context) {

super(context);

init(context);

}


/** 出事化控件 **/

private void init(Context context) {

mBottomPosition = 0;

setCacheColorHint(0);

inflater = LayoutInflater.from(context);


headView = (LinearLayout) inflater.inflate(R.layout.pull_down_head,

null);

arrowImageView = (ImageView) headView

.findViewById(R.id.head_arrowImageView);

arrowImageView.setMinimumWidth(70);

arrowImageView.setMinimumHeight(50);

progressBar = (ProgressBar) headView

.findViewById(R.id.head_progressBar);

tipsTextview = (TextView) headView.findViewById(R.id.head_tipsTextView);

lastUpdatedTextView = (TextView) headView

.findViewById(R.id.head_lastUpdatedTextView);


measureView(headView);


headContentHeight = headView.getMeasuredHeight();


headView.setPadding(0, -1 * headContentHeight, 0, 0);

headView.invalidate();


/** 列表添加头部 **/

addHeaderView(headView, null, false);


setOnScrollListener(this);


animation = new RotateAnimation(0, -180,

RotateAnimation.RELATIVE_TO_SELF, 0.5f,

RotateAnimation.RELATIVE_TO_SELF, 0.5f);

animation.setInterpolator(new LinearInterpolator());

animation.setDuration(250);

animation.setFillAfter(true);


reverseAnimation = new RotateAnimation(-180, 0,

RotateAnimation.RELATIVE_TO_SELF, 0.5f,

RotateAnimation.RELATIVE_TO_SELF, 0.5f);


reverseAnimation.setInterpolator(new LinearInterpolator());

reverseAnimation.setDuration(200);

reverseAnimation.setFillAfter(true);


state = DONE;

}


/** 触摸事件的处理 **/

@Override

public boolean onTouchEvent(MotionEvent ev) {


final int action = ev.getAction();

final int y = (int) ev.getRawY();

cancelLongPress();

switch (action) {

case MotionEvent.ACTION_DOWN: { // 按下的时候

if (firstItemIndex == 0 && !isRecored) {

isRecored = true;

startY = (int) ev.getY();

}

mLastY = y;

final boolean isHandled = mOnScrollOverListener.onMotionDown(ev);

if (isHandled) {

mLastY = y;

return isHandled;

}

break;

}


case MotionEvent.ACTION_MOVE: { // 手指正在移动的时候

int tempY = (int) ev.getY();

if (showRefresh) {


if (!isRecored && firstItemIndex == 0) {

Log.v(TAG, "在move时候记录下位置");

isRecored = true;

startY = tempY;

}

if (state != REFRESHING && isRecored && state != LOADING) {


// 保证在设置padding的过程中,当前的位置一直是在head,否则如果当列表超出屏幕的话,当在上推的时候,列表会同时进行滚动


// 可以松手去刷新了

if (state == RELEASE_To_REFRESH) {


setSelection(0);


// 往上推了,推到了屏幕足够掩盖head的程度,但是还没有推到全部掩盖的地步

if (((tempY - startY) / RATIO < headContentHeight)

&& (tempY - startY) > 0) {

state = PULL_To_REFRESH;

changeHeaderViewByState();


}

// 一下子推到顶了

else if (tempY - startY <= 0) {

state = DONE;

changeHeaderViewByState();


}

// 往下拉了,或者还没有上推到屏幕顶部掩盖head的地步

else {

// 不用进行特别的操作,只用更新paddingTop的值就行了

}

}

// 还没有到达显示松开刷新的时候,DONE或者是PULL_To_REFRESH状态

if (state == PULL_To_REFRESH) {


setSelection(0);


// 下拉到可以进入RELEASE_TO_REFRESH的状态

if ((tempY - startY) / RATIO >= headContentHeight) {

state = RELEASE_To_REFRESH;

isBack = true;

changeHeaderViewByState();


}

// 上推到顶了

else if (tempY - startY <= 0) {

state = DONE;

changeHeaderViewByState();


}

}


// done状态下

if (state == DONE) {

if (tempY - startY > 0) {

state = PULL_To_REFRESH;

changeHeaderViewByState();

}

}


// 更新headView的size

if (state == PULL_To_REFRESH) {

headView.setPadding(0, -1 * headContentHeight

+ (tempY - startY) / RATIO, 0, 0);

}


// 更新headView的paddingTop

if (state == RELEASE_To_REFRESH) {

headView.setPadding(0, (tempY - startY) / RATIO

- headContentHeight, 0, 0);

}


}

}

final int childCount = getChildCount();

if (childCount == 0)

return super.onTouchEvent(ev);

final int itemCount = getAdapter().getCount() - mBottomPosition;

final int deltaY = y - mLastY;

final int lastBottom = getChildAt(childCount - 1).getBottom();

final int end = getHeight() - getPaddingBottom();


final int firstVisiblePosition = getFirstVisiblePosition();


final boolean isHandleMotionMove = mOnScrollOverListener

.onMotionMove(ev, deltaY);


if (isHandleMotionMove) {

mLastY = y;

return true;

}


/** 到达底部 * 到达底部的事件在另外一个类执行 **/

if (firstVisiblePosition + childCount >= itemCount

&& lastBottom <= end && deltaY < 0) {

final boolean isHandleOnListViewBottomAndPullDown;

isHandleOnListViewBottomAndPullDown = mOnScrollOverListener

.onListViewBottomAndPullUp(deltaY);

if (isHandleOnListViewBottomAndPullDown) {

mLastY = y;

return true;

}

}

break;

}


case MotionEvent.ACTION_UP: { // 手指抬起来的时候

if (state != REFRESHING && state != LOADING) {

if (state == DONE) {

// 什么都不做

}

if (state == PULL_To_REFRESH) {

state = DONE;

changeHeaderViewByState();

}


if (state == RELEASE_To_REFRESH) {

state = REFRESHING;

changeHeaderViewByState();


canRefleash = true;


}

}


isRecored = false;

isBack = false;


// /======================

final boolean isHandlerMotionUp = mOnScrollOverListener

.onMotionUp(ev);

if (isHandlerMotionUp) {

mLastY = y;

return true;

}

break;

}

}

mLastY = y;

return super.onTouchEvent(ev);

}


/** 空的 */

private OnScrollOverListener mOnScrollOverListener = new OnScrollOverListener() {


@Override

public boolean onListViewTopAndPullDown(int delta) {

return false;

}


@Override

public boolean onListViewBottomAndPullUp(int delta) {

return false;

}


@Override

public boolean onMotionDown(MotionEvent ev) {

return false;

}


@Override

public boolean onMotionMove(MotionEvent ev, int delta) {

return false;

}


@Override

public boolean onMotionUp(MotionEvent ev) {

return false;

}


};


// =============================== public method

/**

* 可以自定义其中一个条目为头部,头部触发的事件将以这个为准,默认为第一个

* @param index

*            正数第几个,必须在条目数范围之内

*/

public void setTopPosition(int index) {

if (getAdapter() == null)

throw new NullPointerException(

"You must set adapter before setTopPosition!");

if (index < 0)

throw new IllegalArgumentException("Top position must > 0");

}


/**

* 可以自定义其中一个条目为尾部,尾部触发的事件将以这个为准,默认为最后一个

* @param index

*            倒数第几个,必须在条目数范围之内

*/

public void setBottomPosition(int index) {

if (getAdapter() == null)

throw new NullPointerException(

"You must set adapter before setBottonPosition!");

if (index < 0)

throw new IllegalArgumentException("Bottom position must > 0");


mBottomPosition = index;

}


/**

* 设置这个Listener可以监听是否到达顶端,或者是否到达低端等事件</br>

* @see OnScrollOverListener

*/

public void setOnScrollOverListener(

OnScrollOverListener onScrollOverListener) {

mOnScrollOverListener = onScrollOverListener;

}


/**

* 滚动监听接口

* @see ScrollOverListView#setOnScrollOverListener(OnScrollOverListener)

*/

public interface OnScrollOverListener {

/**

* 到达最顶部触发

* @param delta

*            手指点击移动产生的偏移量

* @return

*/

boolean onListViewTopAndPullDown(int delta);


/**

* 到达最底部触发

* @param delta

*            手指点击移动产生的偏移量

* @return

*/

boolean onListViewBottomAndPullUp(int delta);


/**

* 手指触摸按下触发,相当于{@link MotionEvent#ACTION_DOWN}

* @return 返回true表示自己处理

* @see View#onTouchEvent(MotionEvent)

*/

boolean onMotionDown(MotionEvent ev);


/**

* 手指触摸移动触发,相当于{@link MotionEvent#ACTION_MOVE}

* @return 返回true表示自己处理

* @see View#onTouchEvent(MotionEvent)

*/

boolean onMotionMove(MotionEvent ev, int delta);


/**

* 手指触摸后提起触发,相当于{@link MotionEvent#ACTION_UP}

* @return 返回true表示自己处理

* @see View#onTouchEvent(MotionEvent)

*/

boolean onMotionUp(MotionEvent ev);


}


@Override

public void onScroll(AbsListView arg0, int firstVisiableItem, int arg2,

int arg3) {

firstItemIndex = firstVisiableItem;


}


@Override

public void onScrollStateChanged(AbsListView view, int scrollState) {


}


// 此方法直接照搬自网络上的一个下拉刷新的demo,此处是“估计”headView的width以及height

private void measureView(View child) {

ViewGroup.LayoutParams p = child.getLayoutParams();

if (p == null) {

p = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,

ViewGroup.LayoutParams.WRAP_CONTENT);

}

int childWidthSpec = ViewGroup.getChildMeasureSpec(0, 0 + 0, p.width);

int lpHeight = p.height;

int childHeightSpec;

if (lpHeight > 0) {

childHeightSpec = MeasureSpec.makeMeasureSpec(lpHeight,

MeasureSpec.EXACTLY);

} else {

childHeightSpec = MeasureSpec.makeMeasureSpec(0,

MeasureSpec.UNSPECIFIED);

}

child.measure(childWidthSpec, childHeightSpec);

}


public void onRefreshComplete() {

state = DONE;

// 头部更新时候的操作,显示当前系统进行更新的时间

lastUpdatedTextView.setText("最近更新:" + new Date().toLocaleString());

changeHeaderViewByState();

}


// 当状态改变时候,调用该方法,以更新界面

private void changeHeaderViewByState() {

switch (state) {

case RELEASE_To_REFRESH:

arrowImageView.setVisibility(View.VISIBLE);

progressBar.setVisibility(View.GONE);

tipsTextview.setVisibility(View.VISIBLE);

lastUpdatedTextView.setVisibility(View.VISIBLE);


arrowImageView.clearAnimation();

arrowImageView.startAnimation(animation);


tipsTextview.setText("松开刷新");

break;

case PULL_To_REFRESH:

progressBar.setVisibility(View.GONE);

tipsTextview.setVisibility(View.VISIBLE);

lastUpdatedTextView.setVisibility(View.VISIBLE);

arrowImageView.clearAnimation();

arrowImageView.setVisibility(View.VISIBLE);

// 是由RELEASE_To_REFRESH状态转变来的

if (isBack) {

isBack = false;

arrowImageView.clearAnimation();

arrowImageView.startAnimation(reverseAnimation);


tipsTextview.setText("下拉刷新");

} else {

tipsTextview.setText("下拉刷新");

}

break;


case REFRESHING:


headView.setPadding(0, 0, 0, 0);

progressBar.setVisibility(View.VISIBLE);

arrowImageView.clearAnimation();

arrowImageView.setVisibility(View.GONE);

tipsTextview.setText("正在刷新...");

lastUpdatedTextView.setVisibility(View.VISIBLE);

break;

case DONE:

headView.setPadding(0, -1 * headContentHeight, 0, 0);

progressBar.setVisibility(View.GONE);

arrowImageView.clearAnimation();

arrowImageView.setImageResource(R.drawable.pull_down_arrow);

tipsTextview.setText("下拉刷新");

lastUpdatedTextView.setVisibility(View.VISIBLE);

break;

}

}

}

2、要写个自定义控件把这个给加载进去

/**

 * 下拉刷新控件   真正实现下拉刷新的是这个控件, ScrollOverListView只是提供触摸的事件等

 * 

 * @author 

 */


public class PullDownView extends LinearLayout implements OnScrollOverListener {


private static final int START_PULL_DEVIATION = 50; /** 移动误差*/

private static final int WHAT_DID_MORE = 5; /** Handler what 已经获取完更多*/

private static final int WHAT_DID_REFRESH = 3; /** Handler what 已经刷新完*/

/**底部更多的按键**/

private RelativeLayout mFooterView;

/**底部更多的按键**/

private TextView mFooterTextView;

/**底部更多的按键**/

private ProgressBar mFooterLoadingView;

/**已经含有 下拉刷新功能的列表**/

private ScrollOverListView mListView;

/**刷新和更多的事件接口**/

private OnPullDownListener mOnPullDownListener;

private float mMotionDownLastY; /** 按下时候的Y轴坐标*/

private boolean mIsFetchMoreing; /** 是否获取更多中*/

private boolean mIsPullUpDone; /** 是否回推完成*/

private boolean mEnableAutoFetchMore; /** 是否允许自动获取更多*/


public PullDownView(Context context, AttributeSet attrs) {

super(context, attrs);

initHeaderViewAndFooterViewAndListView(context);

}


public PullDownView(Context context) {

super(context);

initHeaderViewAndFooterViewAndListView(context);

}


/*

* ================================== Public method 外部使用,具体就是用这几个就可以了

*/


/**

* 刷新和获取更多事件接口

*/

public interface OnPullDownListener {

/**刷新事件接口  这里要注意的是获取更多完 要关闭 刷新的进度条RefreshComplete()**/

void onRefresh();

/**刷新事件接口  这里要注意的是获取更多完 要关闭 更多的进度条 notifyDidMore()**/

void onMore();

}


/**

* 通知已经获取完更多了,要放在Adapter.notifyDataSetChanged后面

* 当你执行完更多任务之后,调用这个notyfyDidMore() 才会隐藏加载圈等操作

*/

public void notifyDidMore() {

mUIHandler.sendEmptyMessage(WHAT_DID_MORE);

}


/** 刷新完毕 关闭头部滚动条 **/

public void RefreshComplete() {

mUIHandler.sendEmptyMessage(WHAT_DID_REFRESH);

}


/**

* 设置监听器

* @param listener

*/

public void setOnPullDownListener(OnPullDownListener listener) {

mOnPullDownListener = listener;

}


/**

* 获取内嵌的listview

* @return ScrollOverListView

*/

public ListView getListView() {

return mListView;

}


/**

* 是否开启自动获取更多 自动获取更多,将会隐藏footer,并在到达底部的时候自动刷新

* @param index

*            倒数第几个触发

*/

public void enableAutoFetchMore(boolean enable, int index) {

if (enable) {

mListView.setBottomPosition(index);

mFooterLoadingView.setVisibility(View.GONE);

} else {

mFooterTextView.setText("数据加载完毕");

mFooterLoadingView.setVisibility(View.GONE);

}

mEnableAutoFetchMore = enable;

}


/**

* Private method 具体实现下拉刷新等操作

*/


/**

* 初始化界面

*/

private void initHeaderViewAndFooterViewAndListView(Context context) {

setOrientation(LinearLayout.VERTICAL);


/**

* 自定义脚部文件

*/

mFooterView = (RelativeLayout) LayoutInflater.from(context).inflate(

R.layout.pulldown_footer, null);

mFooterTextView = (TextView) mFooterView

.findViewById(R.id.pulldown_footer_text);

mFooterLoadingView = (ProgressBar) mFooterView

.findViewById(R.id.pulldown_footer_loading);

// mFooterView.setOnClickListener(new OnClickListener() {

// @Override

// public void onClick(View v) {

// if (!mIsFetchMoreing) {

//

// mIsFetchMoreing = true;

// mFooterTextView.setText("加载更多中...");

// mFooterLoadingView.setVisibility(View.VISIBLE);

// mOnPullDownListener.onMore();

// }

// }

// });


/**

* ScrollOverListView 同样是考虑到都是使用,所以放在这里 同时因为,需要它的监听事件

*/

mListView = new ScrollOverListView(context);

mListView.setOnScrollOverListener(this);

mListView.setCacheColorHint(0);

addView(mListView, LayoutParams.FILL_PARENT, LayoutParams.FILL_PARENT);


// 空的listener

mOnPullDownListener = new OnPullDownListener() {

@Override

public void onRefresh() {

}


@Override

public void onMore() {

}

};


mListView.addFooterView(mFooterView);

mListView.setFooterDividersEnabled(false);

//mListView.setAdapter(mListView.getAdapter());


}


private Handler mUIHandler = new Handler() {


@Override

public void handleMessage(Message msg) {

switch (msg.what) {

case WHAT_DID_REFRESH: {

mListView.onRefreshComplete();

return;

}


case WHAT_DID_MORE: {

mIsFetchMoreing = false;

mFooterTextView.setText("加载更多");

mFooterLoadingView.setVisibility(View.GONE);

}

}

}


};



/**

* 条目是否填满整个屏幕

*/

private boolean isFillScreenItem() {

final int firstVisiblePosition = mListView.getFirstVisiblePosition();

final int lastVisiblePostion = mListView.getLastVisiblePosition()

- mListView.getFooterViewsCount();

final int visibleItemCount = lastVisiblePostion - firstVisiblePosition

+ 1;

final int totalItemCount = mListView.getCount()

- mListView.getFooterViewsCount();


if (visibleItemCount < totalItemCount)

return true;

return false;

}


/*

* ================================== 实现 OnScrollOverListener接口

*/


@Override

public boolean onListViewTopAndPullDown(int delta) {


return true;

}


@Override

public boolean onListViewBottomAndPullUp(int delta) {

if (!mEnableAutoFetchMore || mIsFetchMoreing)

return false;

/** 数量充满屏幕才触发*/

if (isFillScreenItem()) {

mIsFetchMoreing = true;

mFooterTextView.setText("加载更多中...");

mFooterLoadingView.setVisibility(View.VISIBLE);

mOnPullDownListener.onMore();

return true;

}

return false;

}


@Override

public boolean onMotionDown(MotionEvent ev) {

mIsPullUpDone = false;

mMotionDownLastY = ev.getRawY();


return false;

}


@Override

public boolean onMotionMove(MotionEvent ev, int delta) {

/** 当头部文件回推消失的时候,不允许滚动*/

if (mIsPullUpDone)

return true;


/** 如果开始按下到滑动距离不超过误差值,则不滑动*/

final int absMotionY = (int) Math.abs(ev.getRawY() - mMotionDownLastY);

if (absMotionY < START_PULL_DEVIATION)

return true;


return false;

}


@Override

public boolean onMotionUp(MotionEvent ev) {

if (ScrollOverListView.canRefleash) {

ScrollOverListView.canRefleash = false;

mOnPullDownListener.onRefresh();

}

return false;

}


/**隐藏头部 禁用下拉更新**/

public void setHideHeader() {

mListView.showRefresh = false;

}


/**显示头部 使用下拉更新**/

public void setShowHeader() {

mListView.showRefresh = true;

}


/**隐藏底部 禁用上拉更多**/

public void setHideFooter() {

mFooterView.setVisibility(View.GONE);

mFooterTextView.setVisibility(View.GONE);

mFooterLoadingView.setVisibility(View.GONE);

enableAutoFetchMore(false, 1);

}

/**显示底部 使用上拉更多**/

public void setShowFooter() {

mFooterView.setVisibility(View.VISIBLE);

mFooterTextView.setVisibility(View.VISIBLE);

mFooterLoadingView.setVisibility(View.VISIBLE);

enableAutoFetchMore(true, 1);

}


}

3、实际操作

代码实例:

/**

 * 下拉刷新上拉更多

 * 

 * @author

 */

public class PullDownActivity extends Activity implements OnPullDownListener,

OnItemClickListener {


/** Handler What加载数据完毕 **/

private static final int WHAT_DID_LOAD_DATA = 0;

/** Handler What更新数据完毕 **/

private static final int WHAT_DID_REFRESH = 1;

/** Handler What更多数据完毕 **/

private static final int WHAT_DID_MORE = 2;


private ListView mListView;

private ArrayAdapter<String> mAdapter;

private PullDownView mPullDownView;

private List<String> mStrings = new ArrayList<String>();

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.pulldown);


/**

* 1.使用PullDownView 2.设置OnPullDownListener 3.从mPullDownView里面获取ListView

*/

mPullDownView = (PullDownView) findViewById(R.id.pull_down_view);


mPullDownView.setOnPullDownListener(this);


mListView = mPullDownView.getListView();

mListView.setOnItemClickListener(this);


mAdapter = new ArrayAdapter<String>(this, R.layout.pulldown_item,

mStrings);

mListView.setAdapter(mAdapter);


// 设置可以自动获取更多 滑到最后一个自动获取 改成false将禁用自动获取更多

mPullDownView.enableAutoFetchMore(true, 1);

// 隐藏 并禁用尾部

mPullDownView.setHideFooter();

// 显示并启用自动获取更多

mPullDownView.setShowFooter();

/** 隐藏并且禁用头部刷新*/

mPullDownView.setHideHeader();

/** 显示并且可以使用头部刷新*/

mPullDownView.setShowHeader();

/** 之前 网上很多代码 都会导致刷新事件 跟 上下文菜单同时弹出 这里做测试。。。已经解决 */

mListView

.setOnCreateContextMenuListener(new OnCreateContextMenuListener() {

@Override

public void onCreateContextMenu(ContextMenu menu, View v,

ContextMenuInfo menuInfo) {

}

});


// 加载数据 本类使用

loadData();


}


/** 刷新事件接口 这里要注意的是获取更多完 要关闭 刷新的进度条RefreshComplete() **/

@Override

public void onRefresh() {


new Thread(new Runnable() {


@Override

public void run() {

try {

Thread.sleep(2000);

} catch (InterruptedException e) {

e.printStackTrace();

}


/** 关闭 刷新完毕 ***/

mPullDownView.RefreshComplete();// 这个事线程安全的 可看源代码


// 这个地方有点疑问

Message msg = mUIHandler.obtainMessage(WHAT_DID_REFRESH);

msg.obj = "After refresh " + System.currentTimeMillis();

msg.sendToTarget();

}

}).start();


}


/** 刷新事件接口 这里要注意的是获取更多完 要关闭 更多的进度条 notifyDidMore() **/

@Override

public void onMore() {

new Thread(new Runnable() {


@Override

public void run() {

try {

Thread.sleep(2000);

} catch (InterruptedException e) {

e.printStackTrace();

}


// 告诉它获取更多完毕 这个事线程安全的 可看源代码

mPullDownView.notifyDidMore();

Message msg = mUIHandler.obtainMessage(WHAT_DID_MORE);

msg.obj = "After more " + System.currentTimeMillis();

msg.sendToTarget();

}

}).start();

}


private Handler mUIHandler = new Handler() {


@Override

public void handleMessage(Message msg) {

switch (msg.what) {

case WHAT_DID_LOAD_DATA: {

if (msg.obj != null) {

List<String> strings = (List<String>) msg.obj;

if (!strings.isEmpty()) {

mStrings.addAll(strings);

mAdapter.notifyDataSetChanged();

}

}

Toast.makeText(PullDownActivity.this,

"" + mListView.getAdapter().getCount(),

Toast.LENGTH_LONG).show();

// 诉它数据加载完毕;

break;

}

case WHAT_DID_REFRESH: {

String body = (String) msg.obj;

mStrings.add(0, body);

mAdapter.notifyDataSetChanged();

// 告诉它更新完毕

break;

}


case WHAT_DID_MORE: {

String body = (String) msg.obj;

mStrings.add(body);

mAdapter.notifyDataSetChanged();


break;

}

}


}


};


@Override

public void onItemClick(AdapterView<?> parent, View view, int position,

long id) {

Toast.makeText(this, "啊,你点中我了 " + position, Toast.LENGTH_SHORT).show();

}


// 模拟数据

private String[] mStringArray = { "Abbaye de Belloc"/*

* ,

* "Abbaye du Mont des Cats"

* , "Abertam",

* "Abondance",

* "Ackawi", "Acorn",

* "Adelost",

* "Affidelice au Chablis"

* , "Afuega'l Pitu",

* "Airag", "Airedale",

* "Aisy Cendre",

* "Allgauer Emmentaler"

* , "Alverca",

* "Ambert",

* "American Cheese"

*/};


private void loadData() {

new Thread(new Runnable() {


@Override

public void run() {

try {

Thread.sleep(0000);

} catch (InterruptedException e) {

e.printStackTrace();

}

List<String> strings = new ArrayList<String>();

for (String body : mStringArray) {

strings.add(body);

}

Message msg = mUIHandler.obtainMessage(WHAT_DID_LOAD_DATA);

msg.obj = strings;

msg.sendToTarget();

}

}).start();

}

}

4、零碎小东西

资源文件

pull_down_head.xml

<?xml version="1.0" encoding="utf-8"?>

<!-- ListView的头部 -->


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    android:layout_width="fill_parent"

    android:layout_height="wrap_content" >


    <!-- 内容 -->


    <RelativeLayout

        android:id="@+id/head_contentLayout"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:paddingLeft="30dp" >


        <!-- 箭头图像、进度条 -->


        <FrameLayout

            android:layout_width="wrap_content"

            android:layout_height="wrap_content"

            android:layout_alignParentLeft="true"

            android:layout_centerVertical="true" >


            <!-- 箭头 -->


            <ImageView

                android:id="@+id/head_arrowImageView"

                android:layout_width="wrap_content"

                android:layout_height="wrap_content"

                android:layout_gravity="center"

                android:src="@drawable/pull_down_arrow" />


            <!-- 进度条 -->


            <ProgressBar

                android:id="@+id/head_progressBar"

                style="?android:attr/progressBarStyleSmall"

                android:layout_width="wrap_content"

                android:layout_height="wrap_content"

                android:layout_gravity="center"

                android:visibility="gone" />

        </FrameLayout>


        <!-- 提示、最近更新 -->


        <LinearLayout

            android:layout_width="wrap_content"

            android:layout_height="wrap_content"

            android:layout_centerHorizontal="true"

            android:gravity="center_horizontal"

            android:orientation="vertical" >


            <!-- 提示 -->


            <TextView

                android:id="@+id/head_tipsTextView"

                android:layout_width="wrap_content"

                android:layout_height="wrap_content"

                android:text="下拉刷新"

                android:textColor="#000000"

                android:textSize="20sp" />


            <!-- 最近更新 -->


            <TextView

                android:id="@+id/head_lastUpdatedTextView"

                android:layout_width="wrap_content"

                android:layout_height="wrap_content"

                android:text="上次更新"

                android:textColor="#000000"

                android:textSize="10sp" />

        </LinearLayout>

    </RelativeLayout>


</LinearLayout>

pull_down_footer.xml


<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"

    android:layout_width="fill_parent"

    android:layout_height="wrap_content"

    android:paddingBottom="10dp"

    android:paddingTop="10dp" >


    <TextView

        android:id="@+id/pulldown_footer_text"

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:layout_centerInParent="true"

        android:text="更多"

        android:textSize="15dp" />


    <ProgressBar

        android:id="@+id/pulldown_footer_loading"

        style="@android :style/Widget.ProgressBar.Small.Inverse"

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:layout_centerVertical="true"

        android:layout_marginLeft="20dp"

        android:visibility="gone" />


</RelativeLayout>


基本上的东西就是这么多了,使用起来也是相当方便的,如果有不同见解,欢迎一起交流

© 著作权归作者所有

共有 人打赏支持
董家二少
粉丝 3
博文 34
码字总数 85000
作品 0
海淀
程序员
私信 提问
加载中

评论(3)

董家二少
董家二少

引用来自“hyyz3293”的评论

加载更多是最后一页的话、哪里是如何写的呢?
其实现在这个方式可以不采用了,建议你看看Cube SDK ,我个人感觉这个写的相当不错
h
hyyz3293
加载更多是最后一页的话、哪里是如何写的呢?
weicai
weicai
参考过,,,写的还可以
Jaynm/PullToRefreshListView

PullToRefreshScrollViewDemo Android使用PullToRefresh完成ListView下拉刷新和左滑删除 一、本文主要内容: 使用PullToRefresh完成ListView下拉、上拉刷新; 扩展PullToRefresh完美的实现L...

Jaynm
2016/11/03
0
0
StickyListHeaders怎么实现上拉加载更多,下拉刷新?急

跟饿了么移动版的订餐一样,listview上拉向上滑,分类什么的刚好停在最顶部,可以实现上拉加载更多,下拉刷新。求源代码或方法,谢谢

koma25
2015/10/23
496
0
RecycleView相关

Android ItemTouchHelper 实践 实现 RecyclerView 拖动排序和滑动删除,我想到的是 ViewDragHelper ,或者是第三方库,当我看了 ToDoList 的时候,发现原来官方已经支持 RecyclerView 拖动排...

掘金官方
2017/12/25
0
0
那些年不容错过的智能下拉刷新加载框架

一些值得学习的几个下拉刷新上拉加载开源库 功能齐全的AnimRefreshRecyclerView 根据列表的不同效果选择不同的布局管理器 根据不同的布局管理器设置分割线: 设置Header和Footer 手动刷新 An...

codeGoogle
2017/09/26
0
0
重写ListView实现下拉刷新上拉加载

安卓本身的ListView没有刷新功能,要想实现这一功能,就得继承ListView并重写它的方法。同时也要实现其OnScrollListener这一接口。 下拉刷新,需要在原本的ListView上部添加一个Head,ListV...

ZytCandy
2014/08/07
0
4

没有更多内容

加载失败,请刷新页面

加载更多

day11

architect刘源源
27分钟前
2
0
论学好Linux系统的超级重要性

不知道各位在日常的工作生活中有没有接触过“rm -rf /*”这个命令,因为这个命令搞出来的事情可还不少呢!前段时间就在一个群里看到了有个小伙子,老板让他去维护一下服务器,这小伙也不太懂...

Linux就该这么学
昨天
1
0
git 使用

1,首先在github配置好信息和仓库,然后在本地进行操作 git init git config user.name 'zhangwuer' git config user.email '56789053@qq.com' 2,与远程分支建立连接 git checkout -b test......

天王盖地虎626
昨天
3
0
git checkout 命令详解

在日常的git操作中,git checkout——检出,是我们的常用命令。最为常用的两种情形是创建分支和切换分支。 在下面的命令中,使用了一些简写,在这里说明一下: git st # git statusgit ci ...

shzwork
昨天
8
0
【Nginx】Nginx多级代理,获取客户端真实请求IP以及每级代理IP

Nginx多级代理,获取客户端真实请求IP以及每级代理IP 如图所示,每一级nginx里的location配置里需要加上对应的配置,最后一级nginx是直接到应用,测试时为了方便,直接用echo模块去测试,打印...

薛定谔的旺
昨天
6
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部