I have a Fragment in an Activity and when the device is rotated, onSaveInstanceState() and onViewStateRestored() are called.
onSaveInstanceState():
@Override
public void onSaveInstanceState(Bundle outState){
Log.e("Fragment", "Called onSaveInstanceState");
outState = getSavedInstanceState();
super.onSaveInstanceState(outState);
}
This is what sets the outState bundle in onSaveInstanceState():
public static Bundle getSavedInstanceState(){
Bundle saveState = new Bundle();
//Save view states
saveState.putBoolean("startButtonVisible", buttonStart.getVisibility() == View.VISIBLE);
saveState.putBoolean("stopButtonVisible", buttonStop.getVisibility() == View.VISIBLE);
saveState.putBoolean("resetButtonVisible", buttonReset.getVisibility() == View.VISIBLE);
saveState.putBoolean("saveButtonVisible", buttonSave.getVisibility() == View.VISIBLE);
saveState.putBoolean("openLatestFileButtonVisible", buttonOpenLatestFile.getVisibility() == View.VISIBLE);
saveState.putBoolean("editTextStartValueVisible", editTextStartValue.isEnabled());
saveState.putBoolean("buttonStartValueDownVisible", buttonStartValueDown.isEnabled());
saveState.putBoolean("buttonStartValueUpVisible", buttonStartValueUp.isEnabled());
saveState.putBoolean("editTextEndValueVisible", editTextEndValue.isEnabled());
saveState.putBoolean("buttonEndValueDownVisible", buttonEndValueDown.isEnabled());
saveState.putBoolean("buttonEndValueUpVisible", buttonEndValueUp.isEnabled());
return saveState;
}
This saves the data correctly however when onViewStateRestored() is called, the data is incorrect.
@Override
public void onViewStateRestored(Bundle savedInstanceState){
Log.e("Fragment", "Called onViewStateRestored");
super.onViewStateRestored(savedInstanceState);
//Restore view states
if (savedInstanceState != null){
restoreInstanceState(savedInstanceState);
if (buttonReset.getVisibility() == View.VISIBLE){
updateUi();
}
}
}
When onViewStateRestored() is called, it also calls restoreInstanceState() which is what is supposed to restore the state of each view.
public static void restoreInstanceState(Bundle savedInstanceState){
buttonStart.setVisibility(savedInstanceState.getBoolean("startButtonVisible") ? View.VISIBLE : View.INVISIBLE);
buttonStop.setVisibility(savedInstanceState.getBoolean("stopButtonVisible") ? View.VISIBLE : View.INVISIBLE);
buttonReset.setVisibility(savedInstanceState.getBoolean("resetButtonVisible") ? View.VISIBLE : View.INVISIBLE);
buttonSave.setVisibility(savedInstanceState.getBoolean("saveButtonVisible") ? View.VISIBLE : View.INVISIBLE);
buttonOpenLatestFile.setVisibility(savedInstanceState.getBoolean("openLatestFileButtonVisible") ? View.VISIBLE : View.INVISIBLE);
editTextStartValue.setEnabled(savedInstanceState.getBoolean("editTextStartValueVisible"));
buttonStartValueDown.setEnabled(savedInstanceState.getBoolean("buttonStartValueDownVisible"));
buttonStartValueDown.setImageResource(buttonStartValueDown.isEnabled() ? R.drawable.ic_arrow_down : R.drawable.ic_arrow_down_gray);
buttonStartValueUp.setEnabled(savedInstanceState.getBoolean("buttonStartValueUpVisible"));
buttonStartValueUp.setImageResource(buttonStartValueUp.isEnabled() ? R.drawable.ic_arrow_up : R.drawable.ic_arrow_up_gray);
editTextEndValue.setEnabled(savedInstanceState.getBoolean("editTextEndValueVisible"));
buttonEndValueDown.setEnabled(savedInstanceState.getBoolean("buttonEndValueDownVisible"));
buttonEndValueDown.setImageResource(buttonEndValueDown.isEnabled() ? R.drawable.ic_arrow_down : R.drawable.ic_arrow_down_gray);
buttonEndValueUp.setEnabled(savedInstanceState.getBoolean("buttonEndValueUpVisible"));
buttonEndValueUp.setImageResource(buttonEndValueUp.isEnabled() ? R.drawable.ic_arrow_up : R.drawable.ic_arrow_up_gray);
Log.e("Fragment", "Restored view visibility");
}
Why is the data incorrect when onViewStateRestored() is called? All of the views just become invisible regardless of what the they were before. Are the bundles not the same when saving and restoring the states?
All of this code is however working when calling fragmentManager.beginTransaction().replace(R.id.content_frame, CustomFragmentManager.savedFilesFragment).commit();
and then calling onViewStateRestored() manually.
EDIT: Changed field names.
Solution: I have replaced outState = new Bundle(getSavedInstanceState());
with outState.putAll(getSavedInstanceState());
and it is now working. Calling setRetainInstance(true);
was not necessary.
You should not create state bundle manually by yourself, using outState object that had passed in the method onSaveInstanceState() and restore in onCreate() method. Additional, you shoud call setRetainInstance(true) in your fragment's onCreate().