web-dev-qa-db-fra.com

Manière infaillible de gérer Fragment lors du changement d'orientation

public class MainActivity extends Activity implements MainMenuFragment.OnMainMenuItemSelectedListener {

 @Override
 public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    FragmentManager fragmentManager = getFragmentManager();
    FragmentTransaction fragmentTransaction = fragmentManager
            .beginTransaction();

    // add menu fragment
    MainMenuFragment myFragment = new MainMenuFragment();
    fragmentTransaction.add(R.id.menu_fragment, myFragment);

    //add content
    DetailPart1 content1= new DetailPart1 ();
    fragmentTransaction.add(R.id.content_fragment, content1);
    fragmentTransaction.commit();

}
public void onMainMenuSelected(String tag) {
  //next menu is selected replace existing fragment
}

J'ai besoin d'afficher deux vues de liste côte à côte, menu à gauche et son contenu à droite, par défaut, le premier menu est sélectionné et son contenu est affiché à droite. Le fragment qui affiche le contenu est comme ci-dessous.

public class DetailPart1 extends Fragment {
  ArrayList<HashMap<String, String>> myList = new ArrayList<HashMap<String, String>>();
  ListAdapter adap;
  ListView listview;

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

       if(savedInstanceState!=null){
        myList = (ArrayList)savedInstanceState.getSerializable("MYLIST_obj");
        adap = new LoadImageFromArrayListAdapter(getActivity(),myList );
        listview.setAdapter(adap);
       }else{
        //get list and load in list view
        getlistTask = new GetALLListTasks().execute();
    }


     @Override
   public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    View v = inflater.inflate(R.layout.skyview_fragment, container,false);
           return v;
        }


     @Override
      public void onSaveInstanceState(Bundle outState) {
         super.onSaveInstanceState(outState);
          outState.putSerializable("MYLIST_obj", myList );
        }
    }

OnActivityCreated et onCreateView sont appelés deux fois, il existe de nombreux exemples utilisant des fragments, c'est pourquoi je suis débutant dans ce segment. Je ne peux pas relier cet exemple à mon problème. Je n'ai pas déclaré Android: configChanges dans le fichier manifeste, il me faut l'activité de détruire et de recréer afin que je puisse utiliser une présentation différente en mode paysage.Veuillez vous aider à trouver une solution de contournement

80
user1811741

Vous créez un nouveau fragment à chaque fois que vous allumez l'écran dans votre activité onCreate();, mais vous conservez également les anciens avec super.onCreate(savedInstanceState);. Alors peut-être définir une balise et trouver le fragment s'il existe, ou transmettre un ensemble nul à super.

Cela m'a pris un certain temps à apprendre et ça peut vraiment être un bi **** quand vous travaillez avec des choses comme viewpager.

Je vous recommande de lire à propos de fragments une heure supplémentaire, car ce sujet est couvert.

Voici un exemple de la façon de gérer des fragments lors d’un changement d’orientation régulier:

Activité :

public class MainActivity extends FragmentActivity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        if (savedInstanceState == null) {
            TestFragment test = new TestFragment();
            test.setArguments(getIntent().getExtras());
            getSupportFragmentManager().beginTransaction().replace(Android.R.id.content, test, "your_fragment_tag").commit();
        } else {
            TestFragment test = (TestFragment) getSupportFragmentManager().findFragmentByTag("your_fragment_tag");
        }
    }
}

Fragment :

public class TestFragment extends Fragment {

    public static final String KEY_ITEM = "unique_key";
    public static final String KEY_INDEX = "index_key";
    private String mTime;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_layout, container, false);

        if (savedInstanceState != null) {
            // Restore last state
            mTime = savedInstanceState.getString("time_key");
        } else {
            mTime = "" + Calendar.getInstance().getTimeInMillis();
        }

        TextView title = (TextView) view.findViewById(R.id.fragment_test);
        title.setText(mTime);

        return view;
    }

    @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putString("time_key", mTime);
    }
}
123
Warpzit

Vous pouvez trouver un bon guide sur la manière de conserver les données entre les changements d’orientation et les activités de récréation dans Android) .

Sommaire:

  1. rendre votre fragment conservable:

    setRetainInstance(true);
    
  2. Créer un nouveau fragment uniquement si nécessaire (ou au moins en extraire des données)

    dataFragment = (DataFragment) fm.findFragmentByTag("data");
    
    // create the fragment and data the first time
    if (dataFragment == null) {
    
19
Sergej Werfel