flutter之从零开始搭建(二)之 Navigator路由

2024-08-24 07:48

本文主要是介绍flutter之从零开始搭建(二)之 Navigator路由,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  • flutter之从零开始搭建(一)之 BottomNavigationBar

  • flutter之从零开始搭建(二)之 Navigator路由

  • flutter之从零开始搭建(二)之 网络请求

今天我们来讲讲利用Navigator来跳转页面的功能,承接上一篇flutter之从零开始搭建(一)之 BottomNavigationBar继续讲。

页面跳转是我们在入门学习的必备知识,在flutter中,路由跳转是由Navigator来操作的。

Navigator的跳转有两种,一种是显示跳转,需要我们在MaterialPageRoute中指定widget

  Navigator.of(context).push(new MaterialPageRoute(builder: (context) {//指定跳转的页面return new Demo1();},));

另一种是隐身跳转,这种跳转需要先定义,后使用,跳转方式就像Arouter一样的路径方式,定义部分需要在MaterialApp下定义routes,routes就跟一个Map<path,Page>集合一样,定义好了path对应的page,那么下次跳转,我们就可以针对path去跳转了

 @overrideWidget build(BuildContext context) {return new MaterialApp(home: new Scaffold(body: new Center(child: new RaisedButton(onPressed: _pushPage,child: new Text("跳转"))),),//定义路由routes: <String,WidgetBuilder>{"/demo1":(BuildContext context)=>new Demo1(),},);

跳转使用

 Navigator.of(context).pushNamed("/demo1");

知道路由如何跳转了,那么,我们开始实战吧

实战

我们看到,路由的跳转都带着一个context参数,这其中有无数的坑需要自己去理解,接下来我会展示出来。

还是承接上一篇博文flutter之从零开始搭建(一)之 BottomNavigationBar,我们先看下MyPage页面,先给出如下代码

import 'package:flutter/material.dart';
import 'package:codelang/widget/Demo1.dart';class MyPage extends StatelessWidget {@overrideWidget build(BuildContext context) {return new MaterialApp(theme: new ThemeData(primarySwatch: Colors.blue,),debugShowCheckedModeBanner: false,home: new PageWidget(),routes: <String,WidgetBuilder>{"/demo1":(BuildContext context)=>new Demo1(),},);}
}class PageWidget extends StatefulWidget {@overrideState<StatefulWidget> createState() {// TODO: implement createStatereturn new PageState();}
}class PageState extends State<PageWidget> {@overrideWidget build(BuildContext context) {return new Scaffold(body: new Center(child: new RaisedButton(onPressed: _pushPage,child: new Text("跳转"),),),);}_pushPage(){// Navigator.of(context).pushNamed("/demo1");Navigator.of(context).push(new MaterialPageRoute(builder: (context) {return new Demo1();},));}
}

大家整体看一下代码,其实没什么错误的地方,看起来一切都很正常,然后我们来看下效果图

image

what fuck!!! 跳转的页面怎么没有覆盖全屏,所以,这种方式肯定是不可取的。

我们先停下来想想,我们当前页面PageState是在PageWidget布局上面,PageWidget还有一个上层布局叫MyPage,我们是不是可以理解为,Navigator.of(context)的这个context指向的是上层MyPage布局的context,导致了页面路由跳转是在MyPage页面进行。

那么有什么解决办法呢?既然是context原因,那么,我们必须得拿到MyPage的上一层context,我们再想想,MyPage相当于一个fragment,是由MainPage页面组成的,我们只需要拿将MainPage页面的context传递给MyPage不就行了吗?接下来开干试试。

打开MainPage

给MyPage的构造方法传递MainPage的context

    _bodys = [new HomePage(),new ShopPage(),new MsgPage(),new MyPage(context)];

打开MyPage

MyPage拿到MainPage的parentContext,然后将parentContext传递给PageWidgetPageState可以通过widget.X的形式,可以拿到PageWidget的变量

import 'package:flutter/material.dart';
import 'package:codelang/widget/Demo1.dart';class MyPage extends StatelessWidget {var parentContext;MyPage(this.parentContext);@overrideWidget build(BuildContext context) {return new MaterialApp(title: 'Flutter Demo',theme: new ThemeData(primarySwatch: Colors.blue,),debugShowCheckedModeBanner: false,home: new PageWidget(parentContext),routes: <String,WidgetBuilder>{"/demo1":(BuildContext context)=>new Demo1(),},);}
}class PageWidget extends StatefulWidget {var parentContext;PageWidget(this.parentContext);@overrideState<StatefulWidget> createState() {// TODO: implement createStatereturn new PageState();}
}class PageState extends State<PageWidget> {@overrideWidget build(BuildContext context) {return new Scaffold(body: new Center(child: new RaisedButton(onPressed: _pushPage,child: new Text("跳转"),),),);}_pushPage(){
//    Navigator.of(widget.parentContext).pushNamed("/demo1");Navigator.of(widget.parentContext).push(new MaterialPageRoute(builder: (context) {return new Demo1();},));}
}

看下效果图

image

嗯,很完美的解决,大家有没有看到我_pushPage方法中注释了的pushNamed方式的跳转,大家猜猜,用这种方式会不会出错?5秒钟的思考哦.

  • 倒计时5s
  • 倒计时4s
  • 倒计时3s
  • 倒计时2s
  • 倒计时1s

OK,来揭晓答案,肯定是不行的,我们知道,routes的定义是在MyPage中的,而我们饿路由跳转拿到的parentContextMainPage的,所以,会报找不到这个路由的错误,如何解决呢?我相信到这大家应该都清楚了,那就是在MainPage中去定义这个routes,具体可以看如下

class MainPage extends StatelessWidget {@overrideWidget build(BuildContext context) {return new MaterialApp(debugShowCheckedModeBanner: false,routes: <String, WidgetBuilder>{"/demo1": (BuildContext context) => new Demo1(),},home: new MainPageWidget());}
}

这样跳转就可以了,效果图我就不贴出来了,跟上面一样。


你以为就这样结束了?哈哈,早着呢,在写这篇博文的时候,我又发现了一个好玩的地方,为了区分,我在HomePage页面去写这个例子,代码不多,大家看看

HomePage

import 'package:flutter/material.dart';
import 'package:codelang/widget/Demo1.dart';class HomePage extends StatefulWidget {@overrideState<StatefulWidget> createState() {// TODO: implement createStatereturn new HomeState();}
}class HomeState extends State<HomePage> {@overrideWidget build(BuildContext context) {// TODO: implement buildreturn new MaterialApp(debugShowCheckedModeBanner: false,home: new Scaffold(body: new Center(child: new RaisedButton(onPressed: _pushPage,child: new Text("跳转"))),),);}_pushPage() {
//    Navigator.of(context).pushNamed("/demo1");Navigator.of(context).push(new MaterialPageRoute(builder: (context) {return new Demo1();},));}
}

在这个路由跳转中,我用了context来跳转,大家再猜猜,这种方式可以嘛?5秒钟的思考时间

  • 倒计时5s
  • 倒计时4s
  • 倒计时3s
  • 倒计时2s
  • 倒计时1s

答案揭晓,是可以的哦,为什么这样又可以了呢?

HomePageMyPage的区别在于定义这个widget时,MyPage最外套的一层是StatelessWidget,而HomePage最外套的一层是StatefulWidgetStatelessWidget是一个无状态的widget,难道是他阻隔了context的传递?具体的我也不清楚,只能去猜。

大家再猜猜还是那段注释了的pushNamed的代码,可不可以跳转呢?哈哈,当然是可以的,因为我们在前面的时候,就已经在mainPage中定义了routes


好了,路由这篇说完了,唯一坑就是context的问题,看了这一篇,相信很多人都理解了Navigator如何跳转,下一篇再见吧

gayhub链接

这篇关于flutter之从零开始搭建(二)之 Navigator路由的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1101880

相关文章

利用Python快速搭建Markdown笔记发布系统

《利用Python快速搭建Markdown笔记发布系统》这篇文章主要为大家详细介绍了使用Python生态的成熟工具,在30分钟内搭建一个支持Markdown渲染、分类标签、全文搜索的私有化知识发布系统... 目录引言:为什么要自建知识博客一、技术选型:极简主义开发栈二、系统架构设计三、核心代码实现(分步解析

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

SpringIntegration消息路由之Router的条件路由与过滤功能

《SpringIntegration消息路由之Router的条件路由与过滤功能》本文详细介绍了Router的基础概念、条件路由实现、基于消息头的路由、动态路由与路由表、消息过滤与选择性路由以及错误处理... 目录引言一、Router基础概念二、条件路由实现三、基于消息头的路由四、动态路由与路由表五、消息过滤

Flutter打包APK的几种方式小结

《Flutter打包APK的几种方式小结》Flutter打包不同于RN,Flutter可以在AndroidStudio里编写Flutter代码并最终打包为APK,本篇主要阐述涉及到的几种打包方式,通... 目录前言1. android原生打包APK方式2. Flutter通过原生工程打包方式3. Futte

使用DeepSeek搭建个人知识库(在笔记本电脑上)

《使用DeepSeek搭建个人知识库(在笔记本电脑上)》本文介绍了如何在笔记本电脑上使用DeepSeek和开源工具搭建个人知识库,通过安装DeepSeek和RAGFlow,并使用CherryStudi... 目录部署环境软件清单安装DeepSeek安装Cherry Studio安装RAGFlow设置知识库总

Linux搭建Mysql主从同步的教程

《Linux搭建Mysql主从同步的教程》:本文主要介绍Linux搭建Mysql主从同步的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux搭建mysql主从同步1.启动mysql服务2.修改Mysql主库配置文件/etc/my.cnf3.重启主库my

国内环境搭建私有知识问答库踩坑记录(ollama+deepseek+ragflow)

《国内环境搭建私有知识问答库踩坑记录(ollama+deepseek+ragflow)》本文给大家利用deepseek模型搭建私有知识问答库的详细步骤和遇到的问题及解决办法,感兴趣的朋友一起看看吧... 目录1. 第1步大家在安装完ollama后,需要到系统环境变量中添加两个变量2. 第3步 “在cmd中

Flutter监听当前页面可见与隐藏状态的代码详解

《Flutter监听当前页面可见与隐藏状态的代码详解》文章介绍了如何在Flutter中使用路由观察者来监听应用进入前台或后台状态以及页面的显示和隐藏,并通过代码示例讲解的非常详细,需要的朋友可以参考下... flutter 可以监听 app 进入前台还是后台状态,也可以监听当http://www.cppcn

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.