6

我正在尝试使用视图绑定替换所有 findViewById。但是,我无法使用View Binding更改我的 NavController 代码行。

val navController = findNavController(this, R.id.mainHostFragment)

var binding : ActivityMainBinding
val navController = findNavController(this, binding.mainHostFragment)

我怎样才能做到这一点?

4

4 回答 4

3

您不能用视图绑定替换它。findNavController 不仅仅是在布局中查找视图。

看看这里的源代码

/**
* Find a {@link NavController} given a local {@link Fragment}.
*
* <p>This method will locate the {@link NavController} associated with this Fragment,
* looking first for a {@link NavHostFragment} along the given Fragment's parent chain.
* If a {@link NavController} is not found, this method will look for one along this
* Fragment's {@link Fragment#getView() view hierarchy} as specified by
* {@link Navigation#findNavController(View)}.</p>
*
* @param fragment the locally scoped Fragment for navigation
* @return the locally scoped {@link NavController} for navigating from this {@link Fragment}
* @throws IllegalStateException if the given Fragment does not correspond with a
* {@link NavHost} or is not within a NavHost.
*/
@NonNull
public static NavController findNavController(@NonNull Fragment fragment) {
Fragment findFragment = fragment;
while (findFragment != null) {
    if (findFragment instanceof NavHostFragment) {
        return ((NavHostFragment) findFragment).getNavController();
    }
    Fragment primaryNavFragment = findFragment.getParentFragmentManager()
            .getPrimaryNavigationFragment();
    if (primaryNavFragment instanceof NavHostFragment) {
        return ((NavHostFragment) primaryNavFragment).getNavController();
    }
    findFragment = findFragment.getParentFragment();
}
// Try looking for one associated with the view instead, if applicable
View view = fragment.getView();
if (view != null) {
    return Navigation.findNavController(view);
}
throw new IllegalStateException("Fragment " + fragment
        + " does not have a NavController set");
}

它不仅仅是找到控制器。它遍历、创建片段、创建视图并抛出异常。

视图绑定只是生成一个绑定类,其中包含布局的所有视图。它不适用于查找应用程序的导航控制器。

于 2020-03-18T05:25:25.667 回答
3

在这个答案的帮助下,我找到了一个简单的实现

binding?.apply {
        setContentView(root)
        setSupportActionBar(toolbar)
        navController = (supportFragmentManager
            .findFragmentById(fragmentHost.id) as NavHostFragment)
            .navController
        setupActionBarWithNavController(navController, appBarConfiguration)
        bottom.setupWithNavController(navController)
    }
于 2020-06-30T03:29:53.100 回答
2

findNavController(R.id.nav_host_fragment)应该在将视图附加到这样的活动之后:

setContentView(binding.root)
val navController: NavController = findNavController(R.id.nav_host_fragment) 

更多在这里

于 2021-02-27T14:14:05.720 回答
1

这是我的示例代码,使用视图绑定和导航。

import android.os.Bundle
import androidx.appcompat.app.AppCompatActivity
import androidx.navigation.fragment.NavHostFragment
import androidx.navigation.ui.AppBarConfiguration
import androidx.navigation.ui.setupActionBarWithNavController
import androidx.navigation.ui.setupWithNavController
import com.zeddigital.zigmaster.databinding.ActivityMainBinding


class MainActivity : AppCompatActivity() {

    private lateinit var appBarConfiguration: AppBarConfiguration
    private lateinit var binding : ActivityMainBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        /*
        Use view binding in activities
        
        Call the static inflate() method included in the generated binding class. 
        This creates an instance of the binding class for the activity to use.
        Get a reference to the root view by either calling the getRoot() method or using Kotlin property syntax.
        Pass the root view to setContentView() to make it the active view on the screen.*/
        binding = ActivityMainBinding.inflate(layoutInflater)

        val view = binding.root
        setContentView(view)

        setSupportActionBar(binding.appBarMain.toolbar)

        //val navController = findNavController(R.id.nav_host_fragment)
        val navHostFragment = supportFragmentManager.findFragmentById(R.id.nav_host_fragment) as NavHostFragment
        val navController = navHostFragment.navController

        // Passing each menu ID as a set of Ids because each
        // menu should be considered as top level destinations.
        appBarConfiguration = AppBarConfiguration(setOf(
                R.id.nav_home, R.id.nav_gallery, R.id.nav_slideshow), binding.drawerLayout)
        setupActionBarWithNavController(navController, appBarConfiguration)
        binding.navView.setupWithNavController(navController)

    }
}
于 2021-03-06T02:24:09.637 回答