Właściwie nie ma żadnego najnowszego fragmentu dodanego do stosu, ponieważ możesz dodać kilka lub fragmenty do stosu w jednej transakcji lub po prostu usunąć fragmenty bez dodawania nowego.
Jeśli naprawdę chcesz mieć stos fragmentów i mieć dostęp do fragmentu przez jego indeks w stosie, lepiej mieć warstwę abstrakcji nad FragmentManager
i jego backstackiem. Oto jak możesz to zrobić:
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);
}
}
Teraz zamiast dodawania i usuwania fragmentów dzwoniąc FragmentManager
bezpośrednio, należy użyć push()
, replace()
i pop()
metody FragmentStackManager
. I będziesz mógł uzyskać dostęp do najwyższego fragmentu, po prostu dzwoniąc stack.get(stack.size() - 1)
.
Ale jeśli lubisz hacki, mam inne sposoby na robienie podobnych rzeczy. Jedyne, o czym muszę wspomnieć, to to, że te hacki będą działać tylko z fragmentami wsparcia.
Pierwszy hack polega na dodaniu wszystkich aktywnych fragmentów do menedżera fragmentów. Jeśli po prostu zastąpisz fragmenty jeden po drugim i zdejmiesz ze stosu, ta metoda zwróci najwyższy fragment:
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;
}
}
Drugie podejście jest bardziej hakerskie i pozwala na pobranie wszystkich fragmentów dodanych w ostatniej transakcji, dla której addToBackStack
wywołano:
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;
}
}
Zauważ, że w tym przypadku musisz umieścić tę klasę w android.support.v4.app
pakiecie.
FragmentTransaction.add(int containerViewId, Fragment fragment, String tag)
lubFragmentTransaction.replace(int containerViewId, Fragment fragment, String tag)
doc