防止同一片段多次堆叠 (addToBackStack)
Prevent The Same Fragment From Stacking More Than Once ( addToBackStack)
我有一个包含三个片段的 Activity
让我们称它们为 A、B 和 C. Fragment
A在ActivityonCreate()
中调用。
FragmentA fragA = new FragmentA();
FragmentTransaction transaction = manager.beginTransaction();
transaction.add(R.id.activity2_layout, fragA, "A");
transaction.commit();
并在按下某些按钮时被 Fragment B 或 C 替换,FragmentTransaction 调用 addToBackStack()
.
FragmentB fragB = new FragmentB(); //or C
FragmentTransaction transaction = manager.beginTransaction();
transaction.replace(R.id.activity2_layout, fragB, "B"); //or C
transaction.addToBackStack("B"); //or C
transaction.commit();
但是假设我连续三次调用 Fragment
B,我怎样才能防止它自己堆叠?同时我希望这成为可能: B 调用 > C 调用 > B 调用- 但是当我尝试返回时,我希望 B 只打开一次(C < B ) 而不是 (B < C < B)。所以基本上用新的 backStack 删除了第一个 backStack。
不久前,我在自己的一个项目中解决了您的一个问题,即how can I prevent it from stacking on to it self?
。这是我在那里使用的确切代码:
public void setContent(@NonNull Fragment fragment, boolean addToBackStack) {
Fragment current = getSupportFragmentManager().findFragmentById(R.id.main_content);
if (current == null || !current.getClass().equals(fragment.getClass())) {
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
if (addToBackStack) {
transaction.addToBackStack(null).add(R.id.main_content, fragment).commit();
} else {
transaction.replace(R.id.main_content, fragment).commit();
}
}
}
此代码的作用是检查 class 当前显示的片段是什么,并将其与您要替换的片段进行比较。如果它们相同,则它不会执行任何操作。可以像这样调整此代码以满足您的需求
Fragment newFragment = new FragmentB(); //OR FragmentC
Fragment current = getSupportFragmentManager().findFragmentById(R.id.activity2_layout);
if (!current.getClass().equals(newFragment.getClass())) {
//POP BACKSTACK HERE LIKE EXPLAINED BELOW
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
final String tag = newFragment.getClass().getSimpleName();
transaction.addToBackStack(tag);
transaction.add(R.id.activity2_layout, newFragment, tag);
transaction.commit();
} else {
//THROW AN ERROR OR SOMETHING, HANDLE IT BEING THE SAME
}
确保一次只存在一个特定类型的片段可能可以通过弹出后台堆栈直到同一片段第一次出现之前来实现。我没有为此编写任何代码,但实现该功能应该不会太困难。如果你自己搞不懂,请告诉我,我会试着给你写一个例子:)
来自 How to resume Fragment from BackStack if exists,这可能与您的查询非常相似。
有一种方法可以根据事务名称或提交提供的 ID 弹出返回堆栈(例如,使用片段的 class 名称作为
Thijs 已经提到):
String backStateName = fragment.getClass().getName();
添加到后台:
FragmentTransaction ft = manager.beginTransaction();
ft.replace(R.id.content_frame, fragment);
ft.addToBackStack(backStateName);
ft.commit();
弹出时:
getSupportFragmentManager().popBackStackImmediate (backStateName, 0);
所以在替换片段之前,我们可以像这样在后台检查是否存在:
boolean fragmentPopped = manager.popBackStackImmediate (backStateName, 0);
if (!fragmentPopped){ //fragment not in back stack, create it.
...
...
ft.addToBackStack(backStateName);
ft.commit();
}
请检查上述问题和接受的答案以获得更多解释。
我有一个包含三个片段的 Activity
让我们称它们为 A、B 和 C. Fragment
A在ActivityonCreate()
中调用。
FragmentA fragA = new FragmentA();
FragmentTransaction transaction = manager.beginTransaction();
transaction.add(R.id.activity2_layout, fragA, "A");
transaction.commit();
并在按下某些按钮时被 Fragment B 或 C 替换,FragmentTransaction 调用 addToBackStack()
.
FragmentB fragB = new FragmentB(); //or C
FragmentTransaction transaction = manager.beginTransaction();
transaction.replace(R.id.activity2_layout, fragB, "B"); //or C
transaction.addToBackStack("B"); //or C
transaction.commit();
但是假设我连续三次调用 Fragment
B,我怎样才能防止它自己堆叠?同时我希望这成为可能: B 调用 > C 调用 > B 调用- 但是当我尝试返回时,我希望 B 只打开一次(C < B ) 而不是 (B < C < B)。所以基本上用新的 backStack 删除了第一个 backStack。
不久前,我在自己的一个项目中解决了您的一个问题,即how can I prevent it from stacking on to it self?
。这是我在那里使用的确切代码:
public void setContent(@NonNull Fragment fragment, boolean addToBackStack) {
Fragment current = getSupportFragmentManager().findFragmentById(R.id.main_content);
if (current == null || !current.getClass().equals(fragment.getClass())) {
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
if (addToBackStack) {
transaction.addToBackStack(null).add(R.id.main_content, fragment).commit();
} else {
transaction.replace(R.id.main_content, fragment).commit();
}
}
}
此代码的作用是检查 class 当前显示的片段是什么,并将其与您要替换的片段进行比较。如果它们相同,则它不会执行任何操作。可以像这样调整此代码以满足您的需求
Fragment newFragment = new FragmentB(); //OR FragmentC
Fragment current = getSupportFragmentManager().findFragmentById(R.id.activity2_layout);
if (!current.getClass().equals(newFragment.getClass())) {
//POP BACKSTACK HERE LIKE EXPLAINED BELOW
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
final String tag = newFragment.getClass().getSimpleName();
transaction.addToBackStack(tag);
transaction.add(R.id.activity2_layout, newFragment, tag);
transaction.commit();
} else {
//THROW AN ERROR OR SOMETHING, HANDLE IT BEING THE SAME
}
确保一次只存在一个特定类型的片段可能可以通过弹出后台堆栈直到同一片段第一次出现之前来实现。我没有为此编写任何代码,但实现该功能应该不会太困难。如果你自己搞不懂,请告诉我,我会试着给你写一个例子:)
来自 How to resume Fragment from BackStack if exists,这可能与您的查询非常相似。
有一种方法可以根据事务名称或提交提供的 ID 弹出返回堆栈(例如,使用片段的 class 名称作为 Thijs 已经提到):
String backStateName = fragment.getClass().getName();
添加到后台:
FragmentTransaction ft = manager.beginTransaction();
ft.replace(R.id.content_frame, fragment);
ft.addToBackStack(backStateName);
ft.commit();
弹出时:
getSupportFragmentManager().popBackStackImmediate (backStateName, 0);
所以在替换片段之前,我们可以像这样在后台检查是否存在:
boolean fragmentPopped = manager.popBackStackImmediate (backStateName, 0);
if (!fragmentPopped){ //fragment not in back stack, create it.
...
...
ft.addToBackStack(backStateName);
ft.commit();
}
请检查上述问题和接受的答案以获得更多解释。