RecyclerView在我們的開發(fā)過程中已經(jīng)使用的不能再多了,開發(fā)過程中十分實用准脂,并且使用簡單饿悬,但我們使用RecyclerView的定位滾動,就會發(fā)現(xiàn)問題叽唱。
RecyclerView本身提供的滾動方法有兩個
scrollToPosition(int)
原本以為這個方法的作用是設置選擇位置顯示在屏幕頂端屈呕,但用過的才知道這是什么東西,當設置position為1棺亭,2等虎眨,你會發(fā)現(xiàn)RecyclerView根本就不動,當設置大一點時镶摘,才會發(fā)生滾動嗽桩,但也不會達到我們預想的位置,當再向上設置滾動時凄敢,RecyclerView變的正常了碌冶,如果不仔細觀察你就看不懂了,Recycler怎么滾動完全是看心情啊涝缝。扑庞。譬重。。罐氨。
你仔細觀察或者查閱一下相關說明臀规,你才會明白為什么會產(chǎn)生這樣的結果,這個方法的作用是顯示指定項栅隐,就是把你想置頂?shù)捻楋@示出來塔嬉,但是在屏幕的什么位置是不管的,只要那一項現(xiàn)在看得到了租悄,那它就罷工了谨究! 而且這個方法是瞬間跳到位置,不帶有動畫泣棋。
另一個方法是
scrollBy(int x,int y)
這個方法使用時是傳入滾動的距離记盒,單位是像素,我們使用這個方法要自己控制滾動的距離外傅,在動態(tài)的布局中且各項樣式高度可能都不一樣的情況下纪吮,自己計算高度是很有難度的。
還有兩個與上面兩方法對應的帶有動畫的移動方法分別是
smoothScrollToPosition(int)
smoothScrollBy(int x,int y)
這兩個方法的最終結果與上兩個相同萎胰,只是過程多了滾動動畫碾盟。
上面說了這么多廢話,結論就是這些方法都不能很好解決問題技竟,但是冰肴,當ScrollTo和ScrollBy結合使用的時候,我們的問題就變的好解決很多了榔组!
我們解決問題的方法就是先用scrollToPosition熙尉,將要置頂?shù)捻椣纫苿语@示出來,然后計算這一項離頂部的距離搓扯,用scrollBy完成剩余的移動检痰,將置頂項移動到頂部。
下面我們來看具體的代碼實現(xiàn):
我們先來分析可能遇到的幾種情況哈
第一需要置頂項在屏幕顯示的上方锨推,這個最簡單直接調(diào)用scrollToPosition就會將這一項置頂铅歼。
第二需要置頂項在屏幕中顯示,但需要置頂换可,這時直接調(diào)用scrollBy椎椰,將其置頂。
第三種置頂項在屏幕的下方沾鳄,這時需要先調(diào)用scrollToPosition將置頂項移動到屏幕內(nèi)慨飘,然后就和第二種情況一樣了。
private void moveToPosition(int n) {
int firstItem = mLinearLayoutManager.findFirstVisibleItemPosition();
int lastItem = mLinearLayoutManager.findLastVisibleItemPosition();
if (n <= firstItem ){
mRecyclerView.scrollToPosition(n);
}else if ( n <= lastItem ){
int top = mRecyclerView.getChildAt(n - firstItem).getTop();
mRecyclerView.scrollBy(0, top);
}else{
mRecyclerView.scrollToPosition(n);
move = true;
}
}
對于第三種情況我們需要監(jiān)聽滾動译荞,但我們又不想在其他的情況觸發(fā)瓤的,所以我們增加一個標志位move休弃,第三種情況下才有二次移動。
class RecyclerViewListener extends RecyclerView.OnScrollListener{
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (move){
move = false;
int n = mIndex - mLinearLayoutManager.findFirstVisibleItemPosition();
if ( 0 <= n && n < mRecyclerView.getChildCount()){
int top = mRecyclerView.getChildAt(n).getTop();
mRecyclerView.scrollBy(0, top);
}
}
}
}
對于smoothScroll實現(xiàn)類似但需要監(jiān)聽onScrollStateChanged
private void smoothMoveToPosition(int n) {
int firstItem = mLinearLayoutManager.findFirstVisibleItemPosition();
int lastItem = mLinearLayoutManager.findLastVisibleItemPosition();
if (n <= firstItem ){
mRecyclerView.smoothScrollToPosition(n);
}else if ( n <= lastItem ){
int top = mRecyclerView.getChildAt(n - firstItem).getTop();
mRecyclerView.smoothScrollBy(0, top);
}else{
mRecyclerView.smoothScrollToPosition(n);
move = true;
}
}
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
if (move && newState == RecyclerView.SCROLL_STATE_IDLE ){
move = false;
int n = mIndex - mLinearLayoutManager.findFirstVisibleItemPosition();
if ( 0 <= n && n < mRecyclerView.getChildCount()){
int top = mRecyclerView.getChildAt(n).getTop();
mRecyclerView.smoothScrollBy(0, top);
}
}
}
這樣我們需要的效果就實現(xiàn)了堤瘤。
不要看下面這段玫芦,看完你會打我枣申。
在LinearLayoutManager有這樣一個方法庞溜,開始的時候沒有找到潮太。
public void scrollToPositionWithOffset(int position, int offset)
指定offset為0,就可以搞定了慎皱。。叶骨。
真的不要打我茫多。。忽刽。天揖。。跪帝。今膊。。我也是受害者