I have a very simple code where I use Action Bar with tab fragments. It works fine after load, but after orientation change it goes crazy. The old fragment also visible (why
I found a usable answer in other question.
I need to modify my TabListener (I moved it into my Main activity class as inner class):
private class TabListener<T extends Fragment> implements android.app.ActionBar.TabListener
{
private Fragment mFragment;
private final Activity mActivity;
private final String mTag;
private final Class<T> mClass;
/**
* Constructor used each time a new tab is created.
*
* @param activity
* The host Activity, used to instantiate the fragment
* @param tag
* The identifier tag for the fragment
* @param clz
* The fragment's Class, used to instantiate the fragment
*/
public TabListener(final Activity activity, final String tag, final Class<T> clz)
{
mActivity = activity;
mTag = tag;
mClass = clz;
}
@Override
public void onTabReselected(final Tab tab, final FragmentTransaction ft)
{
// User selected the already selected tab. Usually do nothing.
}
@Override
public void onTabSelected(final Tab tab, final FragmentTransaction ft)
{
mFragment = mActivity.getFragmentManager().findFragmentByTag(mTag);
if (mFragment == null)
{
mFragment = Fragment.instantiate(mActivity, mClass.getName());
ft.add(android.R.id.content, mFragment, mTag);
}
else
{
ft.attach(mFragment);
}
}
@Override
public void onTabUnselected(final Tab tab, final FragmentTransaction ft)
{
if (mFragment != null)
{
ft.detach(mFragment);
}
}
}
So before I add fragment(again), I check it existance (and get its reference) and if it exists I attach it only.
public void onTabSelected(Tab tab, FragmentTransaction ft)
{
// Check if the fragment is already initialized
if (mFragment == null)
{
if(ft.findFragmentById(android.R.id.content) == null){
// If not, instantiate and add it to the activity
mFragment = Fragment.instantiate(mActivity, mClass.getName());
ft.add(android.R.id.content, mFragment, mTag);
}
} else
{
// If it exists, simply attach it in order to show it
ft.attach(mFragment);
}
}
Something like that, because your problem is that your adding the same fragment twice, we just have to find where...
Try using ft.replace(R.id.content, mFragment) in place of ft.attach(mFragment); in onTabSelected function
I found a very simple solution to avoid fragments duplication:
public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) {
Fragment currentFragment = getFragmentManager().findFragmentByTag(CURRENT_FRAGMENT_TAG);
if (currentFragment == null || !currentFragment.getClass().equals(mFragment.getClass())) {
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
ft.add(android.R.id.content, mFragment, CURRENT_FRAGMENT_TAG);
}
}
public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) {
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_CLOSE);
ft.remove(mFragment);
}
The key of the solution is in the condition:
currentFragment == null || !currentFragment.getClass().equals(mFragment.getClass())
This condition is valid ONLY if the classes of the fragments are different. In case you have different instances of the same class you have to put an extra attribute in your fragments to recognize his function (or the association with and to make the condition !currentFragment.getClass().equals(mFragment.getClass()) true: for example you can use the FragmentTransaction tag feature.
Bye, Alex.
I solved this by just looking up the fragment in the tab listener constructor.
public class TabListener<T extends Fragment> implements ActionBar.TabListener
{
private Fragment fragment;
private final SherlockFragmentActivity activity;
private final String tag;
private final Class<T> clazz;
public TabListener(SherlockFragmentActivity activity, String tag, Class<T> clazz)
{
this.activity = activity;
this.tag = tag;
this.clazz = clazz;
FragmentManager manager = ((SherlockFragmentActivity) activity).getSupportFragmentManager();
fragment = manager.findFragmentByTag(tag);
}
...
}