Android系統上拉刷新功能的實現

dw23 9年前發布 | 1K 次閱讀 Java Android

自定義ListView:

public class MyListView extends ListView implements OnScrollListener {

public final int RATIO = 5;// 手拖動時,手指拖動距離與header下移的倍數
private final int LOADING = 0;// 正在加載
private final int RELEASE_To_REFRESH = 2;// 正在下拉狀態,即將松手進行刷新
private final int DONE = 3;// 完成
private final int HIDE_To_REFRESH = 1;// 未下拉或已完成狀態,設置padding為負值把header擋住

private int refreshStatus;// 下拉刷新過程狀態變量

private LinearLayout hearderLayout;
private ImageView arrowImageView;
private ProgressBar progressBar;
private TextView tipsTextview;
private TextView lastUpdatedTime;
private int headContentHeight;
private int headContentWidth;
private RotateAnimation animation;
private RotateAnimation reverseAnimation;
private boolean isStartRefreshEvent = false;// 下拉刷新事件是否已經開始
private boolean isRefreshable = false;// 是否設置下拉刷新監聽
private int firstItemIndex;

private OnRefreshListener refreshListener;

public MyListView (Context context) {
    super(context);
    init(context);
}

public MyListView (Context context, AttributeSet attrs) {
    super(context, attrs);
    init(context);
}

private void init(Context context) {

    LayoutInflater inflater = LayoutInflater.from(context);
    hearderLayout = (LinearLayout) inflater.inflate(R.layout.list_view_header, null);
    // 下拉箭頭
    arrowImageView = (ImageView) hearderLayout.findViewById(R.id.head_arrowImageView);
    // 進度條
    progressBar = (ProgressBar) hearderLayout.findViewById(R.id.head_progressBar);
    // 下拉提示 刷新
    tipsTextview = (TextView) hearderLayout.findViewById(R.id.head_tipsTextView);
    // 最新一次刷新時間
    lastUpdatedTime = (TextView) hearderLayout.findViewById(R.id.head_lastUpdatedTextView);
    // 計算head的高寬
    this.measureView(hearderLayout);
    headContentHeight = hearderLayout.getMeasuredHeight();
    headContentWidth = hearderLayout.getMeasuredWidth();

    // 初始狀態是 隱藏掉head 布局
    hearderLayout.setPadding(0, -1 * headContentHeight, 0, 0);
    hearderLayout.invalidate();

    Log.e("sys", "width:" + headContentWidth + " height:" + headContentHeight);
    // list添加頭文件
    addHeaderView(hearderLayout, null, false);
    setOnScrollListener(this);// TODO
    // 下拉以及恢復動畫
    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);
    refreshStatus = DONE;
}

@Override
public void onScroll(AbsListView arg0, int firstVisiableItem, int arg2, int arg3) {
    firstItemIndex = firstVisiableItem;
}

@Override
public void onScrollStateChanged(AbsListView arg0, int arg1) {
}

private int startY = 0;
@Override
public boolean onTouchEvent(MotionEvent event) {

    if (firstItemIndex == 0 && isRefreshable) {
        switch (event.getAction()) {
        // 在down時候記錄當前Y的位置
        case MotionEvent.ACTION_DOWN:
            if (!isStartRefreshEvent) {
                isStartRefreshEvent = true;
                startY = (int) event.getY();
            }
            break;

        case MotionEvent.ACTION_MOVE:

            int tempY = (int) event.getY();
            int scrollY = (tempY - startY) / RATIO;

            Log.i("sys", "拉開方向 " + scrollY);

            if (!isStartRefreshEvent) {
                isStartRefreshEvent = true;
                startY = tempY;
            } else if (isStartRefreshEvent && refreshStatus != LOADING) {
                // 可以松手去刷新了
                if (refreshStatus == RELEASE_To_REFRESH) {
                    setSelection(0);
                    // header被用戶拉下來后,又往上推了,推到了已經小于header高度時,則推回去,并取消刷新
                    if (scrollY < headContentHeight && scrollY > 0) {
                        refreshStatus = HIDE_To_REFRESH;
                        this.changeHeaderView();
                    }
                    // 一下子推到頂了
                    else if (scrollY <= 0) {
                        refreshStatus = DONE;
                        this.changeHeaderView();
                    }
                }
                // 還沒有到達顯示松開刷新的時候,DONE或者是HIDE_To_REFRESH狀態
                if (refreshStatus == HIDE_To_REFRESH) {
                    setSelection(0);
                    // 下拉至超過header本身高度后,可以進入RELEASE_TO_REFRESH的狀態
                    if (scrollY > headContentHeight) {
                        refreshStatus = RELEASE_To_REFRESH;
                        this.changeHeaderView();
                    }
                    // 上推到頂了
                    else if (scrollY <= 0) {
                        refreshStatus = DONE;
                        this.changeHeaderView();
                    }
                }
                // done狀態下
                if (refreshStatus == DONE) {
                    if (scrollY > 0) {
                        refreshStatus = HIDE_To_REFRESH;
                        this.changeHeaderView();
                    }
                }
                // 更新headView的size
                if (refreshStatus == HIDE_To_REFRESH) {
                    hearderLayout.setPadding(0, -1 * headContentHeight + scrollY, 0, 0);
                }
                // 更新headView的paddingTop
                if (refreshStatus == RELEASE_To_REFRESH) {
                    hearderLayout.setPadding(0, scrollY - headContentHeight, 0, 0);
                }
            }
            break;

        case MotionEvent.ACTION_UP:
            if (refreshStatus != LOADING) {
                if (refreshStatus == DONE) {
                    // 什么都不做
                }
                // 由下拉刷新狀態,到done狀態
                if (refreshStatus == HIDE_To_REFRESH) {
                    refreshStatus = DONE;
                    this.changeHeaderView();
                }

                if (refreshStatus == RELEASE_To_REFRESH) {
                    refreshStatus = LOADING;
                    this.changeHeaderView();
                    onRefresh();
                }
            }
            isStartRefreshEvent = false;
            break;

        }
    }
    return super.onTouchEvent(event);
}

private void changeHeaderView() {
    switch (refreshStatus) {
    // 松開刷新狀態
    case RELEASE_To_REFRESH:
        arrowImageView.setVisibility(View.VISIBLE);
        progressBar.setVisibility(View.GONE);
        tipsTextview.setVisibility(View.VISIBLE);
        lastUpdatedTime.setVisibility(View.VISIBLE);
        arrowImageView.clearAnimation();
        arrowImageView.startAnimation(animation);
        tipsTextview.setText("松開刷新");
        break;
    // 下拉刷新
    case HIDE_To_REFRESH:
        progressBar.setVisibility(View.GONE);
        tipsTextview.setVisibility(View.VISIBLE);
        lastUpdatedTime.setVisibility(View.VISIBLE);
        arrowImageView.setVisibility(View.VISIBLE);
        // 是由RELEASE_To_REFRESH狀態轉變來的
        arrowImageView.clearAnimation();
        arrowImageView.startAnimation(reverseAnimation);
        tipsTextview.setText("下拉刷新");
        break;
    // 刷新中 狀態
    case LOADING:
        hearderLayout.setPadding(0, 0, 0, 0);
        progressBar.setVisibility(View.VISIBLE);
        arrowImageView.clearAnimation();
        arrowImageView.setVisibility(View.GONE);
        tipsTextview.setText("正在刷新...");
        lastUpdatedTime.setVisibility(View.VISIBLE);
        break;
    // 刷新完畢
    case DONE:
        hearderLayout.setPadding(0, -1 * headContentHeight, 0, 0);
        progressBar.setVisibility(View.GONE);
        arrowImageView.clearAnimation();
        // arrowImageView.setImageResource(R.drawable.goicon);
        tipsTextview.setText("下拉刷新");
        lastUpdatedTime.setVisibility(View.VISIBLE);
        break;
    }
}

public void setonRefreshListener(OnRefreshListener refreshListener) {
    this.refreshListener = refreshListener;
    isRefreshable = true;
}

public interface OnRefreshListener {
    public void onRefresh();
}

public void onRefreshFinished() {
    refreshStatus = DONE;
    DateFormat format = new SimpleDateFormat("yyyy年MM月dd日  HH:mm");
    String date = format.format(new Date());
    lastUpdatedTime.setText("最近更新:" + date);
    changeHeaderView();
}

private void onRefresh() {
    if (refreshListener != null) {
        refreshListener.onRefresh();
    }
}

// 此方法計算高度
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 setAdapter(BaseAdapter adapter) {
    DateFormat format = new SimpleDateFormat("yyyy年MM月dd日  HH:mm");
    String date = format.format(new Date());
    lastUpdatedTime.setText("最近更新:" + date);
    super.setAdapter(adapter);
}

} </pre>
layout:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&quot;
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:background="#ffffff" >

<!-- 內容 -->

<RelativeLayout
    android:id="@+id/head_contentLayout"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:paddingLeft="30dip" >

    <!-- 箭頭圖像、進度條 -->
    <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:src="@drawable/arrow"
            android:layout_gravity="center" />

        <!-- 進度條,style集合了一個anim -->
        <ProgressBar
            android:id="@+id/head_progressBar"
            style="@style/wait_anim"
            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:textSize="15dp" />

        <!-- 最近更新 -->
        <TextView
            android:id="@+id/head_lastUpdatedTextView"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="上次更新"
            android:textSize="12dp" />
    </LinearLayout>
</RelativeLayout>

</LinearLayout>

Activity中關鍵代碼:

mListView.setonRefreshListener(new OnRefreshListener() { public void onRefresh() { new AsyncTask<Void, Void, Void>() { protected Void doInBackground(Void... params) { try { Thread.sleep(3000); } catch (Exception e) { e.printStackTrace(); } // dates.addFirst(new AttachmentModule(new Date(), WorkLogContentType.WORKLOG_ATTACHMENT, AttachmentType.Attachment_Nomal)); return null; }

                @Override
                protected void onPostExecute(Void result) {
                    adapter.notifyDataSetChanged();
                    mListView.onRefreshFinished();
                }
            }.execute(null);
        }
    });</pre> 


 本文由用戶 dw23 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!