Замена фрагмента другим фрагментом того же класса

у меня есть fragment (назовем его MyFragment), который раздувает различные макеты в соответствии с параметром, переданным в аргументах.
Все работает хорошо, если MyFragment запускается из другого фрагмента. Но если ... --6-->MyFragment активен, и я хочу запустить новый MyFragment С другим параметром макета fragmentManager не создает новый фрагмент вообще.

data.setInt("Layout index",i);
fragmentTab0 = (Fragment) new MyFragment();
fragmentTab0.setArguments(data);
fragmentTransaction.replace(R.id.fragmentContent, fragmentTab0, "MY");
fragmentTransaction.addToBackStack(null);
fragmentTransaction.commit();

как я могу силу убедите fragmentTransaction снова запустить фрагмент?

Примечание: важным моментом здесь является то, что мне нужно снова надуть макет, который отличается от макета, надутого раньше. Код выглядит так:

public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    switch( getArguments().getInt("Layout index") ) {
    case 1:
        rootView = inflater.inflate(R.layout.firstlayout, container, false);
        break;
    case 2:
        rootView = inflater.inflate(R.layout.secondlayout, container, false);
        break;
    case 3:
        rootView = inflater.inflate(R.layout.thirdlayout, container, false);
        break;
    default: break;
    }       

4 ответов


Обход Решения
Объяснение (Гувер, чтобы увидеть его)

так как исходный код для fragmentTransaction.replace/add/remove is недоступный я не мог найти, что на самом деле происходит. Но это разумно думать, что в какой-то момент он сравнивает текущий класс name с именем класса замены и завершает работу, если они являются тот же.. Спасибо @devconsole за указание источника код. Теперь я знаю, почему это происходит. Этот FragmentManager.removeFragment() метод не сбрасывает состояние фрагмента, он остается возобновленным, затем метод moveToState(CREATED) только initilizes фрагмент if (f.mState < newState) = if (RESUMED < CREATED) = false. Эльза,ergo, он просто возобновляет фрагмент.

поэтому для решения этой проблемы я создал почти пустой фрагмент, единственной целью которого является замена себя целевым фрагментом.

public class JumpFragment {

    public JumpFragment() {
    }

    @Override
    public void onStart() {
        Bundle data = getArguments();
        int containerId = data.getString("containerID");
        String tag = data.getString("tag");
        //Class<?> c = data.get???("class");            
        //Fragment f = (Fragment) c.newInstance();          
        Fragment f = (Fragment) new MyFragment();
        FragmentManager fragmentManager = getFragmentManager();
        FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
        f.setArguments(data);
        fragmentTransaction.replace(containerId, f, tag);
        fragmentTransaction.addToBackStack(null);
        fragmentTransaction.commit();
    }

}

и я использую это:

data.setInt("Layout index",i);
data.setInt("containerID",R.id.fragmentContent);
data.setString("tag","MY");
fragmentTab0 = (Fragment) new JumpFragment();
fragmentTab0.setArguments(data);
fragmentTransaction.replace(R.id.fragmentContent, fragmentTab0);
fragmentTransaction.commit();

теперь ни один фрагмент не заменяется тем же фрагментом класса:

MyFragment - > JumpFragment - > MyFragment

Я не понял, как передать класс через пакет аргументов, чтобы сделать его полностью общим.


далее работал без проблем для меня. Обратите внимание, что я использовал библиотеку поддержки Android.

activity_main.xml:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >

    <Button
        android:id="@+id/button_one"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="ONE" />

    <Button
        android:id="@+id/button_two"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_toRightOf="@id/button_one"
        android:text="TWO" />

    <FrameLayout
        android:id="@+id/main_container"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_below="@id/button_one" >
    </FrameLayout>

</RelativeLayout>

MainActivity.java:

public class MainActivity extends FragmentActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        if (savedInstanceState == null) {
            FragmentManager fragmentManager = getSupportFragmentManager();
            FragmentTransaction transaction = fragmentManager.beginTransaction();
            Fragment fragment = DetailsFragment.newInstance("INITIAL");
            transaction.add(R.id.main_container, fragment);
            transaction.commit();
        }

        findViewById(R.id.button_one).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                update("Button 1 clicked");
            }
        });

        findViewById(R.id.button_two).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                update("Button 2 clicked");
            }
        });
    }

    protected void update(String value) {
        FragmentManager fragmentManager = getSupportFragmentManager();
        FragmentTransaction transaction = fragmentManager.beginTransaction();
        Fragment fragment = DetailsFragment.newInstance(value);
        transaction.replace(R.id.main_container, fragment);
        transaction.commit();
    }

    public static final class DetailsFragment extends Fragment {
        public static DetailsFragment newInstance(String param) {
            DetailsFragment fragment = new DetailsFragment();
            Bundle args = new Bundle();
            args.putString("param", param);
            fragment.setArguments(args);
            return fragment;
        }

        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
            TextView textView = new TextView(container.getContext());
            textView.setText(getArguments().getString("param"));
            return textView;
        }
    }
}

вы пытались сначала удалить свой фрагмент с помощью remove(fragMgr.findFragmentByTag("MY"))а затем добавить новый ? PS : Я предполагаю, что вы не держите никаких ссылок на этот фрагмент.


Если я правильно вас понял: фрагмент, который вы хотите заменить то, что в настоящее время отображается, и пользователь делает что-то, чтобы заставить его показать себя?

Если это правильно, то сделали что-то подобное таким образом:

 public class MyFragment extends Fragment {
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
            // Inflate the layout for this fragment
            final View V = inflater.inflate(R.layout.myLayout, container, false);

            // Call method that fills the layout with data
            displayData(V);

            // Put a listener here that checks for user input
            Button redisplayButton = (Button) V.findViewById(R.id.my_button);
                // if the button is clicked....
               redisplayButton.setOnClickListener(new OnClickListener() {
                    public void onClick(View v){
                    //
                    //  do some stuff
                    //
                    //  ....then eventually...
                    displayData(V); 
                }
              });

        return V;
        }

позже у вас может быть метод displayData (), который определяет, что отображает фрагмент....

    public void displayData(View V){

       //  Do something


    return;
    }

надеюсь, что это помогает!