単一のトランザクションでスタックに複数のフラグメントを追加したり、新しいフラグメントを追加せずにフラグメントを削除したりできるため、実際にはスタックに追加された最新のフラグメントはありません。
本当にフラグメントのスタックがあり、スタック内のインデックスでフラグメントにアクセスできるようにしたい場合は、FragmentManager
とそのバックスタック上に抽象化レイヤーを配置することをお勧めします。方法は次のとおりです。
public class FragmentStackManager {
private final FragmentManager fragmentManager;
private final int containerId;
private final List<Fragment> fragments = new ArrayList<>();
public FragmentStackManager(final FragmentManager fragmentManager,
final int containerId) {
this.fragmentManager = fragmentManager;
this.containerId = containerId;
}
public Parcelable saveState() {
final Bundle state = new Bundle(fragments.size());
for (int i = 0, count = fragments.size(); i < count; ++i) {
fragmentManager.putFragment(state, Integer.toString(i), fragments.get(i));
}
return state;
}
public void restoreState(final Parcelable state) {
if (state instanceof Bundle) {
final Bundle bundle = (Bundle) state;
int index = 0;
while (true) {
final Fragment fragment =
fragmentManager.getFragment(bundle, Integer.toString(index));
if (fragment == null) {
break;
}
fragments.add(fragment);
index += 1;
}
}
}
public void replace(final Fragment fragment) {
fragmentManager.popBackStackImmediate(
null, FragmentManager.POP_BACK_STACK_INCLUSIVE);
fragmentManager.beginTransaction()
.replace(containerId, fragment)
.addToBackStack(null)
.commit();
fragmentManager.executePendingTransactions();
fragments.clear();
fragments.add(fragment);
}
public void push(final Fragment fragment) {
fragmentManager
.beginTransaction()
.replace(containerId, fragment)
.addToBackStack(null)
.commit();
fragmentManager.executePendingTransactions();
fragments.add(fragment);
}
public boolean pop() {
if (isEmpty()) {
return false;
}
fragmentManager.popBackStackImmediate();
fragments.remove(fragments.size() - 1);
return true;
}
public boolean isEmpty() {
return fragments.isEmpty();
}
public int size() {
return fragments.size();
}
public Fragment getFragment(final int index) {
return fragments.get(index);
}
}
今、代わりに呼び出すことにより、フラグメントの追加と削除をFragmentManager
直接、あなたが使用する必要がありpush()
、replace()
およびpop()
メソッドのFragmentStackManager
。また、を呼び出すだけで、最上位のフラグメントにアクセスできますstack.get(stack.size() - 1)
。
しかし、ハッキングが好きなら、私は他の方法で同様のことをしなければなりません。私が言及しなければならない唯一のことは、これらのハックはサポートフラグメントでのみ機能するということです。
最初のハックは、すべてのアクティブなフラグメントをフラグメントマネージャーに追加することです。フラグメントを1つずつ置き換えてスタックからポップすると、このメソッドは最上位のフラグメントを返します。
public class BackStackHelper {
public static List<Fragment> getTopFragments(
final FragmentManager fragmentManager) {
final List<Fragment> fragments = fragmentManager.getFragments();
final List<Fragment> topFragments = new ArrayList<>();
for (final Fragment fragment : fragments) {
if (fragment != null && fragment.isResumed()) {
topFragments.add(fragment);
}
}
return topFragments;
}
}
2番目のアプローチは、イベントのハックで、addToBackStack
呼び出された最後のトランザクションで追加されたすべてのフラグメントを取得できます。
package android.support.v4.app;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class BackStackHelper {
public static List<Fragment> getTopFragments(
final FragmentManager fragmentManager) {
if (fragmentManager.getBackStackEntryCount() == 0) {
return Collections.emptyList();
}
final List<Fragment> fragments = new ArrayList<>();
final int count = fragmentManager.getBackStackEntryCount();
final BackStackRecord record =
(BackStackRecord) fragmentManager.getBackStackEntryAt(count - 1);
BackStackRecord.Op op = record.mHead;
while (op != null) {
switch (op.cmd) {
case BackStackRecord.OP_ADD:
case BackStackRecord.OP_REPLACE:
case BackStackRecord.OP_SHOW:
case BackStackRecord.OP_ATTACH:
fragments.add(op.fragment);
}
op = op.next;
}
return fragments;
}
}
この場合、このクラスをandroid.support.v4.app
パッケージに入れる必要があることに注意してください。
FragmentTransaction.add(int containerViewId, Fragment fragment, String tag)
またはFragmentTransaction.replace(int containerViewId, Fragment fragment, String tag)
doc