RecyclerView下拉刷新與上拉更多

前言

在原來的文章中我提及了如何使用RecyclerView添加headerfooter游昼,今天我們來更深入的擴展一下使用RecyclerView實現(xiàn)常用的下拉刷新與上拉加載更多的功能。當然這些功能的實現(xiàn)也是基于前面的RecyclerView添加header與footer為基礎來實現(xiàn)的尝蠕,不是很了解的可以先看看前面的文章可能能更好的幫助理解烘豌。

依賴

為了方法大家的使用我已經(jīng)把他上傳到Jcenter中了,所以大家可以調(diào)用下面的代碼了直接獲取使用:

compile 'com.idisfkj.enchancerecyclerview:mylibrary:1.1.1'

EnhanceRecyclerView

我將這個擴展的RecyclerView命名為EnhanceRecyclerView看彼,繼承RecyclerView廊佩。我們知道既然要實現(xiàn)下拉刷新與上拉更多自然先要實現(xiàn)頭部與尾部的布局,所以我們先利用前面的知識來為EnhanceRecycleView添加headerfooter

public void initView() {
        View headerView = LayoutInflater.from(getContext()).inflate(R.layout.head_layout, null);
        View footerView = LayoutInflater.from(getContext()).inflate(R.layout.footer_layout, null);
        addHeaderView(headerView);
        addFooterView(footerView);
    }

其中的布局文件就不多說了靖榕,至于addHeaderView與addFooterView方法可以查看我前面的那篇文章标锄,有詳細的介紹

設置監(jiān)聽器

既然要實現(xiàn)下拉刷新與上拉加載,自然少不了對監(jiān)聽器的處理序矩,所以下面來詳細介紹下對監(jiān)聽器OnScrollListenerOnTouchListener的處理鸯绿。

OnScrollListener

EnhanceRecyclerView添加addOnScrollListener實現(xiàn)其中的onScrollStateChangedonScrolled方法。

onScrolled

onScrolled中我們主要做的是獲取EnhanceRcyclerViewitem的總數(shù)量簸淀、視圖顯示中的第一個itemEnhanceRecyclerView中所處的位置與視圖顯示中最后一個itemEnhanceRecyclerView中所處的位置瓶蝴。

對于item的總數(shù)量很好獲取直接調(diào)用

totalCount = getLayoutManager().getItemCount();

由于RecyclerView能實現(xiàn)LinearLayoutManagerGridLayoutManagerStaggeredGridLayoutManager不同的布局租幕,所以另外兩個要根據(jù)不同的manager來獲取,還是看具體代碼吧

if (getLayoutManager() instanceof LinearLayoutManager) {
                    lastItem = ((LinearLayoutManager) getLayoutManager()).findLastVisibleItemPosition();
                    firstVisible = ((LinearLayoutManager) getLayoutManager()).findFirstVisibleItemPosition();
                } else {
                    into = ((StaggeredGridLayoutManager) getLayoutManager()).findLastVisibleItemPositions(into);
                    firstInto = ((StaggeredGridLayoutManager) getLayoutManager()).findFirstVisibleItemPositions(firstInto);
                    lastItem = into[0];
                    firstVisible = firstInto[0];
                }

onScrollStateChanged

獲取到了那三個關鍵數(shù)據(jù)以后舷手,就可以在onScrollStateChanged中實現(xiàn)具體的邏輯,在這個方法中主要實現(xiàn)的是對上拉加載更多的處理

if (lastItem == adapter.getItemCount() + 1 && newState == RecyclerView.SCROLL_STATE_IDLE && !isLoad) {
                    ViewGroup.LayoutParams params = getFooterView(0).getLayoutParams();
                    params.width = RecyclerView.LayoutParams.MATCH_PARENT;
                    params.height = RecyclerView.LayoutParams.WRAP_CONTENT;
                    getFooterView(0).setLayoutParams(params);
                    getFooterView(0).setVisibility(View.VISIBLE);
                    smoothScrollToPosition(totalCount);
                    isLoad = true;
                    loadMoreListener.onLoadMore();
                }
                if (firstVisible == 0) {
                    isTop = true;
                } else {
                    isTop = false;
                    RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) getHeaderView(0).getLayoutParams();
                    params.width = RecyclerView.LayoutParams.MATCH_PARENT;
                    params.height = RecyclerView.LayoutParams.WRAP_CONTENT;
                    params.setMargins(0, -getHeaderView(0).getHeight(), 0, 0);
                    getHeaderView(0).setLayoutParams(params);
                }

簡單說明下劲绪,核心就是判斷lastItem是否處在最后的位置男窟,如果是的話就繼續(xù)加載更多的操作盆赤,這里提供了一個對數(shù)據(jù)處理的接口所以只要實現(xiàn)loadMoreListener.onLoadMore();即可。

上拉加載更多核心就是這么多歉眷,其它的可以查看源碼

OnTouchListener

這個監(jiān)聽器主要是對下拉刷新進行處理牺六。我們要分別對其中我們所熟悉的MotionEvent.ACTION_DOWNMotionEvent.ACTION_MOVEMotionEvent.ACTION_UP進行處理汗捡。ACTION_DOWN就是簡單的獲取按下的坐標位置淑际,這里就不多說了,下面主要的針對另外的兩個進行簡單說明扇住。

ACTION_MOVE

這做的邏輯就是對觸摸后的處理春缕,根據(jù)滑動的距離來動態(tài)的改變header的文本與布局視圖的顯示。

public void touchMove(MotionEvent event) {
        endY = event.getY();
        moveY = endY - startY;
        //防止item向上滑出
        if (moveY > 0 && !isRefreshing) {
            //防止回退文本顯示異常
            scrollToPosition(0);

            if (getHeaderView(0).getVisibility() == GONE)
                getHeaderView(0).setVisibility(VISIBLE);

            RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) getHeaderView(0).getLayoutParams();
            params.width = RecyclerView.LayoutParams.MATCH_PARENT;
            params.height = RecyclerView.LayoutParams.WRAP_CONTENT;
            //使header隨moveY的值從頂部漸漸出現(xiàn)
            if (moveY >= 400) {
                moveY = 100 + moveY / 4;
            } else {
                moveY = moveY / 2;
            }
            viewHeight = getHeaderView(0).getHeight();
            if (viewHeight <= 0)
                viewHeight = 130;
            moveY = moveY - viewHeight;
            params.setMargins(0, (int) moveY, 0, 0);
            getHeaderView(0).setLayoutParams(params);
            if (moveY > 80) {
                text.setText(getResources().getString(R.string.release_to_refresh));
            } else {
                text.setText(getResources().getString(R.string.pull_to_refresh));
            }
        } else {
            if (getHeaderView(0).getVisibility() != GONE && !isRefreshing) {
                getHeaderView(0).setVisibility(GONE);
            }
        }
    }

至于下拉時與頂部的距離變化是通過設置margin來動態(tài)改變的艘蹋。

ACTION_UP

最后的觸摸處理就是在離開屏幕時根據(jù)滑動的距離锄贼,是否調(diào)用加載數(shù)據(jù)的接口,或者隱藏下拉刷新頭部女阀,具體還是看代碼吧宅荤。

public void touchUp() {
        if (!isRefreshing && (endY -startY) != 0 ) {

            RecyclerView.LayoutParams params1 = (RecyclerView.LayoutParams) getHeaderView(0).getLayoutParams();
            params1.width = RecyclerView.LayoutParams.MATCH_PARENT;
            params1.height = RecyclerView.LayoutParams.WRAP_CONTENT;

            if (moveY >= 80) {
                text.setText(getResources().getString(R.string.refreshing));
                params1.setMargins(0, 0, 0, 0);
                isRefreshing = true;
                //刷新數(shù)據(jù)
                pullToRefresh.onRefreshing();
            } else {
                if (viewHeight <= 0)
                    viewHeight = 130;
                params1.setMargins(0, -viewHeight, 0, 0);
                getHeaderView(0).setVisibility(GONE);
            }
            getHeaderView(0).setLayoutParams(params1);
        }
    }

代碼中重要的地方都有指出相信都能看懂,這樣下拉與上拉的邏輯就基本實現(xiàn)了强品,下面來看接口的設計吧

下拉與上拉接口

public interface PullToRefreshListener {
        void onRefreshing();
    }

    public void setPullToRefreshListener(PullToRefreshListener pullToRefresh) {
        if (loadMoreListener == null) {
            initListener();
        }
        this.pullToRefresh = pullToRefresh;
    }

    public interface LoadMoreListener {
        void onLoadMore();
    }

    public void setLoadMoreListener(LoadMoreListener loadMoreListener) {
        if (pullToRefresh == null) {
            initListener();
        }
        this.loadMoreListener = loadMoreListener;
    }

在運用是添加接口監(jiān)聽時初始化前面為EnhanceRecyclerView所設置的監(jiān)聽膘侮。

狀態(tài)重置設置

在調(diào)用下拉刷新或者上拉加載更多之后屈糊,我們?yōu)槠錁嬙焱ㄓ梅椒▽崿F(xiàn)的榛,狀態(tài)的重置與數(shù)據(jù)的更新,方便統(tǒng)一調(diào)用逻锐。

 public void setLoadMoreComplete() {
        RecyclerView.LayoutParams params = (LayoutParams) getFooterView(0).getLayoutParams();
        params.width = 0;
        params.height = 0;
        getFooterView(0).setLayoutParams(params);
        getFooterView(0).setVisibility(View.GONE);
        this.getAdapter().notifyDataSetChanged();
        isLoad = false;
    }

    public void setRefreshComplete() {
        RecyclerView.LayoutParams params1 = (RecyclerView.LayoutParams) getHeaderView(0).getLayoutParams();
        params1.width = RecyclerView.LayoutParams.MATCH_PARENT;
        params1.height = RecyclerView.LayoutParams.WRAP_CONTENT;
        params1.setMargins(0, -getHeaderView(0).getHeight(), 0, 0);
        getHeaderView(0).setLayoutParams(params1);
        getHeaderView(0).setVisibility(GONE);
        this.getAdapter().notifyDataSetChanged();
        isRefreshing = false;
    }

所用工作已經(jīng)完成下面來做個調(diào)用示范

使用

xml中引用

<com.idisfkj.mylibrary.EnhanceRecyclerView
        android:id="@+id/recyclerView"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
</com.idisfkj.mylibrary.EnhanceRecyclerView>

設置監(jiān)聽

 mRecyclerView.setPullToRefreshListener(new com.idisfkj.mylibrary.EnhanceRecyclerView.PullToRefreshListener() {
            @Override
            public void onRefreshing() {
                refreshData();
            }
        });
        mRecyclerView.setLoadMoreListener(new EnhanceRecyclerView.LoadMoreListener() {
            @Override
            public void onLoadMore() {
                loadMoreData();
            }
        });

refreshData()loadMoreData()加載數(shù)據(jù)的邏輯就不展示了夫晌,只是要記住在請求網(wǎng)絡數(shù)據(jù)完之后要在他們中調(diào)用相應的mRecyclerView.setRefreshComplete()mRecyclerView.setLoadMoreComplete()來重置狀態(tài)。

至于其他的Adapter昧诱、LayoutManager等的設置就不多說了晓淀,與原生的RecyclerView是一樣的。

效果

效果圖

總結

其實總的來說難點有兩個

  • 添加headerfooter盏档。這個前面已經(jīng)攻克了凶掰,而且原理也相對簡單
  • 實現(xiàn)觸摸與滑動監(jiān)聽邏輯。這個主要是對邏輯的理解蜈亩,對整個刷新的過程做個整體分析懦窘,就能很好的理解上面的代碼。對其中視圖的動態(tài)顯示做相應的變化與接口的調(diào)用就能很好的處理這些工程稚配。

當然上面的實現(xiàn)可能還有瑕疵畅涂,希望指出,我會相應的做修改或者你們修改后可以提交給我道川,我統(tǒng)一做修改午衰,謝謝立宜!

項目地址:https://github.com/idisfkj/EnhanceRecyclerView

關注

怪談時間到了
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市臊岸,隨后出現(xiàn)的幾起案子橙数,更是在濱河造成了極大的恐慌,老刑警劉巖帅戒,帶你破解...
    沈念sama閱讀 222,627評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件商模,死亡現(xiàn)場離奇詭異,居然都是意外死亡蜘澜,警方通過查閱死者的電腦和手機施流,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鄙信,“玉大人瞪醋,你說我怎么就攤上這事∽肮睿” “怎么了银受?”我有些...
    開封第一講書人閱讀 169,346評論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長鸦采。 經(jīng)常有香客問我宾巍,道長,這世上最難降的妖魔是什么渔伯? 我笑而不...
    開封第一講書人閱讀 60,097評論 1 300
  • 正文 為了忘掉前任顶霞,我火速辦了婚禮,結果婚禮上锣吼,老公的妹妹穿的比我還像新娘选浑。我一直安慰自己,他們只是感情好玄叠,可當我...
    茶點故事閱讀 69,100評論 6 398
  • 文/花漫 我一把揭開白布古徒。 她就那樣靜靜地躺著,像睡著了一般读恃。 火紅的嫁衣襯著肌膚如雪隧膘。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,696評論 1 312
  • 那天寺惫,我揣著相機與錄音疹吃,去河邊找鬼。 笑死肌蜻,一個胖子當著我的面吹牛互墓,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蒋搜,決...
    沈念sama閱讀 41,165評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼篡撵,長吁一口氣:“原來是場噩夢啊……” “哼判莉!你這毒婦竟也來了?” 一聲冷哼從身側響起育谬,我...
    開封第一講書人閱讀 40,108評論 0 277
  • 序言:老撾萬榮一對情侶失蹤券盅,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后膛檀,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體锰镀,經(jīng)...
    沈念sama閱讀 46,646評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,709評論 3 342
  • 正文 我和宋清朗相戀三年咖刃,在試婚紗的時候發(fā)現(xiàn)自己被綠了泳炉。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,861評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡嚎杨,死狀恐怖花鹅,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情枫浙,我是刑警寧澤刨肃,帶...
    沈念sama閱讀 36,527評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站箩帚,受9級特大地震影響真友,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜紧帕,卻給世界環(huán)境...
    茶點故事閱讀 42,196評論 3 336
  • 文/蒙蒙 一盔然、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧焕参,春花似錦轻纪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽潦嘶。三九已至涩嚣,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間掂僵,已是汗流浹背航厚。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留锰蓬,地道東北人幔睬。 一個月前我還...
    沈念sama閱讀 49,287評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像芹扭,于是被迫代替她去往敵國和親麻顶。 傳聞我的和親對象是個殘疾皇子赦抖,可洞房花燭夜當晚...
    茶點故事閱讀 45,860評論 2 361

推薦閱讀更多精彩內(nèi)容