Android Jetpack套件之Navigation导航组件

本文解决了一个在使用Android Navigation组件时遇到的问题:'nocurrentnavigationnode'异常。通过取消Fragment对NavHostFragment的继承解决了该问题。同时,文章详细介绍了Navigation组件的使用步骤,包括创建navigation目录、XML文件、Activity及Fragment,以及如何通过代码实现页面跳转。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首先记录一个坑:no current navigation node

06-26 14:42:25.787 25913-25913/com.bluetree.bytepay2 E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.bluetree.bytepay2, PID: 25913
    java.lang.IllegalStateException: no current navigation node
        at androidx.navigation.NavController.navigate(NavController.java:764)
        at androidx.navigation.NavController.navigate(NavController.java:744)
        at androidx.navigation.NavController.navigate(NavController.java:730)
        at androidx.navigation.NavController.navigate(NavController.java:718)
        at com.bluetree.bytepay2.sample.BlankFragment$1.onClick(BlankFragment.java:83)
        at android.view.View.performClick(View.java:5264)
        at android.view.View$PerformClick.run(View.java:21297)
        at android.os.Handler.handleCallback(Handler.java:743)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:150)
        at android.app.ActivityThread.main(ActivityThread.java:5546)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:794)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:684)

解决方案:navigation.xml 中的Fragment继承了NavHostFragment,取消继承就可以了!

Navigation导航组件

概述

Navigation是什么,有什么意义,内涵之类的这里就不展开篇幅讲了,能贴图片绝不多说半句废话。这篇文章只做一个引入的Navigation的用途,适合新手上路。简单来说,可以用Navigation来进行一些简单的,并直观的页面跳转。

 

使用步骤

引入库就不提,因为当你使用到Navigation的时候,AS会自动帮你导入的,下面简单介绍一下如何使用Navigation

  1. 创建navigation目录
  2. 创建navigation.xml文件
  3. 首先创建一个简单的Activity(SampleNavigationActivity)
  4. 在布局文件(R.layout.activity_sample_navigation)中引入androidx.navigation.fragment.NavHostFragment
  5. 创建两个普通的Fragment(BlankFragment1,BlankFragment2)
  6. 打开navigation.xml,把两个fragment添加进去。用箭头直观地展示跳转逻辑。
  7. 通过一句代码实现跳转逻辑

 

具体实现

 

创建navigation目录

res右键选择新建资源目录

 

创建navigation资源文件

navigation目录右键,如图依次选择

<?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/navigation"
>
<!--todo 添加fragment进行愉快地拖拽-->

</navigation>

新建activity

public class SampleNavigationActivity extends AppCompatActivity  {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_sample_navigation);
    }
}

布局文件中添加navigation容器

如下图,引入NavHostFragment,app:defaultNavHost设置默认的navigation文件,并且拦截返回事件

app:navGraph用于指定navigation.xml文件

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout 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:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".sample.SampleNavigationActivity">

    <fragment
        android:id="@+id/fragment"
        android:name="androidx.navigation.fragment.NavHostFragment"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:defaultNavHost="true"
        app:navGraph="@navigation/navigation" />

</FrameLayout>

创建n个Fragment用于切换

Fragment2也是一样的,这里不重复写了

private const val ARG_PARAM1 = "param1"
private const val ARG_PARAM2 = "param2"

/**
 * A simple [Fragment] subclass.
 *
 */
class BlankFragment1 : Fragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
                              savedInstanceState: Bundle?): View? {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.fragment_blank1, container, false)
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
       
    }

}
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".sample.BlankFragment1">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:text="@string/hello_blank_fragment" />
    <Button
        android:id="@+id/btn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Button" />
</FrameLayout>

 

打开navigation.xml,把两个fragment添加进去,愉快地用箭头直观地展示跳转逻辑

添加左上角的new destination 添加目标界面,然后就可以愉快地玩耍了

 

通过一句代码实现跳转逻辑

通过上面的步骤已经规划好了,最后需要用代码来触发跳转,例如由Fragment1点击按钮跳转到Fragment2,则我们需要编辑Fragment1的按钮点击事件来完成这个逻辑,修改Fragment1的代码如下

class BlankFragment1 : Fragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
                              savedInstanceState: Bundle?): View? {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.fragment_blank1, container, false)
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        view.findViewById<Button>(R.id.btn).setOnClickListener {
            //实现跳转
            Navigation.findNavController(getView()!!).navigate(R.id.action_blankFragment1_to_blankFragment2)
        }
    }

}

例子

https://github.com/KubyWong/JetpackSample.git

 

 

 

 

 

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值