Замена фрагмента другим фрагментом того же класса
у меня есть 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 ответов
Обход Решения
Объяснение (Гувер, чтобы увидеть его)
так как исходный код для. Спасибо @devconsole за указание источника код. Теперь я знаю, почему это происходит. ЭтотfragmentTransaction.replace/add/remove
is недоступный я не мог найти, что на самом деле происходит. Но это разумно думать, что в какой-то момент он сравнивает текущий класс name с именем класса замены и завершает работу, если они являются тот же.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;
}
надеюсь, что это помогает!