2016-03-21 14 views
14

Ich habe diesen Code für einen RecyclerView.Erkennen, wenn RecyclerView beim Scrollen die unterste Position erreicht

recyclerView = (RecyclerView)rootview.findViewById(R.id.fabric_recyclerView); 
    recyclerView.setLayoutManager(layoutManager); 
    recyclerView.addItemDecoration(new RV_Item_Spacing(5)); 
    FabricAdapter fabricAdapter=new FabricAdapter(ViewAdsCollection.getFabricAdsDetailsAsArray()); 
    recyclerView.setAdapter(fabricAdapter); 

Ich muss wissen, wenn der RecyclerView beim Scrollen die unterste Position erreicht. Ist es möglich ? Wenn ja, wie?

+0

Versuchen http://StackOverflow.com/Questions/27841740/how-to-know-whether-a-recyclerview-linearlayoutmanager-is-scrolled-to-top-oder- b – Raghavendra

+0

http: // sta ckoverflow.com/questions/26543131/how-to-implement-endless-list-with-recyclerview –

+1

recyclerView.canScrollVertically (int-Richtung); Was sollte der Parameter sein, den wir hier passieren müssen? – Adrian

Antwort

25

Implementieren Sie einfach eine addOnScrollListener() in Ihrem Recyclerview. Drehe dann innerhalb des Scroll-Listeners den folgenden Code.

RecyclerView.OnScrollListener mScrollListener = new RecyclerView.OnScrollListener() { 
     @Override 
     public void onScrolled(RecyclerView recyclerView, int dx, int dy) { 
      if (mIsLoading) 
       return; 
      int visibleItemCount = mLayoutManager.getChildCount(); 
      int totalItemCount = mLayoutManager.getItemCount(); 
      int pastVisibleItems = mLayoutManager.findFirstVisibleItemPosition(); 
      if (pastVisibleItems + visibleItemCount >= totalItemCount) { 
       //End of list 
      } 
     } 
    }; 
+1

Können Sie bitte etwas über mIsLoading erklären? Wo stellst du es auf oder änderst den Wert? – MiguelHincapieC

+0

mLoading ist nur eine boolesche Variable, die angibt, ob die Ansicht aktiviert ist oder nicht. Für zB; Wenn App die Recycler-Ansicht auffüllt, ist mLoading wahr und wird falsch, sobald die Liste ausgefüllt wurde. –

+0

ohh ok, Tanks Sie für die Antwort. Ich hatte zwar eine wichtigere Rolle xD. – MiguelHincapieC

3

Es gibt meine Implementierung, es ist sehr nützlich für StaggeredGridLayout.

Verbrauch:

private EndlessScrollListener scrollListener = 
     new EndlessScrollListener(new EndlessScrollListener.RefreshList() { 
      @Override public void onRefresh(int pageNumber) { 
       //end of the list 
      } 
     }); 

rvMain.addOnScrollListener(scrollListener); 

Listener Implementierung:

class EndlessScrollListener extends RecyclerView.OnScrollListener { 
private boolean isLoading; 
private boolean hasMorePages; 
private int pageNumber = 0; 
private RefreshList refreshList; 
private boolean isRefreshing; 
private int pastVisibleItems; 

EndlessScrollListener(RefreshList refreshList) { 
    this.isLoading = false; 
    this.hasMorePages = true; 
    this.refreshList = refreshList; 
} 

@Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { 
    super.onScrolled(recyclerView, dx, dy); 
    StaggeredGridLayoutManager manager = 
      (StaggeredGridLayoutManager) recyclerView.getLayoutManager(); 

    int visibleItemCount = manager.getChildCount(); 
    int totalItemCount = manager.getItemCount(); 
    int[] firstVisibleItems = manager.findFirstVisibleItemPositions(null); 
    if (firstVisibleItems != null && firstVisibleItems.length > 0) { 
     pastVisibleItems = firstVisibleItems[0]; 
    } 

    if (visibleItemCount + pastVisibleItems >= totalItemCount && !isLoading) { 
     isLoading = true; 
     if (hasMorePages && !isRefreshing) { 
      isRefreshing = true; 
      new Handler().postDelayed(new Runnable() { 
       @Override public void run() { 
        refreshList.onRefresh(pageNumber); 
       } 
      }, 200); 
     } 
    } else { 
     isLoading = false; 
    } 
} 

public void noMorePages() { 
    this.hasMorePages = false; 
} 

void notifyMorePages() { 
    isRefreshing = false; 
    pageNumber = pageNumber + 1; 
} 

interface RefreshList { 
    void onRefresh(int pageNumber); 
} } 
+0

Warum 200 Millisekunden Verzögerung Callback hinzufügen? – Mani

+0

@Mani Ich erinnere mich nicht genau von diesem Moment, aber vielleicht habe ich es nur für einen reibungslosen Effekt getan ... –

23

gibt es auch eine einfache Möglichkeit, es

rv_convo.addOnScrollListener(new RecyclerView.OnScrollListener() { 
    @Override 
    public void onScrollStateChanged(RecyclerView recyclerView, int newState) { 
     super.onScrollStateChanged(recyclerView, newState); 

     if (!recyclerView.canScrollVertically(1)) { 
      Toast.makeText(ChatDetailsActivity.this,"LAst",Toast.LENGTH_LONG).show(); 

     } 
    } 
}); 

Richtung ganzen Zahlen zu tun: -1 bis , 1 für unten, 0 gibt immer false zurück.

+0

onScrolled() verhindert die Erkennung geschieht zweimal. –

+0

werfen Sie einen Blick auf https://Stackoverflow.com/a/48514857/6674369 –

1

ich auch für diese Frage war auf der Suche, aber ich finde nicht die Antwort, die mich erfüllt, so dass ich schaffen eigene Realisierung recyclerView:

class CustomRecyclerView: RecyclerView{ 

    abstract class TopAndBottomListener{ 
     open fun onBottomNow(onBottomNow:Boolean){} 
     open fun onTopNow(onTopNow:Boolean){} 
    } 


    constructor(c:Context):this(c, null) 
    constructor(c:Context, attr:AttributeSet?):super(c, attr, 0) 
    constructor(c:Context, attr:AttributeSet?, defStyle:Int):super(c, attr, defStyle) 


    private var linearLayoutManager:LinearLayoutManager? = null 
    private var topAndBottomListener:TopAndBottomListener? = null 
    private var onBottomNow = false 
    private var onTopNow = false 
    private var onBottomTopScrollListener:RecyclerView.OnScrollListener? = null 


    fun setTopAndBottomListener(l:TopAndBottomListener?){ 
     if (l != null){ 
      checkLayoutManager() 

      onBottomTopScrollListener = createBottomAndTopScrollListener() 
      addOnScrollListener(onBottomTopScrollListener) 
      topAndBottomListener = l 
     } else { 
      removeOnScrollListener(onBottomTopScrollListener) 
      topAndBottomListener = null 
     } 
    } 

    private fun createBottomAndTopScrollListener() = object :RecyclerView.OnScrollListener(){ 
     override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) { 
      checkOnTop() 
      checkOnBottom() 
     } 
    } 

    private fun checkOnTop(){ 
     val firstVisible = linearLayoutManager!!.findFirstCompletelyVisibleItemPosition() 
     if(firstVisible == 0 || firstVisible == -1 && !canScrollToTop()){ 
      if (!onTopNow) { 
       onTopNow = true 
       topAndBottomListener?.onTopNow(true) 
      } 
     } else if (onTopNow){ 
      onTopNow = false 
      topAndBottomListener?.onTopNow(false) 
     } 
    } 

    private fun checkOnBottom(){ 
     var lastVisible = linearLayoutManager!!.findLastCompletelyVisibleItemPosition() 
     val size = linearLayoutManager!!.itemCount - 1 
     if(lastVisible == size || lastVisible == -1 && !canScrollToBottom()){ 
      if (!onBottomNow){ 
       onBottomNow = true 
       topAndBottomListener?.onBottomNow(true) 
      } 
     } else if(onBottomNow){ 
      onBottomNow = false 
      topAndBottomListener?.onBottomNow(false) 
     } 
    } 


    private fun checkLayoutManager(){ 
     if (layoutManager is LinearLayoutManager) 
      linearLayoutManager = layoutManager as LinearLayoutManager 
     else 
      throw Exception("for using this listener, please set LinearLayoutManager") 
    } 

    private fun canScrollToTop():Boolean = canScrollVertically(-1) 
    private fun canScrollToBottom():Boolean = canScrollVertically(1) 
} 

dann in Ihrer Aktivität/Fragment:

override fun onCreate() { 
    customRecyclerView.layoutManager = LinearLayoutManager(context) 
} 

override fun onResume() { 
    super.onResume() 
    customRecyclerView.setTopAndBottomListener(this) 
} 

override fun onStop() { 
    super.onStop() 
    customRecyclerView.setTopAndBottomListener(null) 
} 

hoffe, es wird hepl jemand ;-)

Verwandte Themen