Android 导航组件:未找到 NavHostFragments 错误
Android Navigation component: No NavHostFragments found error
我目前正在我的应用程序中实现导航组件,但似乎发现无论我做什么,NavHostFragment 都不起作用。
我已经尝试过,重建、失效和重新启动,更改周围的名称,并更新我的 Android Studio,但似乎没有任何效果。我认为这也是为什么我在 MainActivity.kt
文件中尝试获取 NavController 时出错的原因,因为它 returns null.
Android Studio 版本 4.0.1
nav_graph.xml 图片:
nav_graph.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"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/nav_graph"
app:startDestination="@id/mainFragment">
<fragment
android:id="@+id/mainFragment"
android:name="com.example.movieapp.ui.main.MainFragment"
android:label="MainFragment">
<action
android:id="@+id/action_mainFragment_to_searchMovieFragment"
app:destination="@id/searchMovieFragment" />
</fragment>
<fragment
android:id="@+id/searchMovieFragment"
android:name="com.example.movieapp.ui.search.SearchMovieFragment"
android:label="SearchMovieFragment" />
</navigation>
main_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".activity.MainActivity">
<include
layout="@layout/appbar" />
<androidx.fragment.app.FragmentContainerView
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
app:defaultNavHost="true"
app:navGraph="@navigation/nav_graph" />
</androidx.constraintlayout.widget.ConstraintLayout>
MainActivity.kt
class MainActivity : AppCompatActivity() {
private lateinit var toolbar: Toolbar
private val navController by lazy {
(supportFragmentManager.findFragmentById(R.id.main_fragment) as NavHostFragment).navController
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.main_activity)
toolbar = findViewById(R.id.mainToolBar)
setSupportActionBar(toolbar)
setupActionBarWithNavController(navController)
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
val inflater: MenuInflater = menuInflater
inflater.inflate(R.menu.main_menu_bar, menu)
return true
}
}
build.gradle (:app)
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'
android {
compileSdkVersion 29
buildToolsVersion "29.0.3"
defaultConfig {
applicationId "com.example.app"
minSdkVersion 23
targetSdkVersion 29
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
debug {}
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
dataBinding {
enabled true
}
}
}
dependencies {
implementation fileTree(dir: "libs", include: ["*.jar"])
// Retrofit Libraries
implementation 'com.squareup.retrofit2:retrofit:2.6.0'
implementation 'com.squareup.retrofit2:converter-gson:2.6.0'
// Picasso Library
implementation 'com.squareup.picasso:picasso:2.71828'
implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
implementation 'androidx.core:core-ktx:1.3.0'
implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.2.0'
implementation 'androidx.legacy:legacy-support-v4:1.0.0'
implementation 'androidx.recyclerview:recyclerview:1.1.0'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
// Navigation libraries
def nav_version = "2.3.0"
implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"
implementation "androidx.navigation:navigation-ui-ktx:$nav_version"
}
您可以在 activity 中找到 NavController
为:
class MainActivity : AppCompatActivity() {
private lateinit var toolbar: Toolbar
private lateinit var navController: NavController
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.main_activity)
toolbar = findViewById(R.id.mainToolBar)
setSupportActionBar(toolbar)
navController = findNavController(R.id.nav_host_fragment)
setupActionBarWithNavController(navController)
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
val inflater: MenuInflater = menuInflater
inflater.inflate(R.menu.main_menu_bar, menu)
return true
}
}
private val navController by lazy {
(supportFragmentManager.findFragmentById(R.id.main_fragment) as NavHostFragment).navController
}
您的 navController
为空,因为我认为 supportFragmentManager.findFragmentById(R.id.main_fragment)
returns 为空。根据您的 activity 布局,您的 navHost 的 ID 是 R.id.nav_host_fragment
。
我有同样的问题,我通过从 Design、Palette 创建 NavHostFragment 解决了这个问题,然后分配相应的导航资源文件.
我解决问题的方法是转到 Activity.xml 并将片段 class/component 标记从 androidx.fragment.app.FragmentContainerView
更改为仅 fragment
。然后它出现在导航图的主机面板中。
如Doc中所述,您还可以使用布局编辑器添加 NavHostFragment。
- 在项目文件列表中,双击 activity 的布局 XML 文件以在布局编辑器中将其打开。
- 在调色板窗格中,选择容器类别,或者搜索“NavHostFragment”。
- 将 NavHostFragment 视图拖到 activity。
- 接下来,在出现的 Navigation Graphs 对话框中,选择相应的导航图与此 NavHostFragment 关联,然后单击确定。
YOU MAY NEED TO RESTART THE ANDROID STUDIO AFTER THIS (not mentioned
in the doc)
此外,当您尝试使用上述方法添加片段时,您可能会看到“片段”而不是“FragmentContainerView”。
对我来说,我在 FragmentContainerView
中添加了 android:name="androidx.navigation.fragment.NavHostFragment"
并与我一起工作
我们应该确保写下这一行
android:name="androidx.navigation.fragment.NavHostFragment"
<androidx.fragment.app.FragmentContainerView
android:id="@+id/navController"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="0dp"
android:layout_height="0dp"
app:defaultNavHost="true"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:navGraph="@navigation/nav_graph" />
最初我们没有 NavHostFragments。你将每个片段指向彼此,创建了运动动作,但你需要一些东西来承载这个运动。这就是 navigation host fragment
发挥作用的地方。 NavHostFragment 会根据需要在给定布局内交换我们的片段。那么什么布局应该容纳这个导航片段。它是在主要 activity 期间膨胀的布局。于是打开activity_main.xml
.
你应该加上Containers/NavHostFragment
选择NavHostFragment拖拽时,会看到可用的宿主片段。
我目前正在我的应用程序中实现导航组件,但似乎发现无论我做什么,NavHostFragment 都不起作用。
我已经尝试过,重建、失效和重新启动,更改周围的名称,并更新我的 Android Studio,但似乎没有任何效果。我认为这也是为什么我在 MainActivity.kt
文件中尝试获取 NavController 时出错的原因,因为它 returns null.
Android Studio 版本 4.0.1
nav_graph.xml 图片:
nav_graph.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"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/nav_graph"
app:startDestination="@id/mainFragment">
<fragment
android:id="@+id/mainFragment"
android:name="com.example.movieapp.ui.main.MainFragment"
android:label="MainFragment">
<action
android:id="@+id/action_mainFragment_to_searchMovieFragment"
app:destination="@id/searchMovieFragment" />
</fragment>
<fragment
android:id="@+id/searchMovieFragment"
android:name="com.example.movieapp.ui.search.SearchMovieFragment"
android:label="SearchMovieFragment" />
</navigation>
main_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".activity.MainActivity">
<include
layout="@layout/appbar" />
<androidx.fragment.app.FragmentContainerView
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintBottom_toBottomOf="parent"
app:defaultNavHost="true"
app:navGraph="@navigation/nav_graph" />
</androidx.constraintlayout.widget.ConstraintLayout>
MainActivity.kt
class MainActivity : AppCompatActivity() {
private lateinit var toolbar: Toolbar
private val navController by lazy {
(supportFragmentManager.findFragmentById(R.id.main_fragment) as NavHostFragment).navController
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.main_activity)
toolbar = findViewById(R.id.mainToolBar)
setSupportActionBar(toolbar)
setupActionBarWithNavController(navController)
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
val inflater: MenuInflater = menuInflater
inflater.inflate(R.menu.main_menu_bar, menu)
return true
}
}
build.gradle (:app)
apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'kotlin-kapt'
android {
compileSdkVersion 29
buildToolsVersion "29.0.3"
defaultConfig {
applicationId "com.example.app"
minSdkVersion 23
targetSdkVersion 29
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
debug {}
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
dataBinding {
enabled true
}
}
}
dependencies {
implementation fileTree(dir: "libs", include: ["*.jar"])
// Retrofit Libraries
implementation 'com.squareup.retrofit2:retrofit:2.6.0'
implementation 'com.squareup.retrofit2:converter-gson:2.6.0'
// Picasso Library
implementation 'com.squareup.picasso:picasso:2.71828'
implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
implementation 'androidx.core:core-ktx:1.3.0'
implementation 'androidx.appcompat:appcompat:1.1.0'
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0'
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.2.0'
implementation 'androidx.legacy:legacy-support-v4:1.0.0'
implementation 'androidx.recyclerview:recyclerview:1.1.0'
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
// Navigation libraries
def nav_version = "2.3.0"
implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"
implementation "androidx.navigation:navigation-ui-ktx:$nav_version"
}
您可以在 activity 中找到 NavController
为:
class MainActivity : AppCompatActivity() {
private lateinit var toolbar: Toolbar
private lateinit var navController: NavController
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.main_activity)
toolbar = findViewById(R.id.mainToolBar)
setSupportActionBar(toolbar)
navController = findNavController(R.id.nav_host_fragment)
setupActionBarWithNavController(navController)
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
val inflater: MenuInflater = menuInflater
inflater.inflate(R.menu.main_menu_bar, menu)
return true
}
}
private val navController by lazy {
(supportFragmentManager.findFragmentById(R.id.main_fragment) as NavHostFragment).navController
}
您的 navController
为空,因为我认为 supportFragmentManager.findFragmentById(R.id.main_fragment)
returns 为空。根据您的 activity 布局,您的 navHost 的 ID 是 R.id.nav_host_fragment
。
我有同样的问题,我通过从 Design、Palette 创建 NavHostFragment 解决了这个问题,然后分配相应的导航资源文件.
我解决问题的方法是转到 Activity.xml 并将片段 class/component 标记从 androidx.fragment.app.FragmentContainerView
更改为仅 fragment
。然后它出现在导航图的主机面板中。
如Doc中所述,您还可以使用布局编辑器添加 NavHostFragment。
- 在项目文件列表中,双击 activity 的布局 XML 文件以在布局编辑器中将其打开。
- 在调色板窗格中,选择容器类别,或者搜索“NavHostFragment”。
- 将 NavHostFragment 视图拖到 activity。
- 接下来,在出现的 Navigation Graphs 对话框中,选择相应的导航图与此 NavHostFragment 关联,然后单击确定。
YOU MAY NEED TO RESTART THE ANDROID STUDIO AFTER THIS (not mentioned in the doc)
此外,当您尝试使用上述方法添加片段时,您可能会看到“片段”而不是“FragmentContainerView”。
对我来说,我在 FragmentContainerView
中添加了 android:name="androidx.navigation.fragment.NavHostFragment"
并与我一起工作
我们应该确保写下这一行
android:name="androidx.navigation.fragment.NavHostFragment"
<androidx.fragment.app.FragmentContainerView
android:id="@+id/navController"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="0dp"
android:layout_height="0dp"
app:defaultNavHost="true"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:navGraph="@navigation/nav_graph" />
最初我们没有 NavHostFragments。你将每个片段指向彼此,创建了运动动作,但你需要一些东西来承载这个运动。这就是 navigation host fragment
发挥作用的地方。 NavHostFragment 会根据需要在给定布局内交换我们的片段。那么什么布局应该容纳这个导航片段。它是在主要 activity 期间膨胀的布局。于是打开activity_main.xml
.
你应该加上Containers/NavHostFragment
选择NavHostFragment拖拽时,会看到可用的宿主片段。