ChatGPT解决这个技术问题 Extra ChatGPT

检测 RecyclerView 滚动时何时到达最底部位置

我有这个 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 在滚动时何时到达最底部位置。可能吗 ?如果是,如何?

recyclerView.canScrollVertically(int方向);我们必须在这里传递的参数应该是什么?
@Adrian,如果您仍然对这个问题感兴趣:)))) stackoverflow.com/a/48514857/6674369

M
Michael

还有一个简单的方法来做到这一点

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 将始终返回 false。


onScrolled() 防止检测发生两次。
我无法将 ScrollListnerEvent 添加到我的 RecyclerView。 onScrollStateChanged 中的代码不会执行。
如果您只想触发一次(显示 Toast 一次),请在 if 语句中添加布尔标志(类成员变量)并将其设置为 true,如果喜欢:if (!recyclerView.canScrollVertically(1) && !mHasReachedBottomOnce) { mHasReachedBottomOnce = true Toast.makeText(YourActivity.this, "Last", Toast.LENGTH_LONG).show();}
@ bastami82 我使用你建议的技巧来防止吐司两次打印,但它不起作用
newState == RecyclerView.SCROLL_STATE_IDLE 添加到 if 语句中即可。
M
Milind Chaudhary

使用此代码避免重复调用

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

小伙伴们小心!当您尝试向上滑动而回收器中的元素很少时,这也会触发。它还将触发两次关于 newState 被忽略的事件。
只需添加一个额外的检查是否可以向上滚动。如果你不能向上滚动,也不能向下滚动,那么就忽略结果(因为列表不够长)。 recyclerView.canScrollVertically(-1) && !recyclerView.canScrollVertically(1)
D
Dmitry Ryadnenko

只需在您的 recyclerview 上实现 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
            }
        }
    };

你能解释一下mIsLoading吗?你在哪里设置或更改值。
mLoading 只是一个布尔变量,指示视图是否已启用。例如;如果应用程序正在填充 recyclerview,则 mLoading 将为 true,并在填充列表后变为 false。
此解决方案无法正常工作。看看stackoverflow.com/a/48514857/6674369
无法解析 android.support.v7.widget.RecyclerView 上的方法 'findFirstVisibleItemPosition'
@Iman Marashi,您需要投射:(recyclerView.layoutManager as LinearLayoutManager).findFirstVisibleItemPosition()。这项工作。
S
SolArabehety

答案是在 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()(Kotlin 中的 itemCount)。


好方法,但一个问题是 if 循环中的语句将执行多次
你有相同问题的解决方案吗
@Vishnu 一个简单的布尔变量可以解决这个问题。
我认为这是最好的答案!也许我会添加第一个检查:if(dy <= 0) return。在这种情况下,如果滚动方向是顶部,则什么也不做。
F
Fattie

在对该线程中的大多数其他答案不满意之后,我发现了一些我认为更好的东西,而且这里没有任何地方。

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

这很简单,并且当您滚动到顶部或底部时,在所有条件下都会恰好点击一次。


M
Manoj Perumarath

通过上述答案,我没有得到完美的解决方案,因为即使在 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")
        }

我几天前发现的替代解决方案,

  rv_repatriations.addOnScrollListener(object : RecyclerView.OnScrollListener() {
        override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
            super.onScrollStateChanged(recyclerView, newState)
            if (!recyclerView.canScrollVertically(RecyclerView.FOCUS_DOWN) && recyclerView.scrollState == RecyclerView.SCROLL_STATE_IDLE
                && !isLoaded
            ) {
                isLoaded = true
               //do what you want here and after calling the function change the value of boolean
                Log.e("RepatriationFragment", "Scroll end reached")
            }
        }
    })

使用布尔值来确保当我们触底时它不会被多次调用。


但是 recyclerView.canScrollVertically(direction) 方向只是任何整数 + vs - 所以如果它在底部,它可以是 4,如果它在顶部,它可以是 -12。
这个逻辑第一次执行两次
@andrey2ag 你也可以尝试替代解决方案吗?
C
Community

尝试这个

我已经使用了上面的答案,当您在回收站视图结束时,它总是运行,

如果您只想检查一次它是否在底部?示例:- 如果我在底部时有 10 个项目的列表,它会显示我,如果我从上到下滚动它不会再次打印,如果你添加更多列表并且你去那里它会再次显示。

注意:- 当您处理命中 API 的偏移时使用此方法

创建一个名为 EndlessRecyclerViewScrollListener 的类 import android.support.v7.widget.GridLayoutManager;导入 android.support.v7.widget.LinearLayoutManager;导入 android.support.v7.widget.RecyclerView;导入 android.support.v7.widget.StaggeredGridLayoutManager; public abstract class EndlessRecyclerViewScrollListener extends RecyclerView.OnScrollListener { // 在加载更多之前 // 在当前滚动位置下方拥有的最小项目数量。私有int可见阈值= 5; // 你加载的数据的当前偏移索引 private int currentPage = 0; // 上次加载后数据集中的项目总数 private int previousTotalItemCount = 0; // 如果我们仍在等待最后一组数据加载,则为真。私有布尔加载 = true; // 设置起始页索引 private int startingPageIndex = 0; RecyclerView.LayoutManager mLayoutManager;公共 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(); } 公共 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]; } } 返回最大尺寸; } // 这在滚动过程中每秒会发生很多次,所以要小心你放在这里的代码。 // 我们得到了一些有用的参数来帮助我们确定是否需要加载更多数据, // 但首先我们检查我们是否正在等待上一次加载完成。 @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); // 获取列表中的最大元素 lastVisibleItemPosition = getLastVisibleItem(lastVisibleItemPositions); } else if (mLayoutManager instanceof GridLayoutManager) { lastVisibleItemPosition = ((GridLayoutManager) mLayoutManager).findLastVisibleItemPosition(); } else if (mLayoutManager instanceof LinearLayoutManager) { lastVisibleItemPosition = ((LinearLayoutManager) mLayoutManager).findLastVisibleItemPosition(); } // 如果总项目数为零而前一个不是,则假设 // 列表无效并应重置回初始状态 if (totalItemCount < previousTotalItemCount) { this.currentPage = this.startingPageIndex; this.previousTotalItemCount = totalItemCount; if (totalItemCount == 0) { this.loading = true; } } // 如果它仍在加载,我们检查数据集计数是否发生变化 // 如果是,我们断定它已经完成加载并更新当前页面 // 数量和总项目数。 if (loading && (totalItemCount > previousTotalItemCount)) { loading = false;以前的总项目数 = 总项目数; } // 如果它当前没有加载,我们检查我们是否违反了 // 可见阈值并需要重新加载更多数据。 // 如果我们确实需要重新加载更多数据,我们执行 onLoadMore 来获取数据。 // 阈值也应该反映总共有多少列 if (!loading && (lastVisibleItemPosition + visibleThreshold) > totalItemCount) { currentPage++; onLoadMore(currentPage, totalItemCount, view);加载=真; } } // 每当执行新搜索时调用此方法 public void resetState() { this.currentPage = this.startingPageIndex; this.previousTotalItemCount = 0; this.loading = true; } // 定义实际基于页面加载更多数据的流程 public abstract void onLoadMore(int page, int totalItemsCount, RecyclerView view);像这样使用这个类 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(); } }) ;

它在我的最后运行完美,如果您遇到任何问题,请评论我


M
Maykel Llanes Garcia

有我的实现,它对 StaggeredGridLayout 非常有用。

用法 :

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

rvMain.addOnScrollListener(scrollListener);

监听器实现:

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我现在不记得了,但也许我这样做只是为了平滑效果......
A
Andriy Antonov

我也在寻找这个问题,但我没有找到让我满意的答案,所以我创建了自己的recyclerView实现。

其他解决方案不如我的精确。例如:如果最后一项非常大(大量文本),那么其他解决方案的回调将早得多,然后 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)
}

然后在您的活动/片段中:

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

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

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

希望它会帮助某人;-)


其他人的解决方案中哪一部分让您不满意?你应该先解释一下,然后再提出你的答案
@ZamSunk 因为其他解决方案不如我的精确。例如,如果最后一项相当多(大量文本),那么其他解决方案的回调将早得多,然后 recyclerView 真正到达底部。我的解决方案解决了这个问题。
R
Raymond Barrinuevo

使用 Kotlin

   recyclerView.addOnScrollListener(object : RecyclerView.OnScrollListener() {
            override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
                if (!recyclerView.canScrollVertically(1)) {
                    Toast.makeText(context, "Last", Toast.LENGTH_LONG).show();
                }
            }
        })

C
Caner

科特林答案

您可以使用此 Kotlin 函数来实现底部滚动跟随的最佳实践,以创建无限或无限滚动。

// Scroll listener.
private fun setupListenerPostListScroll() {
    val scrollDirectionDown = 1 // Scroll down is +1, up is -1.
    var currentListSize = 0

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

            override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
                super.onScrollStateChanged(recyclerView, newState)

                if (!recyclerView.canScrollVertically(scrollDirectionDown)
                    && newState == RecyclerView.SCROLL_STATE_IDLE
                ) {
                    val listSizeAfterLoading = recyclerView.layoutManager!!.itemCount

                    // List has more item.
                    if (currentListSize != listSizeAfterLoading) {
                        currentListSize = listSizeAfterLoading

                        // Get more posts.
                        postListScrollUpAction(listSizeAfterLoading)
                    }
                    else { // List comes limit.
                        showToastMessageShort("No more items.")
                    }
                }
            }
        })
}

s
shady31

我已经看到很多关于这个问题的回答,我认为他们都没有给出准确的行为作为结果。但是,如果您遵循这种方法,我很肯定您将获得最佳行为。

rvCategories 是你的 RecyclerView

categoriesList 是传递给您的适配器的列表

binding.rvCategories.addOnScrollListener(object : RecyclerView.OnScrollListener() {
  override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
            val position = (recyclerView.layoutManager as LinearLayoutManager).findLastCompletelyVisibleItemPosition()
            if (position + 1 == categoriesList.size) {
               // END OF RECYCLERVIEW IS REACHED
            } else {
                // END OF RECYCLERVIEW IS NOT REACHED
            }
    }
})

H
Hayk Mkrtchyan

大多数答案的结构都很差,并且存在一些问题。一个常见的问题是,如果用户快速滚动,到达结束块会执行多次我找到了一个解决方案,其中结束块只运行 1 次。

override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
    if (yourLayoutManager.findLastVisibleItemPosition() ==
          yourLayoutManager.itemCount - 1 && !recyclerView.canScrollVertically(1)) {
                Logger.log("End reached")
                // Do your operations
       }

       super.onScrolled(recyclerView, dx, dy)
 }

PS 有时如果 RecyclerView 为空,则可能会调用最终侦听器。作为解决方案,您还可以在上述代码中添加此检查。

if (recyclerView.adapter?.itemCount!! > 0)

A
ALEJO MUÑOZ

你可以使用这个,如果你放 1 那将在你停留在列表末尾时显示,如果你现在想要当你停留在列表的开头时,你将 1 更改为 -1

recyclerChat.addOnScrollListener(object : RecyclerView.OnScrollListener() {
        override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
            super.onScrollStateChanged(recyclerView, newState)
            if (!recyclerView.canScrollVertically(1)) {
               
            }
        }
    })

D
Dharman

这是我在阅读了这篇文章中的所有答案后的解决方案。我只想在最后一项显示在列表末尾且 listview 长度大于屏幕高度时显示 loading,这意味着如果列表中只有一个或两个项目,则不会显示 {1 }。

private var isLoadMoreLoading: Boolean = false

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

    override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {
        super.onScrollStateChanged(recyclerView, newState)

        if (!isLoadMoreLoading) {
            if (linearLayoutManager.findLastCompletelyVisibleItemPosition() == (list.size-1)) {
                if (recyclerView.canScrollVertically(-1)) {

                    adapter?.addLoadMoreView()
                    loadMore()
                }
            }
        }
    }
})

private fun loadMore() {
    isLoadMoreLoading = true
    //call loadMore api
}

因为这个 linearLayoutManager.findLastCompletelyVisibleItemPosition() == (list.size-1),我们可以知道最后一项显示,但我们还需要知道 listview 是否可以滚动。

因此我添加了 recyclerView.canScrollVertically(-1)。一旦你点击列表的底部,它就不能再向下滚动了。 -1 表示列表可以向上滚动。这意味着 listview 长度大于屏幕高度。

这个答案在 kotlin 中。


如果您向上滚动,这也会触发,到达最顶部的项目
D
Duna

最简单的方法是在这样的适配器中:

@Override
public void onBindViewHolder(HistoryMessageListAdapter.ItemViewHolder holder, int position) {
            if (position == getItemCount()-1){
                listener.onLastItemReached();
            }
        }

因为一旦最后一个项目被回收,监听器就会被触发。


A
Alex Zezekalo

这是我的解决方案:

    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 是那个枚举?
State 是我的一个枚举,这是用于检查该屏幕状态的标志(我在我的应用程序中使用了带有数据绑定的 MVVM)
如果没有互联网连接,您将如何实现回调?
A
Amuthan S

我们可以使用接口来获取位置

接口:为监听器创建一个接口

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