版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
【移動(dòng)應(yīng)用開發(fā)技術(shù)】App列表之下拉刷新
Android的ListView是應(yīng)用最廣的一個(gè)組件,功能強(qiáng)大,擴(kuò)展性靈活(不局限于ListView本身一個(gè)類),前面的文章有介紹分組,拖拽,3D立體,游標(biāo),圓角,而今天我們要介紹的是另外一個(gè)擴(kuò)展ListView:下拉刷新的ListView。
下拉刷新界面最初流行于iphone應(yīng)用界面,如圖:
然后在Android中也逐漸被應(yīng)用,比如微博,資訊類。
所以,今天要實(shí)現(xiàn)的結(jié)果應(yīng)該也是類似的,先貼出最終完成效果,如下圖,接下來我們一步一步實(shí)現(xiàn)。1.流程分析
下拉刷新最主要的流程是:
(1).下拉,顯示提示頭部界面(HeaderView),這個(gè)過程提示用戶"下拉刷新"
(2).下拉到一定程度,超出了刷新最基本的下拉界限,我們認(rèn)為達(dá)到了刷新的條件,提示用戶可以"松手刷新"了,效果上允許用戶繼續(xù)下拉
(3).用戶松手,可能用戶下拉遠(yuǎn)遠(yuǎn)不止提示頭部界面,所以這一步,先反彈回僅顯示提示頭部界面,然后提示用戶"正在加載"。
(4).加載完成后,隱藏提示頭部界面。
示意圖如下:2.實(shí)現(xiàn)分析
當(dāng)前我們要實(shí)現(xiàn)上述流程,是基于ListView的,所以對(duì)應(yīng)ListView本身的功能我們來分析一下實(shí)現(xiàn)原理:
(1).下拉,顯示提示頭部界面,這個(gè)過程提示用戶"下拉刷新"
a.下拉的操作,首先是監(jiān)聽滾動(dòng),ListView提供了onScroll()方法
b.與下拉類似一個(gè)動(dòng)作向下飛滑,所以ListView的scrollState有3種值:SCROLL_STATE_IDLE,
SCROLL_STATE_TOUCH_SCROLL,
SCROLL_STATE_FLING,意思容易理解,而我們要下拉的觸發(fā)條件是SCROLL_STATE_TOUCH_SCROLL。判斷當(dāng)前的下拉操作狀態(tài),ListView提供了publicvoidonScrollStateChanged(AbsListViewview,intscrollState){}。
c.下拉的過程中,我們可能還需要下拉到多少的邊界值處理,重寫onTouchEvent(MotionEventev){}方法,可依據(jù)ACTION_DOWN,ACTION_MOVE,ACTION_UP實(shí)現(xiàn)更精細(xì)的判斷。
(2).下拉到一定程度,超出了刷新最基本的下拉界限,我們認(rèn)為達(dá)到了刷新的條件,提示用戶可以"松手刷新"了,效果上允許用戶繼續(xù)下拉
a.達(dá)到下拉刷新界限,一般指達(dá)到header的高度的,所以有兩步,第一,獲取header的高度,第二,當(dāng)header.getBottom()>=header的高度時(shí),我們認(rèn)為就達(dá)到了刷新界限值
b.繼續(xù)允許用戶下拉,當(dāng)header完全下拉后,默認(rèn)無法繼續(xù)下拉,但是可以增加header的PaddingTop實(shí)現(xiàn)這種效果
(3).用戶松手,可能用戶下拉遠(yuǎn)遠(yuǎn)不止提示頭部界面,所以這一步,先反彈回僅顯示提示頭部界面,然后提示用戶"正在加載"。
a.松手后反彈,這個(gè)不能一下***回去,看上去太突然,需要一步一步柔性的彈回去,像彈簧一樣,我們可以new一個(gè)Thread循環(huán)計(jì)算減少PaddingTop,直到PaddingTop為0,反彈結(jié)束。
b.正在加載,在子線程里處理后臺(tái)任務(wù)
(4).加載完成后,隱藏提示頭部界面。
a.后臺(tái)任務(wù)完成后,我們需要隱藏header,setSelection(1)即實(shí)現(xiàn)了從第2項(xiàng)開始顯示,間接隱藏了header。上面我們分析了實(shí)現(xiàn)過程的輪廓,接下來,通過細(xì)節(jié)說明和代碼具體實(shí)現(xiàn)。3.初始化
一切狀態(tài)顯示都是用HeaderView顯示的,所以我們需要一個(gè)HeaderView的layout,使用addHeaderView方法添加到ListView中。
同時(shí),默認(rèn)狀態(tài)下,HeaderView是不顯示的,只是在下拉后才顯示,所以我們需要隱藏HeaderView且不影響后續(xù)的下拉顯示,用setSelection(1)。
refresh_list_header.xml布局如下:<?xml
version="1.0"
encoding="utf-8"?>
<LinearLayout
xmlns:android="/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:gravity="center">
<ProgressBar
android:id="@+id/refresh_list_header_progressbar"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:visibility="gone">
</ProgressBar>
<ImageView
android:id="@+id/refresh_list_header_pull_down"
android:layout_width="9dip"
android:layout_height="25dip"
android:layout_gravity="center"
android:src="@drawable/refresh_list_pull_down"
/>
<ImageView
android:id="@+id/refresh_list_header_release_up"
android:layout_width="9dip"
android:layout_height="25dip"
android:layout_gravity="center"
android:src="@drawable/refresh_list_release_up"
android:visibility="gone"
/>
<RelativeLayout
android:layout_width="180dip"
android:layout_height="wrap_content">
<TextView
android:id="@+id/refresh_list_header_text"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:layout_alignParentTop="true"
android:textSize="12dip"
android:textColor="#192F06"
android:paddingTop="8dip"
android:text="@string/app_list_header_refresh_down"/>
<TextView
android:id="@+id/refresh_list_header_last_update"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:layout_below="@id/refresh_list_header_text"
android:textSize="12dip"
android:textColor="#192F06"
android:paddingBottom="8dip"
android:text="@string/app_list_header_refresh_last_update"/>
</RelativeLayout>
</LinearLayout>
代碼中在構(gòu)造函數(shù)中添加init()方法加載如下:
private
LinearLayout
mHeaderLinearLayout
=
null;
private
TextView
mHeaderTextView
=
null;
private
TextView
mHeaderUpdateText
=
null;
private
ImageView
mHeaderPullDownImageView
=
null;
private
ImageView
mHeaderReleaseDownImageView
=
null;
private
ProgressBar
mHeaderProgressBar
=
null;
public
RefreshListView(Context
context)
{
this(context,
null);
}
public
RefreshListView(Context
context,
AttributeSet
attrs)
{
super(context,
attrs);
init(context);
}
void
init(final
Context
context)
{
mHeaderLinearLayout
=
(LinearLayout)
LayoutInflater.from(context).inflate(R.layout.refresh_list_header,
null);
addHeaderView(mHeaderLinearLayout);
mHeaderTextView
=
(TextView)
findViewById(R.id.refresh_list_header_text);
mHeaderUpdateText
=
(TextView)
findViewById(R.id.refresh_list_header_last_update);
mHeaderPullDownImageView
=
(ImageView)
findViewById(R.id.refresh_list_header_pull_down);
mHeaderReleaseDownImageView
=
(ImageView)
findViewById(R.id.refresh_list_header_release_up);
mHeaderProgressBar
=
(ProgressBar)
findViewById(R.id.refresh_list_header_progressbar);
setSelection(1);
}
默認(rèn)就顯示完成了。4.HeaderView的默認(rèn)高度測(cè)量
因?yàn)橄吕紿eaderView全部顯示出來,就由提示"下拉刷新"變?yōu)?松手刷新",全部顯示的出來的測(cè)量標(biāo)準(zhǔn)就是header.getBottom()>=header的高度。
所以,首先我們需要測(cè)量HeaderView的默認(rèn)高度。
//因?yàn)槭窃跇?gòu)造函數(shù)里測(cè)量高度,應(yīng)該先measure一下
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);
}
然后在init的上述代碼后面加上調(diào)用measureView后,使用getMeasureHeight()方法獲取header的高度:
private
int
mHeaderHeight;
void
init(final
Context
context)
{
...
...
measureView(mHeaderLinearLayout);
mHeaderHeight
=
mHeaderLinearLayout.getMeasuredHeight();
}后面我們就會(huì)用到這個(gè)mHeaderHeight.5.scrollState監(jiān)聽記錄
scrollState有3種,使用onScrollStateChanged()方法監(jiān)聽記錄。
private
int
mCurrentScrollState;
@Override
public
void
onScrollStateChanged(AbsListView
view,
int
scrollState)
{
mCurrentScrollState
=
scrollState;
}
然后即可使用mCurrentScrollState作為后面判斷的條件了。6.刷新狀態(tài)分析
因?yàn)橐恍┑胤叫枰牢覀兲幵谡顟B(tài)下還是進(jìn)入下拉刷新狀態(tài)還是松手反彈狀態(tài),比如,
(1).在非正常的狀態(tài)下,我們不小心飛滑了一下(松手的瞬間容易出現(xiàn)這種情況),我們不能setSelection(1)的,否則總是松手后header跳的一下消失掉了。
(2).下拉后要做一個(gè)下拉效果的特殊處理,需要用到OVER_PULL_REFRESH(松手刷新狀態(tài)下)
(3).松手反彈后要做一個(gè)反彈效果的特殊處理,需要用到OVER_PULL_REFRESH和ENTER_PULL_REFRESH。
private
final
static
int
NONE_PULL_REFRESH
=
0;
//正常狀態(tài)
private
final
static
int
ENTER_PULL_REFRESH
=
1;
//進(jìn)入下拉刷新狀態(tài)
private
final
static
int
OVER_PULL_REFRESH
=
2;
//進(jìn)入松手刷新狀態(tài)
private
final
static
int
EXIT_PULL_REFRESH
=
3;
//松手后反彈后加載狀態(tài)
private
int
mPullRefreshState
=
0;
//記錄刷新狀態(tài)
@Override
public
void
onScroll(AbsListView
view,
int
firstVisibleItem,
int
visibleItemCount,
int
totalItemCount)
{
if
(mCurrentScrollState
==SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
==
0
&&
(mHeaderLinearLayout.getBottom()
>=
0
&&
mHeaderLinearLayout.getBottom()
<
mHeaderHeight))
{
//進(jìn)入且僅進(jìn)入下拉刷新狀態(tài)
if
(mPullRefreshState
==
NONE_PULL_REFRESH)
{
mPullRefreshState
=
ENTER_PULL_REFRESH;
}
}
else
if
(mCurrentScrollState
==SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
==
0
&&
(mHeaderLinearLayout.getBottom()
>=
mHeaderHeight))
{
//下拉達(dá)到界限,進(jìn)入松手刷新狀態(tài)
if
(mPullRefreshState
==
ENTER_PULL_REFRESH
||
mPullRefreshState
==
NONE_PULL_REFRESH)
{
mPullRefreshState
=
OVER_PULL_REFRESH;
//下面是進(jìn)入松手刷新狀態(tài)需要做的一個(gè)顯示改變
mDownY
=
mMoveY;//用于后面的下拉特殊效果
mHeaderTextView.setText("松手刷新");
mHeaderPullDownImageView.setVisibility(View.GONE);
mHeaderReleaseDownImageView.setVisibility(View.VISIBLE);
}
}
else
if
(mCurrentScrollState
==SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
!=
0)
{
//不刷新了
if
(mPullRefreshState
==
ENTER_PULL_REFRESH)
{
mPullRefreshState
=
NONE_PULL_REFRESH;
}
}
else
if
(mCurrentScrollState
==
SCROLL_STATE_FLING
&&
firstVisibleItem
==
0)
{
//飛滑狀態(tài),不能顯示出header,也不能影響正常的飛滑
//只在正常情況下才糾正位置
if
(mPullRefreshState
==
NONE_PULL_REFRESH)
{
setSelection(1);
}
}
}mPullRefreshState將是后面我們處理邊界的重要變量。6.下拉效果的特殊處理
所謂的特殊處理,當(dāng)header完全顯示后,下拉只按下拉1/3的距離下拉,給用戶一種艱難下拉,該松手的彈簧感覺。
這個(gè)在onTouchEvent里處理比較方便:
private
float
mDownY;
private
float
mMoveY;
@Override
public
boolean
onTouchEvent(MotionEvent
ev)
{
switch
(ev.getAction())
{
case
MotionEvent.ACTION_DOWN:
//記下按下位置
//改變
mDownY
=
ev.getY();
break;
case
MotionEvent.ACTION_MOVE:
//移動(dòng)時(shí)手指的位置
mMoveY
=
ev.getY();
if
(mPullRefreshState
==
OVER_PULL_REFRESH)
{
//注意下面的mDownY在onScroll的第二個(gè)else中被改變了
mHeaderLinearLayout.setPadding(mHeaderLinearLayout.getPaddingLeft(),
(int)((mMoveY
-
mDownY)/3),
//1/3距離折扣
mHeaderLinearLayout.getPaddingRight(),
mHeaderLinearLayout.getPaddingBottom());
}
break;
case
MotionEvent.ACTION_UP:
...
...
break;
}
return
super.onTouchEvent(ev);
}
//重復(fù)貼出下面這段需要注意的代碼
@Override
public
void
onScroll(AbsListView
view,
int
firstVisibleItem,
int
visibleItemCount,
int
totalItemCount)
{
...
...
else
if
(mCurrentScrollState
==
SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
==
0
&&
(mHeaderLinearLayout.getBottom()
>=
mHeaderHeight))
{
//下拉達(dá)到界限,進(jìn)入松手刷新狀態(tài)
if
(mPullRefreshState
==
ENTER_PULL_REFRESH
||
mPullRefreshState
==
NONE_PULL_REFRESH)
{
mPullRefreshState
=
OVER_PULL_REFRESH;
mDownY
=
mMoveY;
//為下拉1/3折扣效果記錄開始位置
mHeaderTextView.setText("松手刷新");//顯示松手刷新
mHeaderPullDownImageView.setVisibility(View.GONE);//隱藏"下拉刷新"
mHeaderReleaseDownImageView.setVisibility(View.VISIBLE);//顯示向上的箭頭
}
}
...
...
}onScroll里監(jiān)聽到了進(jìn)入松手刷新狀態(tài),onTouchEvent就開始在ACTION_MOVE中處理1/3折扣問題。7.反彈效果的特殊處理
松手后我們需要一個(gè)柔性的反彈效果,意味著我們彈回去的過程需要分一步步走,我的解決方案是:
在子線程里計(jì)算PaddingTop,并減少到原來的3/4,循環(huán)通知主線程,直到PaddingTop小于1(這個(gè)值取一個(gè)小值,合適即可)。
松手后,當(dāng)然是在onTouchEvent的ACTION_UP條件下處理比較方便:
//因?yàn)樯婕暗絟andler數(shù)據(jù)處理,為方便我們定義如下常量
private
final
static
int
REFRESH_BACKING
=
0;
//反彈中
private
final
static
int
REFRESH_BACED
=
1;
//達(dá)到刷新界限,反彈結(jié)束后
private
final
static
int
REFRESH_RETURN
=
2;
//沒有達(dá)到刷新界限,返回
private
final
static
int
REFRESH_DONE
=
3;
//加載數(shù)據(jù)結(jié)束
@Override
public
boolean
onTouchEvent(MotionEvent
ev)
{
switch
(ev.getAction())
{
...
...
case
MotionEvent.ACTION_UP:
//when
you
action
up,
it
will
do
these:
//1.
roll
back
util
header
topPadding
is
0
//2.
hide
the
header
by
setSelection(1)
if
(mPullRefreshState
==
OVER_PULL_REFRESH
||
mPullRefreshState
==
ENTER_PULL_REFRESH)
{
new
Thread()
{
public
void
run()
{
Message
msg;
while(mHeaderLinearLayout.getPaddingTop()
>
1)
{
msg
=
mHandler.obtainMessage();
msg.what
=
REFRESH_BACKING;
mHandler.sendMessage(msg);
try
{
sleep(5);//慢一點(diǎn)反彈,別一下子就彈回去了
}
catch
(InterruptedException
e)
{
e.printStackTrace();
}
}
msg
=
mHandler.obtainMessage();
if
(mPullRefreshState
==
OVER_PULL_REFRESH)
{
msg.what
=
REFRESH_BACED;//加載數(shù)據(jù)完成,結(jié)束返回
}
else
{
msg.what
=
REFRESH_RETURN;//未達(dá)到刷新界限,直接返回
}
mHandler.sendMessage(msg);
};
}.start();
}
break;
}
return
super.onTouchEvent(ev);
}
private
Handler
mHandler
=
new
Handler(){
@Override
public
void
handleMessage(Message
msg)
{
switch
(msg.what)
{
case
REFRESH_BACKING:
mHeaderLinearLayout.setPadding(mHeaderLinearLayout.getPaddingLeft(),
(int)
(mHeaderLinearLayout.getPaddingTop()*0.75f),
mHeaderLinearLayout.getPaddingRight(),
mHeaderLinearLayout.getPaddingBottom());
break;
case
REFRESH_BACED:
mHeaderTextView.setText("正在加載...");
mHeaderProgressBar.setVisibility(View.VISIBLE);
mHeaderPullDownImageView.setVisibility(View.GONE);
mHeaderReleaseDownImageView.setVisibility(View.GONE);
mPullRefreshState
=
EXIT_PULL_REFRESH;
new
Thread()
{
public
void
run()
{
sleep(2000);//處理后臺(tái)加載數(shù)據(jù)
Message
msg
=
mHandler.obtainMessage();
msg.what
=
REFRESH_DONE;
//通知主線程加載數(shù)據(jù)完成
mHandler.sendMessage(msg);
};
}.start();
break;
case
REFRESH_RETURN:
//未達(dá)到刷新界限,返回
mHeaderTextView.setText("下拉刷新");
mHeaderProgressBar.setVisibility(View.INVISIBLE);
mHeaderPullDownImageView.setVisibility(View.VISIBLE);
mHeaderReleaseDownImageView.setVisibility(View.GONE);
mHeaderLinearLayout.setPadding(mHeaderLinearLayout.getPaddingLeft(),
0,
mHeaderLinearLayout.getPaddingRight(),
mHeaderLinearLayout.getPaddingBottom());
mPullRefreshState
=
NONE_PULL_REFRESH;
setSelection(1);
break;
case
REFRESH_DONE:
//刷新結(jié)束后,恢復(fù)原始默認(rèn)狀態(tài)
mHeaderTextView.setText("下拉刷新");
mHeaderProgressBar.setVisibility(View.INVISIBLE);
mHeaderPullDownImageView.setVisibility(View.VISIBLE);
mHeaderReleaseDownImageView.setVisibility(View.GONE);
mHeaderUpdateText.setText(getContext().getString(R.string.app_list_header_refresh_last_update,
mSimpleDateFormat.format(new
Date())));
mHeaderLinearLayout.setPadding(mHeaderLinearLayout.getPaddingLeft(),
0,
mHeaderLinearLayout.getPaddingRight(),
mHeaderLinearLayout.getPaddingBottom());
mPullRefreshState
=
NONE_PULL_REFRESH;
setSelection(1);
break;
default:
break;
}
}
};
為了一下子看的明確,我把效果中的數(shù)據(jù)處理代碼也貼出來了。8.切入數(shù)據(jù)加載過程
上面數(shù)據(jù)后臺(tái)處理我們用sleep(2000)來處理,實(shí)際處理中,作為公共組件,我們也不好把具體代碼直接寫在這里,我們需要一個(gè)更靈活的分離:
(1).定義接口
(2).注入接口
//定義接口
public
interface
RefreshListener
{
Object
refreshing();
//加載數(shù)據(jù)
void
refreshed(Object
obj);
//外部可擴(kuò)展加載完成后的操作
}
//注入接口
private
Object
mRefreshObject
=
null;
//傳值
private
RefreshListener
mRefreshListener
=
null;
public
void
setOnRefreshListener(RefreshListener
refreshListener)
{
this.mRefreshListener
=
refreshListener;
}
//我們需要重寫上面的mHandler如下代碼
case
REFRESH_BACED:
...
...
new
Thread()
{
public
void
run()
{
if
(mRefreshListener
!=
null)
{
mRefreshObject
=
mRefreshListener.refreshing();
}
Message
msg
=
mHandler.obtainMessage();
msg.what
=
REFRESH_DONE;
mHandler.sendMessage(msg);
};
}.start();
break;
case
REFRESH_DONE:
...
...
mPullRefreshState
=
NONE_PULL_REFRESH;
setSelection(1);
if
(mRefreshListener
!=
null)
{
mRefreshListener.refreshed(mRefreshObject);
}
break;
在其他地方我們就可以不修改這個(gè)listview組件的代碼,使用如下:public
xxx
implements
RefreshListener{
@Override
protected
void
onCreate(Bundle
savedInstanceState)
{
super.onCreate(savedInstanceState);
//類似如下
((RefreshListView)
listView).setOnRefreshListener(this);
}
@Override
public
Object
refreshing()
{
String
result
=
null;
//result
=
FileUtils.readTextFile(file);
return
result;
}
@Override
public
void
refreshed(Object
obj)
{
if
(obj
!=
null)
{
//擴(kuò)展操作
}
};
}很方便了。9.擴(kuò)展"更多"功能
下拉刷新之外,我們也可以通過相同方法使用FooterView切入底部"更多"過程,這里我就不詳細(xì)說明了10.源碼
上面的每段代碼都看做是"零部件",需要組合一下。
因?yàn)槲覀兩厦鎸?shí)現(xiàn)了下拉刷新,還增加了"更多"功能,我們直接命名這個(gè)類為RefreshListView吧:package
com.tianxia.lib.baseworld.widget;
import
java.text.SimpleDateFormat;
import
java.util.Date;
import
android.content.Context;
import
android.os.Handler;
import
android.os.Message;
import
android.util.AttributeSet;
import
android.view.LayoutInflater;
import
android.view.MotionEvent;
import
android.view.View;
import
android.view.ViewGroup;
import
android.widget.AbsListView;
import
android.widget.AbsListView.OnScrollListener;
import
android.widget.ImageView;
import
android.widget.LinearLayout;
import
android.widget.ListAdapter;
import
android.widget.ListView;
import
android.widget.ProgressBar;
import
android.widget.TextView;
import
com.tianxia.lib.baseworld.R;
/**
*
下拉刷新,底部更多
*
*/
public
class
RefreshListView
extends
ListView
implements
OnScrollListener{
private
float
mDownY;
private
float
mMoveY;
private
int
mHeaderHeight;
private
int
mCurrentScrollState;
private
final
static
int
NONE_PULL_REFRESH
=
0;
//正常狀態(tài)
private
final
static
int
ENTER_PULL_REFRESH
=
1;
//進(jìn)入下拉刷新狀態(tài)
private
final
static
int
OVER_PULL_REFRESH
=
2;
//進(jìn)入松手刷新狀態(tài)
private
final
static
int
EXIT_PULL_REFRESH
=
3;
//松手后反彈和加載狀態(tài)
private
int
mPullRefreshState
=
0;
//記錄刷新狀態(tài)
private
final
static
int
REFRESH_BACKING
=
0;
//反彈中
private
final
static
int
REFRESH_BACED
=
1;
//達(dá)到刷新界限,反彈結(jié)束后
private
final
static
int
REFRESH_RETURN
=
2;
//沒有達(dá)到刷新界限,返回
private
final
static
int
REFRESH_DONE
=
3;
//加載數(shù)據(jù)結(jié)束
private
LinearLayout
mHeaderLinearLayout
=
null;
private
LinearLayout
mFooterLinearLayout
=
null;
private
TextView
mHeaderTextView
=
null;
private
TextView
mHeaderUpdateText
=
null;
private
ImageView
mHeaderPullDownImageView
=
null;
private
ImageView
mHeaderReleaseDownImageView
=
null;
private
ProgressBar
mHeaderProgressBar
=
null;
private
TextView
mFooterTextView
=
null;
private
ProgressBar
mFooterProgressBar
=
null;
private
SimpleDateFormat
mSimpleDateFormat;
private
Object
mRefreshObject
=
null;
private
RefreshListener
mRefreshListener
=
null;
public
void
setOnRefreshListener(RefreshListener
refreshListener)
{
this.mRefreshListener
=
refreshListener;
}
public
RefreshListView(Context
context)
{
this(context,
null);
}
public
RefreshListView(Context
context,
AttributeSet
attrs)
{
super(context,
attrs);
init(context);
}
void
init(final
Context
context)
{
mHeaderLinearLayout
=
(LinearLayout)
LayoutInflater.from(context).inflate(R.layout.refresh_list_header,
null);
addHeaderView(mHeaderLinearLayout);
mHeaderTextView
=
(TextView)
findViewById(R.id.refresh_list_header_text);
mHeaderUpdateText
=
(TextView)
findViewById(R.id.refresh_list_header_last_update);
mHeaderPullDownImageView
=
(ImageView)
findViewById(R.id.refresh_list_header_pull_down);
mHeaderReleaseDownImageView
=
(ImageView)
findViewById(R.id.refresh_list_header_release_up);
mHeaderProgressBar
=
(ProgressBar)
findViewById(R.id.refresh_list_header_progressbar);
mFooterLinearLayout
=
(LinearLayout)
LayoutInflater.from(context).inflate(R.layout.refresh_list_footer,
null);
addFooterView(mFooterLinearLayout);
mFooterProgressBar
=
(ProgressBar)
findViewById(R.id.refresh_list_footer_progressbar);
mFooterTextView
=
(TextView)
mFooterLinearLayout.findViewById(R.id.refresh_list_footer_text);
mFooterLinearLayout.setOnClickListener(new
OnClickListener()
{
@Override
public
void
onClick(View
v)
{
if
(context.getString(R.string.app_list_footer_more).equals(mFooterTextView.getText()))
{
mFooterTextView.setText(R.string.app_list_footer_loading);
mFooterProgressBar.setVisibility(View.VISIBLE);
if
(mRefreshListener
!=
null)
{
mRefreshListener.more();
}
}
}
});
setSelection(1);
setOnScrollListener(this);
measureView(mHeaderLinearLayout);
mHeaderHeight
=
mHeaderLinearLayout.getMeasuredHeight();
mSimpleDateFormat
=
new
SimpleDateFormat("yyyy-MM-dd
hh:mm");
mHeaderUpdateText.setText(context.getString(R.string.app_list_header_refresh_last_update,
mSimpleDateFormat.format(new
Date())));
}
@Override
public
boolean
onTouchEvent(MotionEvent
ev)
{
switch
(ev.getAction())
{
case
MotionEvent.ACTION_DOWN:
mDownY
=
ev.getY();
break;
case
MotionEvent.ACTION_MOVE:
mMoveY
=
ev.getY();
if
(mPullRefreshState
==
OVER_PULL_REFRESH)
{
mHeaderLinearLayout.setPadding(mHeaderLinearLayout.getPaddingLeft(),
(int)((mMoveY
-
mDownY)/3),
mHeaderLinearLayout.getPaddingRight(),
mHeaderLinearLayout.getPaddingBottom());
}
break;
case
MotionEvent.ACTION_UP:
//when
you
action
up,
it
will
do
these:
//1.
roll
back
util
header
topPadding
is
0
//2.
hide
the
header
by
setSelection(1)
if
(mPullRefreshState
==
OVER_PULL_REFRESH
||
mPullRefreshState
==
ENTER_PULL_REFRESH)
{
new
Thread()
{
public
void
run()
{
Message
msg;
while(mHeaderLinearLayout.getPaddingTop()
>
1)
{
msg
=
mHandler.obtainMessage();
msg.what
=
REFRESH_BACKING;
mHandler.sendMessage(msg);
try
{
sleep(5);
}
catch
(InterruptedException
e)
{
e.printStackTrace();
}
}
msg
=
mHandler.obtainMessage();
if
(mPullRefreshState
==
OVER_PULL_REFRESH)
{
msg.what
=
REFRESH_BACED;
}
else
{
msg.what
=
REFRESH_RETURN;
}
mHandler.sendMessage(msg);
};
}.start();
}
break;
}
return
super.onTouchEvent(ev);
}
@Override
public
void
onScroll(AbsListView
view,
int
firstVisibleItem,
int
visibleItemCount,
int
totalItemCount)
{
if
(mCurrentScrollState
==
SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
==
0
&&
(mHeaderLinearLayout.getBottom()
>=
0
&&
mHeaderLinearLayout.getBottom()
<
mHeaderHeight))
{
//進(jìn)入且僅進(jìn)入下拉刷新狀態(tài)
if
(mPullRefreshState
==
NONE_PULL_REFRESH)
{
mPullRefreshState
=
ENTER_PULL_REFRESH;
}
}
else
if
(mCurrentScrollState
==
SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
==
0
&&
(mHeaderLinearLayout.getBottom()
>=
mHeaderHeight))
{
//下拉達(dá)到界限,進(jìn)入松手刷新狀態(tài)
if
(mPullRefreshState
==
ENTER_PULL_REFRESH
||
mPullRefreshState
==
NONE_PULL_REFRESH)
{
mPullRefreshState
=
OVER_PULL_REFRESH;
mDownY
=
mMoveY;
//為下拉1/3折扣效果記錄開始位置
mHeaderTextView.setText("松手刷新");//顯示松手刷新
mHeaderPullDownImageView.setVisibility(View.GONE);//隱藏"下拉刷新"
mHeaderReleaseDownImageView.setVisibility(View.VISIBLE);//顯示向上的箭頭
}
}
else
if
(mCurrentScrollState
==
SCROLL_STATE_TOUCH_SCROLL
&&
firstVisibleItem
!=
0)
{
//不刷新了
if
(mPullRefreshState
==
ENTER_PULL_REFRESH)
{
mPullRefreshState
=
NONE_PULL_REFRESH;
}
}
else
if
(mCurrentScrollState
==
SCROLL_STATE_FLING
&&
firstVisibleItem
==
0)
{
//飛滑狀態(tài),不能顯示出header,也不能影響正常的飛滑
//只在正常情況下才糾正位置
if
(mPullRefreshState
==
NONE_PULL_REFRESH)
{
setSelection(1);
}
}
}
@Override
public
void
onScrollStateChanged(AbsListView
view,
int
scrollState)
{
mCurrentScrollState
=
scrollState;
}
@Override
public
void
setAdapter(ListAdapter
adapter)
{
super.setAdapter(adapter);
setSelection(1);
}
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);
}
private
Handler
mHandler
=
new
Handler(){
@Override
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 二零二五年個(gè)人住房抵押借款協(xié)議(家庭財(cái)富傳承)3篇
- 2025版新能源汽車充電站建設(shè)合同含政府補(bǔ)貼及稅收優(yōu)惠條款
- 2025年二零二五班主任學(xué)生社會(huì)實(shí)踐與志愿服務(wù)合同3篇
- 2024年環(huán)保型圍墻材料采購及安裝合作協(xié)議3篇
- 2025版高效節(jié)能型10KV線路及變臺(tái)安裝工程施工及驗(yàn)收合同2篇
- 2025版游樂場(chǎng)游樂設(shè)施檢測(cè)合同3篇
- 2023年全腦開發(fā)項(xiàng)目融資計(jì)劃書
- 2024年版:高等教育機(jī)構(gòu)實(shí)驗(yàn)室設(shè)備管理合同
- 2025年度智慧城市建設(shè)合同范本3篇
- 2025版科技研發(fā)中心租賃承包經(jīng)營服務(wù)協(xié)議3篇
- 國家開放大學(xué)法學(xué)本科《商法》歷年期末考試試題及答案題庫
- 金匱要略知到智慧樹章節(jié)測(cè)試課后答案2024年秋浙江中醫(yī)藥大學(xué)
- 【MOOC】有機(jī)化學(xué)實(shí)驗(yàn)-南京工業(yè)大學(xué) 中國大學(xué)慕課MOOC答案
- 2024年婦??乒ぷ骺偨Y(jié)及計(jì)劃
- 北京理工大學(xué)《數(shù)據(jù)結(jié)構(gòu)與算法設(shè)計(jì)》2022-2023學(xué)年第一學(xué)期期末試卷
- 錨桿(索)支護(hù)工技能理論考試題庫200題(含答案)
- 影視后期制作團(tuán)隊(duì)薪酬激勵(lì)方案
- 人教版2024年小學(xué)二年級(jí)上學(xué)期語文期末考試往年真題
- 新版高中物理必做實(shí)驗(yàn)?zāi)夸浖捌鞑?(電子版)
- 中國慢性冠脈綜合征患者診斷及管理指南2024版解讀
- (正式版)SHT 3551-2024 石油化工儀表工程施工及驗(yàn)收規(guī)范
評(píng)論
0/150
提交評(píng)論