Назад к вопросам
Middle
67
questionbank

Как бы ты реализовал навигацию с использованием кнопок, чтобы она сохраняла стек?

Sobes Copilot

Получайте ответы в реальном времени

Ответ от нейросети

sobes.tech AI

Использовал бы Navigation Architecture Component.

  1. Создание NavGraph: Определил бы Destinations (фрагменты или Activity) и Actions (переходы между ними) в XML-ресурсе.

    <?xml version="1.0" encoding="utf-8"?>
    <navigation xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:id="@+id/nav_graph"
        app:startDestination="@id/fragmentA"> // Начальный пункт
    
        <fragment
            android:id="@+id/fragmentA"
            android:name="com.example.FragmentA"
            android:label="FragmentA" >
            <action
                android:id="@+id/action_fragmentA_to_fragmentB"
                app:destination="@id/fragmentB" />
        </fragment>
        <fragment
            android:id="@+id/fragmentB"
            android:name="com.example.FragmentB"
            android:label="FragmentB" >
            <action
                android:id="@+id/action_fragmentB_to_fragmentC"
                app:destination="@id/fragmentC" />
        </fragment>
         <fragment
            android:id="@+id/fragmentC"
            android:name="com.example.FragmentC"
            android:label="FragmentC" />
    </navigation>
    
  2. Размещение NavHost: Добавил бы NavHost (например, NavHostFragment) в макет Активити, к которому привязан NavGraph.

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
    
        <androidx.fragment.app.FragmentContainerView
            android:id="@+id/nav_host_fragment"
            android:name="androidx.navigation.fragment.NavHostFragment"
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="1"
            app:defaultNavHost="true"
            app:navGraph="@navigation/nav_graph" /> // Привязка NavGraph
    
        <!-- Здесь могут быть кнопки навигации -->
    </LinearLayout>
    
  3. Добавление кнопок навигации: Использовал бы BottomNavigationView или отдельные кнопки. Для BottomNavigationView привязал бы его к NavController.

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
    
        <androidx.fragment.app.FragmentContainerView
            android:id="@+id/nav_host_fragment"
            android:name="androidx.navigation.fragment.NavHostFragment"
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="1"
            app:defaultNavHost="true"
            app:navGraph="@navigation/nav_graph" />
    
        <Button
            android:id="@+id/button_go_to_b"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Go to B" />
    
        <Button
            android:id="@+id/button_go_to_c"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Go to C" />
    </LinearLayout>
    
  4. Обработка нажатий кнопок в коде: Получал бы экземпляр NavController и вызывал бы navigate() при нажатии на кнопку. NavController автоматически управляет back stack, сохраняя предыдущие Destination.

    // В Activity или Fragment
    import androidx.navigation.findNavController
    
    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
    
        val navController = findNavController(R.id.nav_host_fragment) // Получение NavController
    
        view.findViewById<Button>(R.id.button_go_to_b).setOnClickListener {
            navController.navigate(R.id.action_fragmentA_to_fragmentB) // Переход по Action
        }
    
        view.findViewById<Button>(R.id.button_go_to_c).setOnClickListener {
            // Пример прямого перехода к Destination, можно использовать action_fragmentA_to_fragmentC если он есть
             navController.navigate(R.id.fragmentC)
        }
    }
    

Для более сложной навигации с несколькими вкладками (BottomNavigationView, DrawerLayout) использовал бы setupWithNavController().

// В Activity с BottomNavigationView
import androidx.navigation.ui.setupWithNavController
import com.google.android.material.bottomnavigation.BottomNavigationView

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main)

    val navController = findNavController(R.id.nav_host_fragment)
    val bottomNavView = findViewById<BottomNavigationView>(R.id.bottom_nav_view)
    bottomNavView.setupWithNavController(navController) // Привязка BottomNavigationView к NavController
}

Navigation Component по умолчанию обрабатывает back stack. Нажатие кнопки "Назад" будет возвращать пользователя к предыдущему destination в back stack. Для управления поведением back stack при navigate() можно использовать NavOptions с popUpTo.