Отдельный задний стек для каждой вкладки в Bottomnavigationview Android с использованием фрагментов
Я ввожу BottomNavigationView
для навигации в приложении для Android. Я использую фрагменты, чтобы установить содержимое каждой вкладки.
Я знаю, как настроить один фрагмент для каждой вкладки, а затем переключать фрагменты при нажатии на вкладку. Но как я могу иметь отдельный стек для каждой вкладки? Вот код для настройки одного фрагмента:
Fragment selectedFragment = ItemsFragment.newInstance();
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
transaction.replace(R.id.content, selectedFragment);
transaction.commit();
пример Fragment
A и B будут находиться под вкладками 1 и Fragment
C и D под вкладкой 2. При запуске приложения отображается фрагмент A и Выбрана вкладка 1. Тогда Fragment
A может быть заменен фрагментом B. Когда выбрана вкладка 2, должен отображаться фрагмент C. Если вкладка 1 выбрана Fragment
B должен снова отображаться. На данный момент, должно быть возможно использовать кнопку назад, чтобы показать фрагмент A.
и вот код для настройки next fragment
в этом разделе:
Fragment selectedFragment = ItemsFragment.newInstance();
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.replace(R.id.content, selectedFragment);
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
ft.addToBackStack(null);
ft.commit();
4 ответов
наконец, я нашел решение, оно было вдохновлено предыдущим ответом на StackOverflow:отдельный задний стек для каждой вкладки в Android с помощью фрагментов
Я только заменил TabHost на BottomNavigationView и вот код:
Основным Видом Деятельности
public class MainActivity extends AppCompatActivity {
private HashMap<String, Stack<Fragment>> mStacks;
public static final String TAB_HOME = "tab_home";
public static final String TAB_DASHBOARD = "tab_dashboard";
public static final String TAB_NOTIFICATIONS = "tab_notifications";
private String mCurrentTab;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
BottomNavigationView navigation = (BottomNavigationView) findViewById(R.id.navigation);
navigation.setOnNavigationItemSelectedListener(mOnNavigationItemSelectedListener);
mStacks = new HashMap<String, Stack<Fragment>>();
mStacks.put(TAB_HOME, new Stack<Fragment>());
mStacks.put(TAB_DASHBOARD, new Stack<Fragment>());
mStacks.put(TAB_NOTIFICATIONS, new Stack<Fragment>());
navigation.setSelectedItemId(R.id.navigation_home);
}
private BottomNavigationView.OnNavigationItemSelectedListener mOnNavigationItemSelectedListener
= new BottomNavigationView.OnNavigationItemSelectedListener() {
@Override
public boolean onNavigationItemSelected(@NonNull MenuItem item) {
switch (item.getItemId()) {
case R.id.navigation_home:
selectedTab(TAB_HOME);
return true;
case R.id.navigation_dashboard:
selectedTab(TAB_DASHBOARD);
return true;
case R.id.navigation_notifications:
selectedTab(TAB_NOTIFICATIONS);
return true;
}
return false;
}
};
private void gotoFragment(Fragment selectedFragment)
{
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content, selectedFragment);
fragmentTransaction.commit();
}
private void selectedTab(String tabId)
{
mCurrentTab = tabId;
if(mStacks.get(tabId).size() == 0){
/*
* First time this tab is selected. So add first fragment of that tab.
* Dont need animation, so that argument is false.
* We are adding a new fragment which is not present in stack. So add to stack is true.
*/
if(tabId.equals(TAB_HOME)){
pushFragments(tabId, new HomeFragment(),true);
}else if(tabId.equals(TAB_DASHBOARD)){
pushFragments(tabId, new DashboardFragment(),true);
}else if(tabId.equals(TAB_NOTIFICATIONS)){
pushFragments(tabId, new NotificationsFragment(),true);
}
}else {
/*
* We are switching tabs, and target tab is already has atleast one fragment.
* No need of animation, no need of stack pushing. Just show the target fragment
*/
pushFragments(tabId, mStacks.get(tabId).lastElement(),false);
}
}
public void pushFragments(String tag, Fragment fragment, boolean shouldAdd){
if(shouldAdd)
mStacks.get(tag).push(fragment);
FragmentManager manager = getSupportFragmentManager();
FragmentTransaction ft = manager.beginTransaction();
ft.replace(R.id.content, fragment);
ft.commit();
}
public void popFragments(){
/*
* Select the second last fragment in current tab's stack..
* which will be shown after the fragment transaction given below
*/
Fragment fragment = mStacks.get(mCurrentTab).elementAt(mStacks.get(mCurrentTab).size() - 2);
/*pop current fragment from stack.. */
mStacks.get(mCurrentTab).pop();
/* We have the target fragment in hand.. Just show it.. Show a standard navigation animation*/
FragmentManager manager = getSupportFragmentManager();
FragmentTransaction ft = manager.beginTransaction();
ft.replace(R.id.content, fragment);
ft.commit();
}
@Override
public void onBackPressed() {
if(mStacks.get(mCurrentTab).size() == 1){
// We are already showing first fragment of current tab, so when back pressed, we will finish this activity..
finish();
return;
}
/* Goto previous fragment in navigation stack of this tab */
popFragments();
}
}
пример фрагмента Home
public class HomeFragment extends Fragment {
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_home, container, false);
Button gotoNextFragment = (Button) view.findViewById(R.id.gotoHome2);
gotoNextFragment.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
((MainActivity)getActivity()).pushFragments(MainActivity.TAB_HOME, new Home2Fragment(),true);
}
});
return view;
}
}
стоит отметить, что поведение, которое вы описываете, идет вразрез с рекомендациями Google. https://material.io/guidelines/components/bottom-navigation.html#bottom-navigation-behavior
навигация по нижней панели навигации должна сбросить состояние задачи.
другими словами, наличие фрагмента A и фрагмента B" внутри " вкладки 1 нормально, но если пользователь открывает фрагмент B, нажимает вкладку 2, а затем снова нажимает вкладку 1, они должны видеть Фрагмент А.
это поведение поддерживается новым компонентом архитектуры навигации (https://developer.android.com/topic/libraries/architecture/navigation/).
по сути, можно использовать NavHostFragment
, который является фрагментом, который контролирует свой собственный задний стек:
каждый NavHostFragment имеет NavController, который определяет допустимую навигацию внутри узла навигации. Это включает график навигации, а также состояние навигации, такое как текущее местоположение и задний стек, который будет сохранен и восстановлен вместе с самим NavHostFragment. https://developer.android.com/reference/androidx/navigation/fragment/NavHostFragment
вот пример:https://github.com/deisold/navigation
вместо использования метода replace используйте add fragment,
вместо этого метода футов.заменить(ИД Р..содержание, selectedFragment);
использовать футов.добавить(ИД Р..содержание, selectedFragment);
Fragment selectedFragment = ItemsFragment.newInstance();
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.(R.id.content, selectedFragment);
ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE);
ft.addToBackStack(null);
ft.commit();