ตรวจจับเมื่อ RecyclerView มาถึงตำแหน่งล่างสุดขณะที่เลื่อน


106

ฉันมีรหัสนี้สำหรับ RecyclerView

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

ฉันต้องการทราบเมื่อ RecyclerView มาถึงตำแหน่งล่างสุดขณะที่เลื่อน เป็นไปได้ไหม ? ถ้าใช่อย่างไร




1
RecyclerView.canScrollVertically (ทิศทาง int); พารามิเตอร์ที่เราต้องผ่านตรงนี้ควรเป็นอย่างไร
Adrian

@Adrian ในกรณีที่คุณยังสนใจคำถามนี้ :)))) stackoverflow.com/a/48514857/6674369
Andriy Antonov

คำตอบ:


186

นอกจากนี้ยังมีวิธีง่ายๆในการทำ

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

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

        }
    }
});

จำนวนเต็มทิศทาง: -1 สำหรับขึ้น 1 สำหรับลง 0 จะแสดงผลเท็จเสมอ


17
onScrolled () ป้องกันไม่ให้การตรวจจับเกิดขึ้นสองครั้ง
Ian Wambai

ฉันไม่สามารถเพิ่ม ScrollListnerEvent ใน My RecyclerView รหัสใน onScrollStateChanged ไม่ทำงาน
Sandeep Yohans

1
หากคุณต้องการเพียงสิ่งนี้เพื่อทริกเกอร์หนึ่งครั้ง (แสดง Toast ครั้งเดียว) เพิ่มบูลีนแฟล็ก (ตัวแปรสมาชิกคลาส) ในคำสั่ง if และตั้งค่าเป็น true ภายในหากต้องการ: if (!recyclerView.canScrollVertically(1) && !mHasReachedBottomOnce) { mHasReachedBottomOnce = true Toast.makeText(YourActivity.this, "Last", Toast.LENGTH_LONG).show();}
bastami82

@ bastami82 ฉันใช้เคล็ดลับตามที่คุณแนะนำเพื่อป้องกันไม่ให้พิมพ์ขนมปังสองครั้ง แต่ไม่ได้ผล
Vishwa Pratap

4
เพิ่มnewState == RecyclerView.SCROLL_STATE_IDLEเข้าไปในifคำสั่งจะได้ผล
Lee Chun Hoe

73

ใช้รหัสนี้เพื่อหลีกเลี่ยงการโทรซ้ำ

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

            if (!recyclerView.canScrollVertically(1) && newState==RecyclerView.SCROLL_STATE_IDLE) {
                Log.d("-----","end");
                
            }
        }
    });

2
ฉันลองแล้วอาจจะตอบได้ แต่มันง่ายและเหมาะสำหรับฉัน ขอบคุณ
Vishwa Pratap

@Venkatesh ยินดีต้อนรับ.
Milind Chaudhary

คำตอบเดียวที่ไม่ได้ให้ข้อบกพร่องอื่น ๆ แก่ฉัน
The Fluffy T Rex

@TheFluffyTRex ขอบคุณ
Milind Chaudhary

ระวัง! นอกจากนี้ยังจะทริกเกอร์เมื่อคุณพยายามปัดขึ้นในขณะที่มีองค์ประกอบไม่กี่อย่างในตัวรีไซเคิล นอกจากนี้ยังเป็นการกระตุ้น TWICE เกี่ยวกับสถานะใหม่ที่ถูกละเลย
Ammar Mohammad

48

เพียงใช้ addOnScrollListener () ในมุมมองรีไซเคิลของคุณ จากนั้นภายในตัวฟังการเลื่อนใช้โค้ดด้านล่าง

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
คุณช่วยอธิบายเล็กน้อยเกี่ยวกับ mIsLoading ได้ไหม คุณตั้งค่าหรือเปลี่ยนค่าที่ไหน
MiguelHincapieC

mLoading เป็นเพียงตัวแปรบูลีนที่ระบุว่ามุมมองมีส่วนร่วมหรือไม่ เช่น; หากแอปกำลังเติมข้อมูลมุมมองรีไซเคิล mLoading จะเป็นจริงและจะกลายเป็นเท็จเมื่อมีการเติมรายการ
Febi M Felix

1
โซลูชันนี้ทำงานไม่ถูกต้อง ดูที่stackoverflow.com/a/48514857/6674369
Andriy Antonov

3
ไม่สามารถแก้ไขวิธีการ'findFirstVisibleItemPosition'บนandroid.support.v7.widget.RecyclerView
Iman Marashi

2
@Iman Marashi คุณต้องหล่อ: (recyclerView.layoutManager as LinearLayoutManager).findFirstVisibleItemPosition(). งานนี้.
bitvale

17

คำตอบอยู่ใน Kotlin มันจะทำงานใน Java IntelliJ ควรแปลงให้คุณหากคุณคัดลอกและวาง

recyclerView.addOnScrollListener(object : RecyclerView.OnScrollListener(){

    override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {

        // 3 lines below are not needed.
        Log.d("TAG","Last visible item is: ${gridLayoutManager.findLastVisibleItemPosition()}")
        Log.d("TAG","Item count is: ${gridLayoutManager.itemCount}")
        Log.d("TAG","end? : ${gridLayoutManager.findLastVisibleItemPosition() == gridLayoutManager.itemCount-1}")

        if(gridLayoutManager.findLastVisibleItemPosition() == gridLayoutManager.itemCount-1){
            // We have reached the end of the recycler view.
        }

        super.onScrolled(recyclerView, dx, dy)
    }
})

สิ่งนี้จะใช้ได้เช่นกันLinearLayoutManagerเพราะมีวิธีการเดียวกันกับที่ใช้ข้างต้น ได้แก่findLastVisibleItemPosition()และgetItemCount()( itemCountใน Kotlin).


7
แนวทางที่ดี แต่ประเด็นหนึ่งคือข้อความใน if loop จะดำเนินการหลายครั้ง
วิษณุ

หากคุณได้รับการแก้ไขสำหรับปัญหาเดียวกัน
Vishwa Pratap

1
@ วิษณุตัวแปรบูลีนอย่างง่ายสามารถแก้ไขได้
daka

11

ฉันไม่ได้รับคำตอบที่สมบูรณ์แบบจากคำตอบข้างต้นเพราะมันถูกกระตุ้นสองครั้งแม้จะเปิดอยู่ก็ตาม onScrolled

override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
            super.onScrolled(recyclerView, dx, dy)
           if( !recyclerView.canScrollVertically(RecyclerView.FOCUS_DOWN))
               context?.toast("Scroll end reached")
        }

2
แต่recyclerView.canScrollVertically(direction)ทิศทางเป็นเพียงจำนวนเต็มใด ๆ + vs - ดังนั้นจึงเป็น 4 ได้ถ้าอยู่ล่างสุดหรือ -12 ถ้าอยู่ด้านบน
Xenolion

6

ลองสิ่งนี้

ฉันได้ใช้คำตอบข้างต้นมันทำงานเสมอเมื่อคุณไปที่จุดสิ้นสุดของมุมมองรีไซเคิล

หากต้องการตรวจสอบเพียงครั้งเดียวว่าอยู่ด้านล่างหรือไม่? ตัวอย่าง: - ถ้าฉันมีรายการ 10 รายการเมื่อใดก็ตามที่ฉันไปที่ด้านล่างมันจะแสดงฉันและอีกครั้งถ้าฉันเลื่อนจากบนลงล่างมันจะไม่พิมพ์อีกและถ้าคุณเพิ่มรายการมากขึ้นและคุณไปที่นั่นอีกครั้งมันจะแสดงอีกครั้ง

หมายเหตุ: -ใช้วิธีนี้เมื่อคุณจัดการกับoffsetในการกดปุ่ม API

  1. สร้างคลาสชื่อ EndlessRecyclerViewScrollListener

        import android.support.v7.widget.GridLayoutManager;
        import android.support.v7.widget.LinearLayoutManager;
        import android.support.v7.widget.RecyclerView;
        import android.support.v7.widget.StaggeredGridLayoutManager;
    
        public abstract class EndlessRecyclerViewScrollListener extends RecyclerView.OnScrollListener {
            // The minimum amount of items to have below your current scroll position
            // before loading more.
            private int visibleThreshold = 5;
            // The current offset index of data you have loaded
            private int currentPage = 0;
            // The total number of items in the dataset after the last load
            private int previousTotalItemCount = 0;
            // True if we are still waiting for the last set of data to load.
            private boolean loading = true;
            // Sets the starting page index
            private int startingPageIndex = 0;
    
            RecyclerView.LayoutManager mLayoutManager;
    
            public EndlessRecyclerViewScrollListener(LinearLayoutManager layoutManager) {
                this.mLayoutManager = layoutManager;
            }
    
        //    public EndlessRecyclerViewScrollListener() {
        //        this.mLayoutManager = layoutManager;
        //        visibleThreshold = visibleThreshold * layoutManager.getSpanCount();
        //    }
    
            public EndlessRecyclerViewScrollListener(StaggeredGridLayoutManager layoutManager) {
                this.mLayoutManager = layoutManager;
                visibleThreshold = visibleThreshold * layoutManager.getSpanCount();
            }
    
            public int getLastVisibleItem(int[] lastVisibleItemPositions) {
                int maxSize = 0;
                for (int i = 0; i < lastVisibleItemPositions.length; i++) {
                    if (i == 0) {
                        maxSize = lastVisibleItemPositions[i];
                    }
                    else if (lastVisibleItemPositions[i] > maxSize) {
                        maxSize = lastVisibleItemPositions[i];
                    }
                }
                return maxSize;
            }
    
            // This happens many times a second during a scroll, so be wary of the code you place here.
            // We are given a few useful parameters to help us work out if we need to load some more data,
            // but first we check if we are waiting for the previous load to finish.
            @Override
            public void onScrolled(RecyclerView view, int dx, int dy) {
                int lastVisibleItemPosition = 0;
                int totalItemCount = mLayoutManager.getItemCount();
    
                if (mLayoutManager instanceof StaggeredGridLayoutManager) {
                    int[] lastVisibleItemPositions = ((StaggeredGridLayoutManager) mLayoutManager).findLastVisibleItemPositions(null);
                    // get maximum element within the list
                    lastVisibleItemPosition = getLastVisibleItem(lastVisibleItemPositions);
                } else if (mLayoutManager instanceof GridLayoutManager) {
                    lastVisibleItemPosition = ((GridLayoutManager) mLayoutManager).findLastVisibleItemPosition();
                } else if (mLayoutManager instanceof LinearLayoutManager) {
                    lastVisibleItemPosition = ((LinearLayoutManager) mLayoutManager).findLastVisibleItemPosition();
                }
    
                // If the total item count is zero and the previous isn't, assume the
                // list is invalidated and should be reset back to initial state
                if (totalItemCount < previousTotalItemCount) {
                    this.currentPage = this.startingPageIndex;
                    this.previousTotalItemCount = totalItemCount;
                    if (totalItemCount == 0) {
                        this.loading = true;
                    }
                }
                // If it’s still loading, we check to see if the dataset count has
                // changed, if so we conclude it has finished loading and update the current page
                // number and total item count.
                if (loading && (totalItemCount > previousTotalItemCount)) {
                    loading = false;
                    previousTotalItemCount = totalItemCount;
                }
    
                // If it isn’t currently loading, we check to see if we have breached
                // the visibleThreshold and need to reload more data.
                // If we do need to reload some more data, we execute onLoadMore to fetch the data.
                // threshold should reflect how many total columns there are too
                if (!loading && (lastVisibleItemPosition + visibleThreshold) > totalItemCount) {
                    currentPage++;
                    onLoadMore(currentPage, totalItemCount, view);
                    loading = true;
                }
            }
    
            // Call this method whenever performing new searches
            public void resetState() {
                this.currentPage = this.startingPageIndex;
                this.previousTotalItemCount = 0;
                this.loading = true;
            }
    
            // Defines the process for actually loading more data based on page
            public abstract void onLoadMore(int page, int totalItemsCount, RecyclerView view);
    
        }
    
  2. ใช้คลาสนี้แบบนี้

         LinearLayoutManager linearLayoutManager = new LinearLayoutManager(getActivity());
            recyclerView.setLayoutManager(linearLayoutManager);
            recyclerView.addOnScrollListener(new EndlessRecyclerViewScrollListener( linearLayoutManager) {
                @Override
                public void onLoadMore(int page, int totalItemsCount, RecyclerView view) {
                    Toast.makeText(getActivity(),"LAst",Toast.LENGTH_LONG).show();
                }
            });
    

มันทำงานได้สมบูรณ์แบบในตอนท้ายของฉันฝากเตือนฉันหากคุณได้รับปัญหาใด ๆ


สิ่งนี้มีประโยชน์มาก :)
Devrath

4

StaggeredGridLayoutมีการดำเนินงานของฉันคือมันเป็นประโยชน์มากสำหรับ

การใช้งาน:

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

rvMain.addOnScrollListener(scrollListener);

การใช้งาน Listener:

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

ทำไมต้องเพิ่มความล่าช้า 200 มิลลิวินาทีในการโทรกลับ
มณี

@Mani ฉันจำไม่ได้ว่าในตอนนี้ แต่บางทีฉันอาจจะทำเพื่อผลที่ราบรื่น ...
Aleksey Timoshchenko

4

หลังจากไม่พอใจกับคำตอบอื่น ๆ ส่วนใหญ่ในชุดข้อความนี้ฉันพบสิ่งที่ฉันคิดว่าดีกว่าและไม่มีที่ไหนในที่นี่

recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
    @Override
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        if (!recyclerView.canScrollVertically(1) && dy > 0)
        {
             //scrolled to bottom
        }else if (!recyclerView.canScrollVertically(-1) && dy < 0)
        {
            //scrolled to bottom
        }
    }
});

นี่เป็นเรื่องง่ายและจะตีหนึ่งครั้งภายใต้เงื่อนไขทั้งหมดเมื่อคุณเลื่อนไปด้านบนหรือด้านล่าง


3

ฉันกำลังค้นหาคำถามนี้เช่นกัน แต่ฉันไม่พบคำตอบที่ทำให้ฉันพอใจดังนั้นฉันจึงสร้างการใช้งานรีไซเคิล View

โซลูชันอื่น ๆ มีความแม่นยำน้อยกว่าของฉัน ตัวอย่างเช่นหากรายการสุดท้ายค่อนข้างใหญ่ (ข้อความจำนวนมาก) การเรียกกลับของโซลูชันอื่น ๆ จะมาก่อนหน้านี้มากจากนั้นรีไซเคิลดูจริงจะถึงด้านล่าง

การละลายของฉันแก้ไขปัญหานี้

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

จากนั้นในกิจกรรม / ส่วนของคุณ:

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

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

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

หวังว่ามันจะช่วยใครสักคน ;-)


1
ส่วนไหนของการแก้ปัญหาจากคนอื่นที่ไม่พอใจคุณ? คุณควรอธิบายก่อนที่จะแนะนำคำตอบของคุณ
Zam Sunk

@ZamSunk เพราะโซลูชันอื่น ๆ มีความแม่นยำน้อยกว่าของฉัน ตัวอย่างเช่นหากรายการสุดท้ายค่อนข้างน้อย (ข้อความจำนวนมาก) การเรียกกลับของโซลูชันอื่น ๆ จะมาก่อนหน้านี้มากจากนั้นรีไซเคิลดูจริงถึงด้านล่าง การละลายของฉันแก้ไขปัญหานี้
Andriy Antonov

0

นี่คือทางออกของฉัน:

    val onScrollListener = object : RecyclerView.OnScrollListener() {

    override fun onScrolled(recyclerView: RecyclerView?, dx: Int, dy: Int) {
        directionDown = dy > 0
    }

    override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
        if (recyclerView.canScrollVertically(1).not()
                && state != State.UPDATING
                && newState == RecyclerView.SCROLL_STATE_IDLE
                && directionDown) {
               state = State.UPDATING
            // TODO do what you want  when you reach bottom, direction
            // is down and flag for non-duplicate execution
        }
    }
}

คุณได้รับที่ไหน state enum นั้นมาจากไหน?
raditya gumay

สถานะเป็นสิ่งที่น่าสนใจของฉันนี่คือค่าสถานะสำหรับการตรวจสอบสถานะสำหรับหน้าจอนั้น (ฉันใช้ MVVM กับ databinding ที่แอปของฉัน)
Alex Zezekalo

คุณจะใช้การโทรกลับอย่างไรหากไม่มีการเชื่อมต่ออินเทอร์เน็ต
Aks4125

0

เราสามารถใช้ Interface เพื่อรับตำแหน่ง

อินเทอร์เฟซ: สร้างอินเทอร์เฟซสำหรับผู้ฟัง

public interface OnTopReachListener { void onTopReached(int position);}

กิจกรรม:

mediaRecycleAdapter = new MediaRecycleAdapter(Class.this, taskList); recycle.setAdapter(mediaRecycleAdapter); mediaRecycleAdapter.setOnSchrollPostionListener(new OnTopReachListener() {
@Override
public void onTopReached(int position) {
    Log.i("Position","onTopReached "+position);  
}
});

อะแดปเตอร์:

public void setOnSchrollPostionListener(OnTopReachListener topReachListener) {
    this.topReachListener = topReachListener;}@Override public void onBindViewHolder(MyViewHolder holder, int position) {if(position == 0) {
  topReachListener.onTopReached(position);}}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.