我正在寻找一种方法来滚动RecyclerView,以显示选中的项目在顶部。
在一个ListView中,我能够通过使用scrollTo(x,y)来做到这一点,并获得需要居中的元素的顶部。
喜欢的东西:
@Override
public void onItemClick(View v, int pos){
mylistView.scrollTo(0, v.getTop());
}
问题是RecyclerView在使用它的scrollTo方法时返回一个错误说
RecyclerView不支持滚动到绝对位置
我如何滚动一个RecyclerView把选定的项目放在视图的顶部?
如果你想自动滚动而不显示滚动运动,那么你需要写以下代码:
mRecyclerView.getLayoutManager().scrollToPosition(position);
如果你想显示滚动运动,那么你需要添加以下代码。
步骤1:你需要声明SmoothScroller。
RecyclerView.SmoothScroller smoothScroller = new
LinearSmoothScroller(this.getApplicationContext()) {
@Override
protected int getVerticalSnapPreference() {
return LinearSmoothScroller.SNAP_TO_START;
}
};
=>步骤2:你需要添加这段代码任何事件,你想要执行滚动到特定的位置。
首先你需要设置目标位置为平滑滚动。
smoothScroller.setTargetPosition(position);
然后你需要将SmoothScroller设置为LayoutManager。
mRecyclerView.getLayoutManager().startSmoothScroll(smoothScroller);
简介
没有一个答案解释如何在顶部显示最后一项。因此,答案只适用于上面或下面仍然有足够的项来填充剩余的RecyclerView的项。例如,如果有59个元素,第56个元素被选中,它应该在顶部,如下图所示:
那么,让我们看看如何在下一段中实现它。
解决方案
我们可以使用linearLayoutManager来处理这些情况。scrollToPositionWithOffset(pos, 0)和额外的逻辑在适配器的RecyclerView -通过添加一个自定义边距下面的最后一项(如果最后一项是不可见的,那么这意味着有足够的空间填充RecyclerView)。自定义边距可以是根视图高度和项目高度之间的差值。因此,您的adaptor for RecyclerView将如下所示:
...
@Override
public void onBindViewHolder(ViewHolder holder, final int position) {
...
int bottomHeight = 0;
int itemHeight = holder.itemView.getMeasuredHeight();
// if it's the last item then add a bottom margin that is enough to bring it to the top
if (position == mDataSet.length - 1) {
bottomHeight = Math.max(0, mRootView.getMeasuredHeight() - itemHeight);
}
RecyclerView.LayoutParams params = (RecyclerView.LayoutParams)holder.itemView.getLayoutParams();
params.setMargins(0, 0, params.rightMargin, bottomHeight);
holder.itemView.setLayoutParams(params);
...
}
...
我可以在这里添加的是如何使它与DiffUtil和ListAdapter一起工作
你可能会注意到调用recyclerView. scrolltopposition (pos)或(recyclerView. scrolltopposition)。layoutManager作为LinearLayoutManager)。如果在adapter.submitList后直接调用scrollToPositionWithOffset(pos, offset)将不起作用。这是因为differ在后台线程中查找更改,然后异步地将更改通知适配器。在一个SO上,我看到了几个错误的答案和不必要的延迟等来解决这个问题。
为了正确地处理这种情况,submitList有一个回调函数,在应用更改时调用它。
因此,在这种情况下,正确的kotlin实现是:
//memorise target item here and a scroll offset if needed
adapter.submitList(items) {
val pos = /* here you may find a new position of the item or just use just a static position. It depends on your case */
recyclerView.scrollToPosition(pos)
}
//or
adapter.submitList(items) { recyclerView.smoothScrollToPosition(pos) }
//or etc
adapter.submitList(items) { (recyclerView.layoutManager as LinearLayoutManager).scrollToPositionWithOffset(pos, offset) }