話很少說直接代碼android
代碼結構,主要是兩個類,實現頭部功能的封裝,寫出一個自定義控件
安全
一、頭部封裝類
網絡
/**ide
* 該類主要是完成 頭部部分的功能封裝佈局
* 測試
* 一個能夠監聽ListView是否滾動到最頂部或最底部的自定義控件 只能監聽由觸摸產生的,若是是ListView自己Flying致使的,則不能監聽</br>動畫
* 若是加以改進,能夠實現監聽scroll滾動的具體位置等this
* spa
* @author 進.net
*/
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;
}
}
}
二、要寫個自定義控件把這個給加載進去
/**
* 下拉刷新控件 真正實現下拉刷新的是這個控件, 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);
}
}
三、實際操做
代碼實例:
/**
* 下拉刷新上拉更多
*
* @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();
}
}
四、零碎小東西
資源文件
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>
基本上的東西就是這麼多了,使用起來也是至關方便的,若是有不一樣看法,歡迎一塊兒交流