-
Notifications
You must be signed in to change notification settings - Fork 56
injected_navigator_api
This is an easy to use implementation of the new Navigator version 2 API.
IMPORTANT Navigation is extract to its own package: navigation_builder. it has the same api with minor changes.
Now states_rebuilder
use navigation_builder
without any breaking changes compared to previous versions.
- Setting Navigator 2
- InjectedNavigator
- Navigation
- InjectedNavigator reactivity
- Page transition animation
- Nested routes
- Redirection
For more simple and practical examples of navigation, please refer to the navigation's set of [examples](https://github.com/GIfatahTH/states_rebuilder/blob/dev/examples/ex004_00_navigation
To set navigator 2 you only need to define a global InjectedNavigator
final variable and use MaterialApp.router
or CupertinoApp.router
like this:
final myNavigator = RM.injectNavigator(
routes: {
...
},
);
class MyApp extends StatelessWidget {
const MyApp({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return MaterialApp.router(
routeInformationParser: myNavigator.routeInformationParser,
routerDelegate: myNavigator.routerDelegate,
);
}
}
The first step to set Navigator 2 API is to define how route names are mapped to their corresponding pages using RM.injectNavigator
method:
final InjectedNavigator myNavigator = RM.injectNavigator(
// Define your routes map
routes: {
'/': (RouteData data) => Home(),
// redirect all paths that starts with '/home' to '/' path
'/home/*': (RouteData data) => data.redirectTo('/'),
'/page1': (RouteData data) => Page1(),
'/page1/page11': (RouteData data) => Page11(),
'/page2/:id': (RouteData data) {
// Extract path parameters from dynamic links
final id = data.pathParams['id'];
// Or inside Page2 you can use `context.routeData.pathParams['id']`
return Page2(id: id);
},
'/page3/:kind(all|popular|favorite)': (RouteData data) {
// Use custom regular expression
final kind = data.pathParams['kind'];
return Page3(kind: kind);
},
'/page4': (RouteData data) {
// Extract query parameters from links
// Ex link is `/page4?age=4`
final age = data.queryParams['age'];
// Or inside Page4 you can use `context.routeData.queryParams['age']`
return Page4(age: age);
},
'/page5/bookId': (RouteData data) {
// As deep link can have data out of boundary.
try {
final bookId = data.queryParams['bookId'];
final book = books[int.parse(bookId)];
return Page5(book: book);
} catch {
// bookId can be a non valid number or it can be greater than books length.
// Dispay the unknownRoute widget
return data.unknownRoute;
}
},
// Using sub routes
'/page6': (RouteData data) => RouteWidget(
builder: (Widget routerOutlet) {
return MyParentWidget(
child: routerOutlet;
// Or inside MyParentWidget you can use `context.routerOutlet`
)
},
routes: {
'/': (RouteData data) => Page6(),
'/page51': (RouteData data) => Page61(),
},
),
},
);
The first parameter is routes
which is of type Map<String, Widget Function(RouteData)>
.
The map entries of the routes
parameter can be:
-
Simple routes such as:
'/': (RouteData data) => Home(), '/page1': (RouteData data) => Page1(),
-
Simple nested routes:
'/': (RouteData data) => Home(), '/page1': (RouteData data) => Page1(), '/page1/page11': (RouteData data) => Page11(), '/page1/page11/page111': (RouteData data) => Page111(), '/page1/page11/page111/page1111': (RouteData data) => Page1111(),
When the app first starts and if the initial location is set to
/page1/page11/page111/page1111
, the route stack will hold['/', '/page1', '/page1/page11', '/page1/page11/page111', '/page1/page11/page111/page111']
. That is, the five pages are inflated on top of each other.But if you are in page
'/'
and you navigate to/page1/page11/page111/page1111
usingmyNavigator.to
method, the route stack will hold only two pages:['/', '/page1/page11/page111/page111']
.You can navigate to
/page1/page11/page111/page1111
and inflate all intermediate routes usingmyNavigator.toDeeply
. For more information see Imperative navigation .You can simplify the routes above using
RouteWidget
:'/': (RouteData data) => Home(), '/page1': (RouteData data) => RouteWidget( routes: { '/': (RouteData data) => Page1(), '/page11': (RouteData data) => RouteWidget( routes: { '/': (RouteData data) => Page11(), '/page111': (RouteData data) => Page111(), '/page111/page1111': (RouteData data) => Page1111(), }, ), } ),
Both way of writing routes are equivalent.
-
Dynamic link routes:
'/': (RouteData data) => Home(), '/books/': (RouteData data) => Books(), '/books/:bookId': (RouteData data) { final bookId = data.pathParam['bookId']; return BooksDetails(bookId: bookId), // Or just return BooksDetails() without parameters and get the book id using: // `context.routeData.pathParam['bookId']` inside the builder method of BooksDetails. }, '/books/:bookId/authors': (RouteData data) => data.redirectTo('/authors'), '/authors': (RouteData data) { // As we are redirected here from '/books/:bookId/authors' we can get the book id. final bookId = data.pathParam['bookId']; // The location we are redirected from. // For example `books/1/authors` final redirectedFrom = data.redirectedFrom.location; return Authors(); }, '/authors/:authorId': (RouteData data) { final authorId = data.pathParam['authorId']; return AuthorDetails(); },
From the exposed
RouteData
we can get useful routing information. Let's suppose we are navigating tobooks/1
url:-
RouteData.location
: holds the current location we are navigating to. (books/1
in our example). -
RouteData.path
: the current resolved route path.(books/:bookId
in our example). -
RouteData.baseLocation
: the base parent location.(/
in our example). -
RouteData.pathParams
: a map of extracted path parameters.({'bookId'; '1'}
in our example). -
RouteData.queryParams
: a map of extracted query parameters.({}
in our example). But if the navigation url isbooks/1?q=1
, thequeryParams
equals{'q': '1'}
; -
RouteData.redirectedFrom
: theRouteData
we are redirected from.(null
in our example). But if we are navigating tobooks/1/authors
we will be redirected to/authors
route. From their weredirectedFrom.location
will be equal tobooks/1/authors
. SeeRedirection for more information
-
-
Custom Regular expression routes:
'/page3/:kind(all|popular|favorite)': (RouteData data) { // Use custom regular expression final kind = data.pathParams['kind']; // Or in any child of Page3, you can use context.pathParams return Page3(kind: kind); },
You can pass path parameters using regular expression.
/page/:name(ANY_VALID_REG_EXPRESSION)
What's between parenthesis is the regular expression.If the regular expression is invalid, the route is considered unknown.
A star (*) matches all sub paths:
-
/*
or just*
, matches all location urls. -
/page1/*
, matches all location urls that starts with/page1
.
-
-
RouteWidget
:
RouteWidget
is used for multiple reasons:-
For organization: Let's take this routes map:
routes: { '/': (RouteData data) => Home(), '/page1': (RouteData data) => Page1(), '/page1/page11': (RouteData data) => Page11(), '/page1/page11/page111': (RouteData data) => Page111(), '/page1/page11/page111/page1111': (RouteData data) => Page1111(), }
You can simplify the routes above using
RouteWidget
:routes: { '/': (RouteData data) => Home(), // '/page1' is considered the route name of this sub route. '/page1': (RouteData data) => RouteWidget( routes: { // '/' here means the home page of '/page1' route, which is '/page1' '/': (RouteData data) => Page1(), '/page11': (RouteData data) => RouteWidget( routes: { '/': (RouteData data) => Page11(), '/page111': (RouteData data) => Page111(), '/page111/page1111': (RouteData data) => Page1111(), }, ), } ), }
Both way of writing routes are equivalent.
-
For custom page transition animation: You can dedicate a particular page transition to a specified route.
final myNavigator = RM.injectNavigator( // Default transition transitionDuration: RM.transitions.leftToRight(), routes: { '/': (_) => HomePage(), '/Page1': (_) => RouteWidget( builder: (_) => Page1(), // You can use one of the predefined transitions transitionsBuilder: (context, animation, secondaryAnimation, child) { // Custom transition implementation return ...; }, ), '/page2': (_) => Page2(), }, );
pages
'/'
and'/page2'
will use the default transition whereas'/page1'
will use its own defined custom page transition. See Page transition animation section for detailed information. -
For sub routes (nested routes):
// '/page5' is the name of the sub route '/page5': (RouteData data) => RouteWidget( builder: (Widget routerOutlet) { return MyParentWidget( child: routerOutlet; // Or inside MyParentWidget you can use `context.routerOutlet` ) }, routes: { '/': (RouteData data) => Page5(), '/page51': (RouteData data) => Page51(), }, ),
Each sub route has its own stack of pages. The builder method is used to wrap the route outlet inside another widget. Only the outlet widget will be animated on page transition. See Nested routes for more information.
-
To navigate imperatively:
myNavigator.to('/page1');
myNavigator.toReplacement('/page1', argument: 'myArgument');
myNavigator.toAndRemoveUntil('/page1', queryParam: {'id':'1'});
myNavigator.back();
myNavigator.backUntil('/page1');
myNavigator.removePage('/page1');
To navigate declaratively:
myNavigator.setRouteStack(
(pages){
// exposes a copy of the current route stack
return [...newPagesList];
}
)
To navigate to pageless routes, show dialogs and snackBars without BuildContext
:
myNavigator.toPageless(HomePage());
RM.navigate.to(HomePage());
RM.navigate.toDialog(AlertDialog( ... ));
RM.scaffoldShow.snackbar(SnackBar( ... ));
For more information on navigation see Navigation
By default when the app first tarts it will route to '/'
. You can change this default behavior by setting the initialLocation
.
Let's take this route
routes: {
'/': (RouteData data) => Home(),
'/page1': (RouteData data) => Page1(),
'/page1/page11': (RouteData data) => Page11(),
'/page1/page11/page111': (RouteData data) => Page111(),
'/page1/page11/page111/page1111': (RouteData data) => Page1111(),
}
When the app first starts and if the initial location is set to /page1/page11/page111/page1111
, the route stack will hold ['/', '/page1', '/page1/page11', '/page1/page11/page111', '/page1/page11/page111/page111']
. That is, the five pages are inflated on top of each other.
The same thing happens in deep linking.
By default if the location fails to resolve to any of the known routes, a 404 widget is displayed.
You can set your custom 404 widget using the parameter unknownRoute
. It exposes the location url to go to.
As deep link can have data out of boundary, you can check for the validity of the extracted data and dispay the unknownRoute
if data is not valid.
final InjectedNavigator myNavigator = RM.injectNavigator(
routes: {
'/': (RouteData data) => Home(),
'/page1/bookId': (RouteData data) {
try {
final bookId = data.queryParams['bookId'];
final book = books[int.parse(bookId)];
return Page1(book: book);
} catch {
// bookId can be a non valid number or it can be greater than books length.
// Dispay the unknownRoute widget
return data.unknownRoute;
}
},
},
);
You can set the parameter ignoreUnknownRoutes
to true to ignore all unknown routes and the app stays in the last known route.
final InjectedNavigator myNavigator = RM.injectNavigator(
ignoreUnknownRoutes: true,
routes: {
'/': (RouteData data) => Home(),
...
},
);
The builder parameters is used to wrap the router outlet widget with other widgets.
In the following example the entire app is wrapped with a Scaffold
that has an AppBar
where the title displays the current location the app is in.
final myNavigator = RM.injectNavigator(
builder: (Widget routeOutlet) {
// If you extract this Scaffold to a Widget class, you do not
// need to use the Builder widget
return Scaffold(
appBar: AppBar(
title:Builder( // Needed only to get a child BuildContext
builder: (context) {
final location = context.routeData.location;
return Text('Routing to: $location');
},
),
),
body: routeOutlet,
);
},
routes: {
'/': (_) => HomePage(),
'/Page1': Page1(),
},
);
The builder callback exposes the route outlet widget. You can also obtain the route outlet widget in any of the child widget tree using context.routeOutlet
relying of the InheritedWidget
mechanism.
Notice that the InjectedNavigator
is a reactive state so we can listen to it using ReactiveStatelessWidget
. See InjectedNavigator reactivity.
By default pages are wrapped with the appropriate MaterialPage
ro CupertinoPage
depending on whether you use MaterialApp.router
or CupertinoApp.router
.
In case you want to provide your own Page
implementation you can use pageBuilder
parameter.
pageBuilder: (MaterialPageArgument arg) {
return MaterialPage(
key: arg.key,
child: arg.child,
name: arg.name,
arguments: arg.arguments,
maintainState: arg.maintainState,
fullscreenDialog: arg.fullscreenDialog,
);
},
The pageBuilder exposes MaterialPageArgument
object.
MaterialPageArgument({
LocalKey? key,
required Widget child,
String? name,
Object? arguments,
required bool maintainState,
required bool fullscreenDialog,
})
By default pages are wrapped with the appropriate MaterialPage
ro CupertinoPage
depending on whether you use MaterialApp.router
or CupertinoApp.router
.
If you want to use MaterialApp.router
and wrap the pages with CupertinoPage
, set shouldUseCupertinoPage
to true.
You can use pageBuilder
for more customization.
By default, depending on the target platform, Flutter choses the adequate page transition animation for all routes.
You can provide you own page transition to override the default transition for all routes using transitionsBuilder
final myNavigator = RM.injectNavigator(
transitionsBuilder: (context, animation, secondaryAnimation, child) {
return // Your page transition implementation
},
routes: {
...
},
);
You can also use one of the predefined transitions using RM.transitions
final myNavigator = RM.injectNavigator(
transitionsBuilder: RM.transitions.leftToRight(),
routes: {
...
},
);
You can also force pages to transit without animation using RM.transitions.none()
final myNavigator = RM.injectNavigator(
transitionsBuilder: RM.transitions.none(),
routes: {
...
},
);
The transitionsBuilder
defined here will be use for all routes. In case you want, you have the ability to override this global behavior for a particular route or just for a particular call of navigation.See Page transition animation for detailed discussion.
Define a duration for the transition animation.
This is a call back that fires every time a location is successfully resolved and just before navigation. You can use this callback for globule redirection.
It exposes the current state of the InjectedNavigator
(RouteData
).
Let's take this example:
final myNavigator = RM.injectNavigator(
onNavigate: (RouteData data) {
final toLocation = data.location;
if (toLocation == '/homePage' && userIsNotSigned) {
return data.redirectTo('/signInPage');
}
if (toLocation == '/signInPage' && userIsSigned) {
return data.redirectTo('/homePage');
}
//You can also check query or path parameters
if (data.queryParams['userId'] == '1') {
return data.redirectTo('/superUserPage');
}
},
routes: {
'/signInPage': (RouteData data) => SignInPage(),
'/homePage': (RouteData data) => HomePage(),
},
);
If the app is navigating to '/homePage'
and if the user is not signed yet, the app is redirected to navigate to the sign in page.
Also if the app is navigating to '/signInPage'
and if the user is already signed, the app is redirected to navigate to the home screen.
You can check for any property the RouteData
exposes.
Head to Redirection section for more information.
See onNavigate
en action following this example.
This callback is fired every time a route is removed and the app navigate back. It can be used to prevent leaving a page unless data is validated.
final myNavigator = RM.injectNavigator(
onNavigateBack: (RouteData data) {
if(data == null){
// onNavigateBack is called with null data when the hard back button of Android
// device is pressed with no page to go back to (the tack route length is one)
// Typically here we display a dialog to let the user to choose between staying
// in the app or exiting
// return true, the app exists
// return false the app stay alive
return false;
}
final backFrom = data.location;
if (backFrom == '/SingInFormPage' && formIsNotSaved) {
RM.navigate.toDialog(
AlertDialog(
content: Text('The form is not saved yet! Do you want to exit?'),
actions: [
ElevatedButton(
onPressed: () => RM.navigate.forceBack(),
child: Text('Yes'),
),
ElevatedButton(
onPressed: () => RM.navigate.back(),
child: Text('No'),
),
],
),
);
return false;
}
},
routes: {
'/SingInFormPage': (RouteData data) => SingInFormPage(),
'/homePage': (RouteData data) => HomePage(),
},
);
Here if the app is navigating back from sign in form page and if the form is not saved yet, the back navigation is cancelled and a Dialog
is popped asking for back navigation confirmation.
If the user chooses No, the app stays in the sign form page. In contrast if the user choose YES, the app is forcefully popped and both the Dialog
and the sign in form page are removed from the routing stack.
Here is a working example using onNavigateBack
.
If set to true, a message is printed in the console informing us about the state of the navigation.
This is the full RM.injectedNavigator
API
InjectedNavigator injectNavigator({
required Map<String, Widget Function(RouteData)> routes,
String? initialLocation,
Widget Function(String)? unknownRoute,
Widget Function(Widget)? builder,
Page<dynamic> Function(MaterialPageArgument)? pageBuilder,
bool shouldUseCupertinoPage = false,
Widget Function(BuildContext, Animation<double>, Animation<double>, Widget)? transitionsBuilder,
Duration? transitionDuration,
Redirect? Function(RouteData)? onNavigate,
bool? Function(RouteData)? onNavigateBack,
bool debugPrintWhenRouted = false,
bool ignoreUnknownRoutes =false,
})
After defining InjectedNavigator
variable and setting MaterialApp.router
, your app is ready for navigation witch can be done imperatively or declaratively.
To navigate imperatively, we use one of the methods defined in our InjectedNavigator
object which we suppose to name myNavigator
:
-
myNavigator.to('/page1')
:
Here '/page1'
route will be added on top of the route stack. If you are used to Navigator 1 API,myNavigator.to('/page1')
is exactly equivalent toRM.navigate.toNamed('/page1')
. You still can use both with Navigator 2.Future<T?> to<T extends Object?>( String routeName, { Object? arguments, Map<String, String>? queryParams, bool fullscreenDialog = false, bool maintainState = true, Widget Function(BuildContext, Animation<double>, Animation<double>, Widget)? transitionsBuilder, })
You can pass
arguments
orqueryParams
to the route. You can also decide whether the route isfullscreenDialog
and whether tomaintainsState
or not. You can also override the global page transition with the one you define here via thetransitionBuilder
parameter. ThetransitionBuilder
defined here will be applied to this particular navigation call. Any further call ofmyNavigator.to
method will use the default page transition.You can wait for results from the push route:
onPressed: ()async { final result = await myNavigator.to('/page1'); // On page `'/page1'`, if you call `myNavigator.back('This is the result')` print(result); // prints 'This is the result' }
This is a working example.
-
myNavigator.toDeeply('/page1/page11')
: Deeply navigate to the given routeName. Deep navigation means that the root stack is cleaned and pages corresponding to sub paths are added to the stack. Suppose our navigator is :final myNavigator = RM.injectNavigator( routes: { '/': (RouteData data) => HomePage(), '/page1': (RouteData data) => Page1(), '/page1/page11': (RouteData data) => Page11(), '/page1/page11/page111': (RouteData data) => Page111(), }, );
On app start up, the route stack is
['/']
. If we callmyNavigator.to('/page1/page11/page111')
, the route stack is['/', '/page1/page11/page111']
. In contrast, if we invokemyNavigator.toDeeply('/page1/page11/page111')
, the route stack is['/', '/page1', '/page1/page11', '/page1/page11/page111']
.This is a working example.
-
myNavigator.toReplacement('/page1')
: Here '/page1'
route will be added on top of the route stack and the last page will be removed. This is exactly equivalent toRM.navigate.toReplacementNamed('/page1')
. You still can use both with Navigator 2.Future<T?> toReplacement<T extends Object?, TO extends Object?>( String routeName, { TO? result, Object? arguments, Map<String, String>? queryParams, bool fullscreenDialog = false, bool maintainState = true, })
-
myNavigator.toAndRemoveUntil('/page1')
: Here '/page1'
route will be added on top of the route stack and all the previous routes until meeting the route with defined route nameuntilRouteName
are removed. If the argumentuntilRouteName
is not defined, all previous pages are removed. This is exactly equivalent toRM.navigate.toNamedAndRemoveUntil('/page1')
. You still can use both with Navigator 2.Future<T?> toAndRemoveUntil<T extends Object?>( String newRouteName, { String? untilRouteName, Object? arguments, Map<String, String>? queryParams, bool fullscreenDialog = false, bool maintainState = true, })
-
myNavigator.back()
: Pop the top-most route off the route stack. You can pass a result to complete the future that had been returned from pushing the popped route. This is exactly equivalent toRM.navigate.back()
. You still can use both with Navigator 2.void back<T extends Object>([T? result])
-
myNavigator.forceBack()
: Pop the top-most route off the route stack with all pageless route associated with it and without callingonNavigateBack
hook. This is exactly equivalent toRM.navigate.forceBack()
.void forceBack<T extends Object>([T? result])
The difference between
back
andforceBack
:back
invokesonNavigateBack
hook and pops the top most route carelessly its a page or pageless route. For example, if aDialog
(aDialog
is an example pageless route) is displayed and we invokeback
, only the Dialog is popped off. In ContrastforceBack
does not invokeonNavigateBack
hook and pops the top most page route with all pageless routes associated with it. For example, if aDialog
is displayed and we invokeforceBack
, the dialog and the top most page route are popped off.Here is an example where we want to prevent the user from leaving the from page before save data:
final myForm = RM.injectForm(); // final myNavigator = RM.injectNavigator( onNavigateBack: (RouteData data) { final backFrom = data.location; // an InjectedForm is dirty when data is modified and not saved yet if (backFrom == '/SingInFormPage' && myForm.isDirty) { RM.navigate.toDialog( AlertDialog( content: Text('The form is not saved yet! Do you want to exit?'), actions: [ ElevatedButton( // Pop off hte Dialog and exit the SingInFormPage onPressed: () => RM.navigate.forceBack(), child: Text('Yes'), ), ElevatedButton( // Pop off hte Dialog and stay in the SingInFormPage onPressed: () => RM.navigate.back(), child: Text('No'), ), ], ), ); return false; } }, routes: { '/SingInFormPage': (RouteData data) => SingInFormPage(), '/homePage': (RouteData data) => HomePage(), }, );
Here is The working example.
-
myNavigator.backUntil('/page1')
: Navigate back and remove all the previous routes until meeting the route with defined name. This is exactly equivalent toRM.navigate.backUntil('/page1')
.void backUntil(String untilRouteName)
In declarative navigation you have access to a copy of the route stack and you have the freedom to return a new stack the way you like.
myNavigator.setRouteStack(
(pages){
// exposes a copy of the current route stack
return [...newPagesList];
}
)
By default the exposes copy of route stack is that of the root route stack. If you are working with nested routes, each sub route will have its own stack.
To declaratively set the stack of a sub route use the subRouteName
parameter.
myNavigator.setRouteStack(
(pages){
// exposes a copy of the current route stack
return [...newPagesList];
},
subRouteName: '/page1',
)
Here is The working example.
Here is a quote from Flutter documentation:
The section introduces the concept of Pages to the Navigator and divides the Routes managed by the Navigator into two groups: Some Routes are backed by a Page, others are not. The latter are called pageless Routes.
myNavigator.toPageless(NextPage());
RM.navigate.to(NextPage());
You can specify a name to the route (e.g., "/settings"). It will be used with backUntil
, toAndRemoveUntil
, toAndRemoveUntil
, and toNamedAndRemoveUntil
.
RM.navigate.to(NextPage(), name: '/routeName');
// calling backUntil:
RM.navigate.backUntil('/routeName'); // Flutter: popUntil
Dialogs when displayed are pushed into the route stack. It is for this reason, in states_rebuilder, dialogs are treated as navigation:
In Flutter to show a dialog:
showDialog<T>(
context: navigatorState.context,
builder: (_) => Dialog(),
);
In states_rebuilder to show a dialog:
RM.navigate.toDialog(Dialog());
For sure, states_rebuilder is less boilerplate, but it is also more intuitive. In states_rebuilder we make it clear that we are navigating to the dialog, so to close a dialog, we just pop it from the route stack.
So states_rebuilder follows the naming convention as in Flutter SDK, with the change from show
in Flutter to to
in states_rebuilder.
RM.navigate.toDialog(DialogWidget()); // Flutter: showDialog
RM.navigate.toCupertinoDialog(CupertinoDialogWidget()); // Flutter: showCupertinoDialog
RM.navigate.toBottomSheet(BottomSheetWidget()); // Flutter: showModalBottomSheet
RM.navigate.toCupertinoModalPopup(CupertinoModalPopupWidget()); // Flutter: showCupertinoModalPopup
For all other dialogs, menus, bottom sheets, not mentioned here, you can use is as defined by flutter using RM.context
:
example:
showSearch(
context: RM.context,
delegate: MyDelegate(),
)
Some side effects require a BuildContext of a scaffold child widget.
In state_states_rebuilder to be able to display them outside the widget tree without explicitly specifying the BuildContext, we need to tell states_rebuild which BuildContext to use first.
This can be done either:
onPressed: () {
RM.scaffold.context= context;
RM.scaffold.showBottomSheet(...);
}
Or
onPressed: () {
modelRM.setState(
(s)=> doSomeThing(),
context: context,
onData: (_,__) {
RM.scaffold.showBottomSheet(...);
},
),
}
}
If you have one of the states_rebuilder widgets that is a child of the Scaffold
, you no longer need to specify a BuildContext
. The BuildContext
of this widget will be used.
Since SnackBars
, for example, depend on ScaffoldState
and aren't pushed to the route stack, we don't treat them as navigation like we did with dialogs.
To distinguish them from Dialogs and to emphasize that they need a Scaffold-related BuildContext
, we use RM.scaffold
instead of RM.navigate
.
RM.scaffold.showBottomSheet(BottomSheetWidget()); // Flutter: Scaffold.of(context).showBottomSheet
RM.scaffold.showSnackBar(SnackBarWidget()); // Flutter: Scaffold.of(context).showSnackBar
RM.scaffold.openDrawer(); // Flutter: Scaffold.of(context).openDrawer
RM.scaffold.openEndDrawer(); // Flutter: Scaffold.of(context).openEndDrawer
For anything, not mentioned here, you can use the scaffoldState exposed by states_rebuilder.
The defined InjectedNavigator is of type ReactiveModel<RouteData>
. So it is reactive and can be listen to using ReactiveStatelessWidget
, OnBuilder
widgets.
Pages are transited using the default Flutter
animation depending on the target platform.
Page transition can be customized for there levels:
While defining InjectedNavigator
object, you can set your custom page transition animation to be used for all route transitions.
final myNavigator = RM.injectNavigator(
transitionsBuilder: (context, animation, secondaryAnimation, child) {
return // Your page transition implementation
},
routes: {
...
},
);
You can dedicate a particular page transition to a specific route while other routes still use the default page transition.
dart final myNavigator = RM.injectNavigator( // Default transition transitionDuration: RM.transitions.leftToRight(), routes: { '/': (_) => HomePage(), '/Page1': (_) => RouteWidget( builder: (_) => Page1(), // You can use one of the predefined transitions transitionsBuilder: (context, animation, secondaryAnimation, child) { // Custom transition implementation return ...; }, ), '/page2': (_) => Page2(), }, );
Pages '/'
and '/page2'
will use the default transition whereas '/page1'
will use its own defined custom page transition.
You can customize the page transition for a particular navigation call:
dart myNavigator.to( '/page1', transitionsBuilder: (context, animation, secondaryAnimation, child) { // Custom transition implementation return ...; }, )
The defined transition here will be used once for this navigation call.Any further call of myNavigator.to
method will use the default page transition or the route transition in case it is used.
You are free to defined the transition you want using the parameters exposed by transitionBuilder
. For example:
final myNavigator = RM.injectNavigator(
transitionsBuilder: (context, animation, secondaryAnimation, child) {
return FadeTransition(
opacity: animation.drive(
CurveTween(curve: Curves.easeIn),
),
child: child,
);
},
routes: {
...
},
);
You can also use one of the predefined page transitions:
RM.transitions.bottomToUP();
RM.transitions.upToBottom();
RM.transitions.leftToRight();
RM.transitions.rightToLeft();
RM.transitions.none(); // Pages are transited instantly without animation
Other predefined page transitions can be added in the future.
Head Here to see different level of page transition animation example.
You can get the page animation and secondary animation form any page using context.animation
and context.secondaryAnimation
relying on InheritedWidget
principle.
You can use the obtained animations to customize the incoming page animation.
Here is an example using this concept inspired form ResoCoder's tutorial, Flutter custom staggered page transition animation.
Route redirection can be done in two levels, at route level and in global level.
If a route is redirected in both route and global level, the route level takes the priority over the global redirection.
final InjectedNavigator myNavigator = RM.injectNavigator(
// Define your routes map
routes: {
'/': (RouteData data) => data.redirectTo('/home'),
'/home': (RouteData data) => Home(),
// redirect all paths that starts with '/home' to '/home' path
'/home/*': (RouteData data) => data.redirectTo('/home'),
// redirect dynamic links
'/books/:bookId/authors': (RouteData data) => data.redirectTo('/authors'),
'/authors': (RouteData data) {
// As we are redirected here from '/books/:bookId/authors' we can get the book id.
final bookId = data.pathParam['bookId'];
// The location we are redirected from.
// For example `books/1/authors`
final redirectedFrom = data.redirectedFrom.location;
// Or inside Authors widget we use context.routeData.redirectedFrom.location
return Authors();
},
},
);
From the route you are redirected to, you can obtain information about the route you are redirected from via RouteData.redirectedFrom
field.
For example, from the route /books/:bookId/authors
we are redirected to the /authors
route. From the latter route we can get the bookId and display the author of the book.
Global redirections are done inside OnNavigate
callBack.
final myNavigator = RM.injectNavigator(
onNavigate: (RouteData data) {
final toLocation = data.location;
if (toLocation != '/signInPage' && userIsNotSigned) {
return data.redirectTo('/signInPage');
}
if (toLocation == '/signInPage' && userIsSigned) {
return data.redirectTo('/homePage');
}
//You can also check query or path parameters
if (data.queryParams['userId'] == '1') {
return data.redirectTo('/superUserPage');
}
},
routes: {
'/signInPage': (RouteData data) => SignInPage(),
'/homePage': (RouteData data) => HomePage(),
},
);
If the user is not signed in and if he is navigating to any page other than the sign in page, he will be redirected to the sign in page.
Form the sign in page, the user must sign in first to be able to continue.
From the /signInPage
page we can get the location the user is redirected from and let him navigate to it.
Here is what may the sign in method look like:
void signIn(String name, String password) async {
final success = await repository.signIn(name, password);
if(success){
// Here the user is successfully signed in.
final locationRedirectedFrom = context.routeData.redirectedFrom?.location;
if(locationRedirectedFrom != null){
// If the app is redirected form any location (deep link for example), it will continue navigate to it.
myNavigator.to(locationRedirectedFrom);
}else{
// If the app accesses the 'signInPage' directly without redirection, it will navigate to the home page for example.
myNavigator.to('homePage');
}
}
}
Here is The full example example
Do not fear cyclic redirection. If it happens the unknownRoute is pushed.
final navigator = RM.injectNavigator(
routes: {
'/': (data) => data.redirectTo('/home'),
'/home': (data) => const HomePage(),
// page1 redirect to itself
'/page1': (data) => data.redirectTo('/page1'),
// page2 redirect to page3 and page3 redirect back to page2
'/page2': (data) => data.redirectTo('/page3'),
'/page3': (data) => data.redirectTo('/page2'),
// /page4 route is redirect form onNavigate callback to page5
// and page5 redirect locally to page4
'/page4': (data) => const PageWidget(title: 'Never Reached Page'),
'/page5': (data) => data.redirectTo('/page4'),
},
onNavigate: (data) {
final location = data.location;
if (location == '/page4') {
return data.redirectTo('/page5');
}
},
);
Here is the working example