ListView下拉刷新,上拉加載

代碼結構,主要是兩個類,實現頭部功能的封裝,寫出一個自定義控件

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>

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章