is it possible to switch between Fragments without re-creating them all the time? If so, how?
In the documentation I found an example of how to replace Fragments.
// Create new fragment and transaction
Fragment newFragment = new ExampleFragment();
FragmentTransaction transaction = getFragmentManager().beginTransaction();
// Replace whatever is in the fragment_container view with this fragment,
// and add the transaction to the back stack
transaction.replace(R.id.fragment_container, newFragment);
transaction.addToBackStack(null);
// Commit the transaction
transaction.commit();
But I don't want to create my Fragments from the scratch every time I need them.
I also found this example of hiding/showing Fragments:
// The content view embeds two fragments; now retrieve them and attach
// their "hide" button.
FragmentManager fm = getFragmentManager();
addShowHideListener(R.id.frag1hide, fm.findFragmentById(R.id.fragment1));
addShowHideListener(R.id.frag2hide, fm.findFragmentById(R.id.fragment2));
But how would I create a fragment with an ID outside an XML file?
I think this might be related to this question, but there isn't an answer. :/
Thank you very much in advance, jellyfish
Edit:
That's how I'm doing it now:
Fragment shown = fragmentManager.findFragmentByTag(shownFragment);
//...
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
if (shown != null) fragmentTransaction.hide(shown);
//switch statetement for menu selection, just one example:
SettingsFragment set = (SettingsFragment) fragmentManager.findFragmentByTag(SET);
Toast.makeText(this, "Settings:" + set, Toast.LENGTH_LONG).show();
if (set == null)
{
set = new SettingsFragment();
fragmentTransaction.add(R.id.framelayout_content, set, SET);
}
else fragmentTransaction.show(set);
shownFragment = SET;
fragmentTransaction.commit();
If I call up the settings, then something else, and then go back to settings, the toast gives me "null" first and "Settings:SettingsFragment{40ef..." second.
However, if I replace fragmentTransaction.add(R.id.framelayout_content, set, SET);
with fragmentTransaction.replace(R.id.framelayout_content, set, SET);
I keep getting "null", "null", "null"... so it doesn't seem to find the Fragment by tag.
Edit2:
Adding fragmentTransaction.addToBackStack(null);
did the trick. :)
This saves the whole hiding/memorizing which fragment is shown part so I suppose it's the most elegant solution for this.
I found this tutorial quite helpful on the topic.
Edit3:
Looking at my code I realized I could get rid of some parts, so I changed it to:
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
if (shown != null) fragmentTransaction.hide(shown);
Settings set = (Settings) fragmentManager.findFragmentByTag(SET);
if (set == null) set = new Settings();
fragmentTransaction.replace(R.id.framelayout_content, set, SET);
fragmentTransaction.addToBackStack(null);
fragmentTransaction.commit();
However, this invoked an IllegalStateException: Fragment already added
, much the same like here. Is there an easy way to prevent this? Otherwise I think I might switch back to the hide/show bit.
Use replace() to replace an existing fragment in a container with an instance of a new fragment class that you provide. Calling replace() is equivalent to calling remove() with a fragment in a container and adding a new fragment to that same container.
Show activity on this post. A framelayout, Relative View and a few others represents a view in android and is extended from viewgroup. A Fragment is a an an Object that is used to represent a portion of a user interface and is usually hosted in an activity. A fragment has a viewgroup which you can assign an XML layout.
and then you can use the FragmentManager to create a FragmentTransaction which allows us to add fragments to the FrameLayout at runtime: // Begin the transaction FragmentTransaction ft = getSupportFragmentManager(). beginTransaction(); // Replace the contents of the container with the new fragment ft. replace(R.
It could depend on what you are trying to avoid being re-created.
// Replace whatever is in the fragment_container view with this fragment,
// and add the transaction to the back stack
transaction.replace(R.id.fragment_container, newFragment);
transaction.addToBackStack(null);
In your example example when you hit the back button from your newFragment the previous fragment will be shown (you'll get an onCreateView, onActivityCreated but no onCreate) so that fragment isn't being re-created as such. As for you newFragment you can still keep it around if you plan to use it again updating any internal state as required in say onCreate or onActivityCreated.
EDIT:
If you simply have a menu list with each entry invoking a different fragment in a right pane then adding to the back stack is not what you want. For this you might get away with calling add(...)
on each fragment up-front and simply hide/show each fragment as required (I've not tested this). Otherwise I would suggest holding a reference to each fragment, call replace(...)
on selecting a different menu item ensuring that you don't add to the back stack.
To avoid the
IllegalStateException: Fragment already added
I found a workaround that is working fine for me: use remove(AFrag)
and add(BFrag)
in your transaction, instead of replace()
.
It looks like it is a bug: 4th comment in the accepted answer.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With