2016-03-21 20 views
14

Mam ten kod dla widoku RecyclerView.Wykryj, kiedy RecyclerView osiąga najbardziej dolną pozycję podczas przewijania

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); 

Muszę wiedzieć, kiedy RecyclerView osiąga najniższe położenie podczas przewijania. Czy to możliwe ? Jeśli tak, w jaki sposób?

+0

Wypróbuj http://stackoverflow.com/questions/27841740/how-to-know-whether-a-recyclerview-linearlayoutmanager-is-scrolled-to-top-or- b – Raghavendra

+0

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

+1

recyclerView.canScrollVertically (int kierunek); Jaki powinien być parametr, który musimy tutaj przekazać? – Adrian

Odpowiedz

25

Wystarczy zaimplementować metodę addOnScrollListener() w widoku recyclerview. Następnie wewnątrz odbiornika przewijania zaimplementuj poniższy kod.

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

Czy możesz wyjaśnić trochę o mIsLoading, proszę? gdzie ustawiasz go lub zmieniasz wartość. – MiguelHincapieC

+0

mLoading jest po prostu zmienną typu boolean wskazującą, czy widok jest włączony, czy nie. Na przykład; jeśli aplikacja zapełnia recyrkulaturę, wówczas po zapisaniu listy mLoading stanie się prawdą i stanie się wartością false. –

+0

ohh ok, tankuje cię za odpowiedź. Chociaż miałam ważniejszą rolę xD. – MiguelHincapieC

3

Istnieje moje wdrożenie, jest bardzo przydatne dla StaggeredGridLayout.

Zastosowanie:

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

rvMain.addOnScrollListener(scrollListener); 

Listener realizacja:

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

Dlaczego dodać 200 milisekund opóźnienia do oddzwonienia? – Mani

+0

@Mani Nie pamiętam dokładnie do tej chwili, ale może zrobiłem to po prostu dla gładkiego efektu ... –

23

jest również prosty sposób to zrobić

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(); 

     } 
    } 
}); 

kierunkowych liczb całkowitych: -1 do góry , 1 w dół, 0 zawsze zwróci false.

+0

onScrolled() zapobiega dwukrotnemu wykryciu zdarzenia. –

+0

zobacz https://stackoverflow.com/a/48514857/6674369 –

1

Byłem też poszukiwania na to pytanie, ale nie mogę znaleźć odpowiedzi, które udowodniły mi, więc tworzę własną realizację 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) 
} 

potem w swojej działalności/fragmentu:

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

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

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

mam nadzieję, że ktoś go rozwiąże ;-)

Powiązane problemy