Question
最近在写 SideBar 的时候遇到一个问题,当执行 Recyclerview 的 smoothScrollToPosition(position)
的时候,Recyclerview 看上去并没有滚动到指定位置。
Analysis
当然,这并不是方法的bug,而是 smoothScrollToPosition(position)
的执行效果有三种情况,需要区分。
-
目标position在第一个可见项之前 。
这种情况调用smoothScrollToPosition
能够平滑的滚动到指定位置,并且置顶。
-
目标position在第一个可见项之后,最后一个可见项之前。
这种情况下,调用smoothScrollToPosition
不会有任何效果···
-
目标position在最后一个可见项之后。
这种情况调用smoothScrollToPosition
会把目标项滑动到屏幕最下方···
Solution
鉴于这三种情况,我想大多数情况下都无法满足我们的滑动要求。为了实现 Recyclerview 把指定 item 滑动到屏幕顶端的需求,我们需要对上面三种情况分别处理。
/** 目标项是否在最后一个可见项之后*/
private boolean mShouldScroll;
/** 记录目标项位置*/
private int mToPosition;
/**
* 滑动到指定位置
* @param mRecyclerView
* @param position
*/
private void smoothMoveToPosition(RecyclerView mRecyclerView, final int position) {
int firstItem = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(0));
int lastItem = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(mRecyclerView.getChildCount() - 1));
if (position < firstItem) {
mRecyclerView.smoothScrollToPosition(position);
} else if (position <= lastItem) {
int movePosition = position - firstItem;
if (movePosition >= 0 && movePosition < mRecyclerView.getChildCount()) {
int top = mRecyclerView.getChildAt(movePosition).getTop();
mRecyclerView.smoothScrollBy(0, top);
}
}else {
mRecyclerView.smoothScrollToPosition(position);
mToPosition = position;
mShouldScroll = true;
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
再通过onScrollStateChanged控制再次调用smoothMoveToPosition
mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if (mShouldScroll){
mShouldScroll = false;
smoothMoveToPosition(mRecyclerView,mToPosition);
}
}
});
}
目前这个解决方法有两个已知的问题
- 当目标项在最后一个可见项之后的时候,由于我们先执行
smoothScrollToPosition
方法,然后在OnScrollListener
中执行smoothMoveToPosition
方法,在滑动的时候不够连贯。 - 在手动滑动的时候执行该方法,会有极小的概率滑动的位置出现偏差。
如果你有更好解决办法,希望不吝指教。