We believe in AI and every day we innovate to make it better than yesterday.
We believe in helping others to benefit from the wonders of AI and also in
extending a hand to guide them to step their journey to adapt with future.
Google’s UI toolkit, Flutter has caught the attention of a lot of developers, business owners etc. App development can be made much faster using Flutter and it also offers expressive and flexible UI. In a previous article, I shared my insights on using Flutter for mobile app development.
Navigation between widgets/screens is very important for all mobile applications. In this article, I’ll explain in step by step how to implement navigation between widgets and screens in Flutter and pass data between Widgets/Screens. The completely open-source SDK of Flutter is based on the Dart programming language, which is new, but easy to learn and get started with.
Steps to navigate between Widgets/Screens:
Create two widgets (screens) in your flutter lib folder.
Use Navigator.push(context, Route<T>)Navigator.push(context, Route<T>) method to navigate to Widget/Screen.
Use Navigator.pop(context)Navigator.pop(context)method to navigate back to the previous Widget/Screen in the stack.
Implement Named Routing by using initialRouteinitialRouteand routesroutesproperties inMaterialAppMaterialApp widget.
Use Navigator.pushNamed(context,routeName)Navigator.pushNamed(context,routeName)method to navigate between Widgets using named routes.
Share Data between components using arguments property in pushNamed()pushNamed() and pop()pop() method in Navigator
Create Two Widgets / Screens in your Flutter Lib folder:
Create two stateless widgets under lib/screens folder, one is called home_screen.dart and another one is profile_screen.dart.
In home_screen.dart, add a FlatButtonFlatButton widget as follows to navigate to profile screen.
In order to navigate between widgets/screens, Flutter offers NavigatorNavigator class which has many methods.
Inside the anonymous function of the onPressedonPressed, we are invoking the navigateToProfile(BuildContext context)navigateToProfile(BuildContext context). We pass the contextcontext from our build method because we need to use it for the flutter Navigator.push()Navigator.push() method.
To navigate to the profile screen, we should use the pushpush method of the NavigatorNavigator, the push method needs a contextcontext as a first argument and Route<T>Route<T> as the second argument. In our case, we can use built-in MaterialRoute()MaterialRoute() which flutter offers.
The MaterialRoute(builder: (ctx) => widget())MaterialRoute(builder: (ctx) => widget()) has a property called builder which accepts a function with a context parameter and returns a Widget. In our case, we need to navigate to Profile Screen. To do that, we are invoking the ProfileScreen()ProfileScreen() widget inside the builder anonymous method
Navigate back to Previous Screens in Flutter – Back to Home Screen
As you see in the preview, that the AppBarAppBar itself has a back button for navigating back to the previous screen. But anyway, we may face a situation where we need to create a button inside the body of the scaffold and navigate back to previous screens.
To navigate back, we can use the poppop method of the NavigatorNavigator. The pop()pop() method removes the current widget which was pushed using the push()push() method, which in turn moves back to the previous screen.
In other words, the navigation in flutter works like a stack. If we push a widget, It will be placed on top of the other screens. If we invoke the pop()pop() method, then the screen at the top of the stack will be removed that results in navigating back to the previous screen.
Let’s create a backToHomebackToHome method in Profile Screen Widget to handle the Navigator.of(context).pop()Navigator.of(context).pop() as follows
For a large application, creating a MaterialRouteMaterialRoute on the fly for navigation is not an ideal solution. We need to maintain all the routes in one place, to solve that, flutter offers named routing.
Define Routes in MaterialApp
The MaterialAppMaterialApp has a way to define routes for our application. The initialRouteinitialRoute property is used to define the default page or landing page of our application. The routesroutes property is a Map with String as Key and Function which returns Widget as it’s value.
Before defining our routes in the material app, we need to add the route name for our Widgets/Screens. We can hard code the route names in the routes argument but that’s not the ideal way to do, because we need to hard code the route name in all the places where we need to navigate. So, the best practice is to use static property inside each Widget/Screen as follows.
Inside the home_screen.dart, define the routeNamerouteName property as follows:
static const routeName = '/';
static const routeName ='/';
static const routeName = '/';
In your profile_screen.dart, define the routeNamerouteName property as follows: (you can use any property name you want, I am using routeNamerouteName)
static const routeName = '/profile';
static const routeName ='/profile';
static const routeName = '/profile';
Add initialRouteinitialRoute and routesroutes properties in MaterialAppMaterialApp (main.dart), Don’t forget to remove homehome property when we use initialRouteinitialRoute for our application.
import 'package:flutter/material.dart';
import './screens/home_screen.dart';
import './screens/profile_screen.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute: '/',
routes: {
HomeScreen.routeName: (ctx) => HomeScreen(),
ProfileScreen.routeName: (ctx) => ProfileScreen()
},
);
}
}
import'package:flutter/material.dart';
import'./screens/home_screen.dart';
import'./screens/profile_screen.dart';
void main()=> runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context){
return MaterialApp(
title:'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute:'/',
routes:{
HomeScreen.routeName:(ctx)=> HomeScreen(),
ProfileScreen.routeName:(ctx)=> ProfileScreen()
},
);
}
}
import 'package:flutter/material.dart';
import './screens/home_screen.dart';
import './screens/profile_screen.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute: '/',
routes: {
HomeScreen.routeName: (ctx) => HomeScreen(),
ProfileScreen.routeName: (ctx) => ProfileScreen()
},
);
}
}
The HomeScreen.routeNameHomeScreen.routeName and ProfileScreen.routeNameProfileScreen.routeName are the static properties that we have defined in our Widgets. These are mapped to anonymous function which returns our respective widgets.
Navigate using Named Routes in Flutter
To use named routing in our home screen widget, we can replace the pushpush method to pushNamedpushNamed method. The pushNamedpushNamed method, the first argument is the contextcontext and the second one is the route name. Let’s modify our home screen widget.
As of now, we are using the stack method of flutter navigation. So to remove the old pages completely from the stack while navigating to the new page, we can replace the pushNamedpushNamed with pushReplacementNamedpushReplacementNamed. It will clear the current page from the stack and push the new page into the stack.
In the preview, you can clearly see that the back button in the app bar is no more provided by flutter. Because the stack has only one item that is the current widget. And also, you can’t use the pop()pop() method if there are no previous screens in the stack. If you press the ‘back to home’ button then you’ll see only a black blank screen.
To move to the previous screen properly, you need to use the pushReplacementNamedpushReplacementNamed method inside the ‘back to home’ button. but I am not going to do that now. As I have to show you how to pass data between routes in the next section.
How to pass data between routes – Navigation in Flutter?
In a large application, you may face a situation where you need to pass data between routes, that is from one widget to another widget.
Pass/Share Data using pushNamed or pushReplacementNamed
The pushNamedpushNamed and pushReplacementNamedpushReplacementNamed have a property called arguments. We can assign any value to the argument and pass it to the navigating widget.
For brevity, I am sharing some text data to the Profile Screen Widget. In order to access the data in the profile screen widget, we need to use ModalRoute.of(context).settings.argumentModalRoute.of(context).settings.argument inside the build method.
You can assign the value to a variable, and bind the data in the widget. I have passed the data to the Text Widget above the FlatButtonFlatButton.
Pass/Share Data using Pop method – Navigation in Flutter
The Navigator.pop()Navigator.pop() method has a second optional argument which is used to pass data to the previous screens when we navigate back. Let’s implement a small example in our profile screen and home screen.
In your profile screen, send a boolean value via pop()pop() method as follows
In your home screen widget, use pushNamedpushNamed. The pushNamedpushNamed, pushpush and pushReplacementNamedpushReplacementNamed will return a Future. If you are from a JavaScript background, The Future is a Promise in Dart. You can use thenthen method which accepts a function as a parameter to receive data from the profile screen to the home screen.
Declare a property called showTextshowText with a value false in your home screen widget. Add the thenthen method to the pushNamed and update the showTextshowText value once it’s resolved.
Based on the showTextshowText value, we can show a different Text Widget in our home screen component. Just to test that it’s working. So our final home screen widget code is:
import 'package:flutter/material.dart';
import './screens/home_screen.dart';
import './screens/profile_screen.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute: '/',
routes: {
HomeScreen.routeName: (ctx) => HomeScreen(),
ProfileScreen.routeName: (ctx) => ProfileScreen()
},
);
}
}
import'package:flutter/material.dart';
import'./screens/home_screen.dart';
import'./screens/profile_screen.dart';
void main()=> runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context){
return MaterialApp(
title:'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute:'/',
routes:{
HomeScreen.routeName:(ctx)=> HomeScreen(),
ProfileScreen.routeName:(ctx)=> ProfileScreen()
},
);
}
}
import 'package:flutter/material.dart';
import './screens/home_screen.dart';
import './screens/profile_screen.dart';
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
// This widget is the root of your application.
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
initialRoute: '/',
routes: {
HomeScreen.routeName: (ctx) => HomeScreen(),
ProfileScreen.routeName: (ctx) => ProfileScreen()
},
);
}
}
Conclusion – Navigation in Flutter
I hope, you have learned how to navigate between widgets in flutter and also how to share data back and forth using ModalRoute and pop method in flutter.
Senior Frontend Developer with a proven track record working on industry-leading projects. Able to apply user-centric, design-thinking with technical ... Read more