How to save scroll position of RecyclerView in Android?
Asked Answered
H

9

25

I have Recycler view which lays inside of SwipeRefreshLayout. Also, have ability to open each item in another activity. After returning back to Recycler I need scroll to chosen item, or to previous Y. How to do that?

Yes, I googled, found articles in StackOverFlow about saving instance of layout manager, like this one: RecyclerView store / restore state between activities. But, it doesn't help me.

UPDATE

Right now I have this kind of resolving problem, but, of course, it also doesn't work.

private int scrollPosition;

...//onViewCreated - it is fragment
recyclerView.setHasFixedSize(true);
LinearLayoutManager llm = new LinearLayoutManager(getActivity());
recyclerView.setLayoutManager(llm);
data = new ArrayList<>();
adapter.setData(getActivity(), data);
recyclerView.setAdapter(adapter);
...

@Override
public void onResume() {
    super.onResume();
    recyclerView.setScrollY(scrollPosition);
}

@Override
public void onPause() {
    super.onPause();
    scrollPosition = recyclerView.getScrollY();
}

Yes, I have tried scrollTo(int, int) - doen't work.

Now I tried just scroll, for example, to Y = 100, but it doesn't scrolling at all.

Haplo answered 12/4, 2016 at 8:52 Comment(5)
Please post your code & what you have achieved so far.Summons
Possible duplicate of how to save recyclerview scroll position , with recyclerview.state or noMeadowlark
@MadhukarHebbar I checked it, and it doesn't work also...Haplo
@Haplo this one worked for me. Try it!Meadowlark
@MadhukarHebbar Yess!!! Thank you a lot :D And here is also my fault - I should scrolling after adapter.notifyDataSetChanged();Haplo
M
53

Save the current state of recycle view position @onPause:

    positionIndex= llManager.findFirstVisibleItemPosition();
    View startView = rv.getChildAt(0);
    topView = (startView == null) ? 0 : (startView.getTop() - rv.getPaddingTop());

Restore the scroll position @onResume:

    if (positionIndex!= -1) {
        llManager.scrollToPositionWithOffset(positionIndex, topView);
    }

or another way can be @onPause:

long currentVisiblePosition = 0;
currentVisiblePosition = ((LinearLayoutManager)rv.getLayoutManager()).findFirstCompletelyVisibleItemPosition();

restore @onResume:

((LinearLayoutManager) rv.getLayoutManager()).scrollToPosition(currentVisiblePosition);
currentVisiblePosition = 0;
Meadowlark answered 12/4, 2016 at 9:54 Comment(3)
Great solution..!!Maquette
can you share a full exampleOutlast
In my case, it was necessary to add marginTop in order to accurately restore the position topView = (startView == null) ? 0 : (startView.getTop() - startView.getMarginTop() - rv.getPaddingTop());Awlwort
A
31

A lot of these answers seem to be over complicating it.

The LayoutManager supports onRestoreInstanceState out of the box so there is no need to save scroll positions etc. The built in method already saves pixel perfect positions.

example fragment code (null checking etc removed for clarity):

private Parcelable listState;
private RecyclerView list;

@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    listState=savedInstanceState.getParcelable("ListState");

}

@Override
public void onSaveInstanceState(Bundle outState) {
    super.onSaveInstanceState(outState);

    outState.putParcelable("ListState", list.getLayoutManager().onSaveInstanceState());

}

then just call

list.getLayoutManager().onRestoreInstanceState(listState);

once your data has been reattached to your RecyclerView

Alcaic answered 1/8, 2017 at 11:23 Comment(13)
WoW! Easy and simple solution :)Vintage
I think this is the best solution!!Detergency
I had to add android:configChanges="orientation|screenSize" to AndroidManifest.xml. Than the code above worked. ThanksFluff
@LeonardoAssunção Then you did it wrong. That is not required and not recommended.Alcaic
@Alcaic I have a similar issue here: stackoverflow.com/questions/52176812/…. I would appreciate any insights you have on how to solve.Lianneliao
Does this work when navigating back from another activity?Rrhoea
@Rrhoea it wont work in that case because onSaveIntanceState() wont be called. However, since listState is defined as a global variable, u may use onStop() of the activity to assign a 'Parcelable' to listState but only if it is null, e.g., if (listState == null) listState = list.getLayoutManager().onSaveInstanceState(). Hopefully, it should work. Do post ur findings!Rare
Since I'm using fragments, I have the above code line in onDestroyView() instead of onStop(). @Alcaic your tip combined with this solution really solved my problem. Thanks!Rare
So that implementation handled both screen rotation and navigating back from another fragment? @ShahoodulHassanRrhoea
@Rrhoea Yep...@Alcaic solution dealt with screen rotation. For navigating back from another fragment, I had to add that line in onDestroyView. So yes, this implementation handled both scenarios.Rare
hi where we can call this method list.getLayoutManager().onRestoreInstanceState(listState);Dutiable
can anyone tell where do i put this line list.getLayoutManager().onRestoreInstanceState(listState);Cochineal
The answer already say this: "once your data has been reattached to your RecyclerView"Alcaic
Q
7

Beginning from version 1.2.0-alpha02 of androidx recyclerView library, it is now automatically managed. Just add it with:

implementation "androidx.recyclerview:recyclerview:1.2.0-alpha02"

And use:

adapter.stateRestorationPolicy = StateRestorationPolicy.PREVENT_WHEN_EMPTY

The StateRestorationPolicy enum has 3 options:

  • ALLOW — the default state, that restores the RecyclerView state immediately, in the next layout pass
  • PREVENT_WHEN_EMPTY — restores the RecyclerView state only when the adapter is not empty (adapter.getItemCount() > 0). If your data is loaded async, the RecyclerView waits until data is loaded and only then the state is restored. If you have default items, like headers or load progress indicators as part of your Adapter, then you should use the PREVENT option, unless the default items are added using MergeAdapter. MergeAdapter waits for all of its adapters to be ready and only then it restores the state.
  • PREVENT — all state restoration is deferred until you set ALLOW or PREVENT_WHEN_EMPTY.

Note that at the time of this answer, recyclerView library is still in alpha03, but alpha phase is not suitable for production purposes.

Quartis answered 5/5, 2020 at 9:18 Comment(1)
This just made restoring state 100 times easier. Thanks for pointing this out! Worked perfectly and I'm using 1.2.0-alpha05 in production.Heaveho
S
2

User your recycler view linearlayoutmanager for getting scroll position

int position = 0;
if (linearLayoutManager != null) {
   scrollPosition = inearLayoutManager.findFirstVisibleItemPosition();
}

and when restoring use following code

if (linearLayoutManager != null) {
  cardRecyclerView.scrollToPosition(mScrollPosition);
}

Hope this helps you

Stich answered 12/4, 2016 at 9:27 Comment(1)
sorry, but that answer was first in comments under my question.Haplo
P
0

to save position to Preferences, add this to your onStop()

 int currentVisiblePosition = ((LinearLayoutManager) recyclerView.getLayoutManager()).findFirstCompletelyVisibleItemPosition();
 getPreferences(MODE_PRIVATE).edit().putInt("listPosition", currentVisiblePosition).apply();

then restore position like this

 if (getItemCount() == 0) {
     int savedListPosition = getPreferences(MODE_PRIVATE).getInt("listPosition", 0);
     recyclerView.getLayoutManager().scrollToPosition(savedListPosition); }

this last code should be added inside an event of the Adapter (not sure witch event but in my case was onEvent() - com.google.firebase.firestore.EventListener)

Pastiness answered 3/9, 2018 at 9:54 Comment(0)
C
0

For some reason there are a lot of quite misleading tips/suggestions on how to save and restore scroll position in your_scrolling_container upon orientation changes.

Taking current scroll position and saving it in Activity’s onSaveInstanceState Extending a certain scrollable View to do same there Preventing Activity from being destroyed on rotation And yeah, they are working fine, but…

But in fact, everything is much simpler, because Android is already doing it for you!

If you take a closer look at RecyclerView/ListView/ScrollView/NestedScrollView sources, you’ll see that each of them is saving its scroll position in onSaveInstanceState. And during the first layout pass they are trying to scroll to this position in onLayout method.

There are only 2 things you need to do, to make sure it’s gonna work fine:

  1. Set an id for your scrollable view, which is probably already done. Otherwise Android won’t be able to save View state automatically.

  2. Provide a data before the first layout pass, to have the same scroll boundaries you had before rotation. That’s the step where developers usually have some issues.

Calfskin answered 20/2, 2019 at 17:26 Comment(0)
B
0

The easiest and transition compatible way I found is:

   @Override
public void onPause() {
    super.onPause();
    recyclerView.setLayoutFrozen(true);
}

@Override
public void onResume() {
    super.onResume();
    recyclerView.setLayoutFrozen(false);
}
Balsamic answered 14/7, 2019 at 12:38 Comment(0)
M
0

in onSaveInstanceState() method of fragment you can save the scroll position of RecycleView

@Override
  public void onSaveInstanceState(Bundle outState) {
   super.onSaveInstanceState(outState);
   LinearLayoutManager layoutManager = (LinearLayoutManager) 
   recyclerView.getLayoutManager();
   outState.putInt("scrolled_position", 
   layoutManager.findFirstCompletelyVisibleItemPosition());
}

then you can retrieve saved scroll position in onViewStateRestored() method

@Override
  public void onViewStateRestored(@Nullable Bundle savedInstanceState) {
  super.onViewStateRestored(savedInstanceState);
  if (savedInstanceState != null) {
    int scrollPosition = savedInstanceState.getInt("scrolled_position");
    recyclerView.scrollToPosition(scrollPosition);
  }
}
Misstate answered 23/1, 2023 at 6:40 Comment(0)
S
-1

You can use scrollToPosition or smoothScrollToPosition to scroll to any item position in RecyclerView.

If you want to scroll to item position in adapter, then you would have to use adapter's scrollToPosition or smoothScrollToPosition.

Sheridan answered 12/4, 2016 at 8:56 Comment(1)

© 2022 - 2024 — McMap. All rights reserved.