voyager 0.3.0 voyager: ^0.3.0 copied to clipboard
Router and basic dependency injection library for Flutter. Define navigation paths in YAML and power them up with custom plugins.
To boldly resolve where no Dart has resolved before.
Router, requirements & dependency injection library for Flutter.
Features #
If your app is a list of screens with respective paths then this library is for you.
- YAML/JSON based Navigation Spec
- support for query parameters
- support for global parameters
- path subsections
- parameters interpolation in subsections
- logicless
- deliverable over the air (think Firebase remote config)
- code generator for paths (COMING SOON)
- Highly customizable plugin architecture.
- VoyagerWidget to embed your
path
at any point - Provider to inject any data coming with the
path
Getting started #
To use this plugin, add voyager
as a dependency in your pubspec.yaml file.
You should ensure that you add the router as a dependency in your flutter project.
dependencies:
voyager: "^0.3.0"
You can also reference the git repo directly if you want:
dependencies:
voyager:
git: git://github.com/vishna/voyager.git
Then in the code, make sure you import voyager package:
import 'package:voyager/voyager.dart';
Navigation Spec #
It’s best to start with describing what paths your app will have and what subsections will they be made of.
---
'/home' :
type: 'home'
screen: HomeWidget
title: "This is Home"
'/other/:title' :
type: 'other'
screen: OtherWidget
title: "This is %{title}"
You can either put this in assets as a yaml file or use triple quotes '''
and keep it in your code as a string. The String approach while a bit uglier allows for faster reloads while updating assets requires project rebuild.
Creating Router Instance #
Your router requires paths and plugins as constructor parameters. Getting paths is quite straightforwad and basically means parsing that YAML file we just defined.
final paths = loadPathsFromString('''
---
'/home' :
type: 'home'
screen: HomeWidget
title: "This is Home"
'/other/:title' :
type: 'other'
screen: OtherWidget
title: "This is %{title}"
''');
or if the file is in the assets folder, you can:
final paths = loadPathsFromAssets("assets/navigation.yml");
NOTE: JSON support is available as of version 0.2.3
, please check voyager_test.dart for reference.
The other important ingredient of voyager router are plugins. You need to tell router what kind of plugins you plan to use and those depend on what you have written in the navigation file. In our example we use 3 type
, screen
and title
. This library comes with predefined plugins for type
& screen
and we'll create our own plugin for title
.
final plugins = [
[
TypePlugin(),
ScreenPlugin({ // provide widget builders for expressions used in YAML
"HomeWidget": (context) => HomeWidget(),
"OtherWidget": (context) => OtherWidget(),
}),
TitlePlugin()
]
];
Now you're all set for getting your router instance:
Future<RouterNG> router = loadRouter(paths, plugins)
Custom Plugins #
You can define as many plugins as you want. Here's how you could handle the title
nodes from the example navigation yaml.
class TitlePlugin extends RouterPlugin {
TitlePlugin() : super("title"); // YAML node to intercept
@override
void outputFor(RouterContext context, dynamic config, Voyager voyager) {
// config can be anything that is passed from YAML
voyager["title"] = config.toString(); // output of this plugin
}
}
NOTE: Above plugin is redundant, Voyager will repackage the primitive types from configuration and you don't need to do anything 😎 Use plugins to resolve primitive types to custom types , e.g. take a look at IconPlugin from the example app.
Router's Default Output: Voyager #
Voyager
instance is the composite output of all the relevant plugins that are nested under the path being resolved. Observe:
Voyager voyager = router.find("/home")
print(voyager["title"]); // originates from the title plugin, prints: "This is home"
print(voyager["type"]); // originates from the type plugin, prints: "home"
assert(voyager["screenBuilder"] is WidgetBuilder); // originates from the screen plugin
NOTE: Any attempt to modify voyager keys will fail unless done from plugin's outputFor
method. If you want to add some values to Voyager later on, use Voyager.storage
public map.
Embed any screen path with VoyagerWidget #
If your path uses screen
plugin you can try using VoyagerWidget
and embed any path you want like this:
VoyagerWidget(path: "/home", router: router);
NOTE: You can even omit passing router instance if this VoyagerWidget
is nested within other VoyagerWidget
.
Inject your information via Provider #
If you use VoyagerWidget
to create screens for your paths, you can obtain Voyager
anywhere from BuildContext
using Provider:
final voyager = Provider.of<Voyager>(context);
Now going back to our mystery OtherWidget
class from the example navigation spec, that widget could be implemented something like this:
class OtherWidget extends StatelessWidget {
@override
Widget build(BuildContext context) {
final voyager = Provider.of<Voyager>(context); // injecting voyager from build context
final title = voyager["title"]; // assuming title plugin worked and title is here 🙈
return Scaffold(
appBar: AppBar(
title: Text(title), // et voilà
),
body: Center(
child: Text("Other Page"),
)
);
}
}
Integrating with MaterialApp #
Defining inital path & handling navigation
final initalPath = "/my/fancy/super/path"
MaterialApp(
home: VoyagerWidget(path: initalPath, router: router),
onGenerateRoute: router.generator()
)
NOTE: You can use MaterialApp.initalRoute
but please read this first if you find MaterialApp.initalRoute
is not working for you. TL;DR: It's working as intended ¯\_(ツ)_/¯
Navigation #
Having BuildContext
and Material.onGenerateRoute
set up, you can simply:
Navigator.of(context).pushNamed("/path/to/go");
If you need to push new screen from elsewhere you probably should set navigatorKey to your MaterialApp
Custom Transitions #
The article "Create Custom Router Transition in Flutter using PageRouteBuilder" by Agung Surya explains in detail how to create custom reusable transtions.
Essentially you need to extend a PageRouteBuilder
class and pass it a widget you want to be transitioning to. In our case that widget is a VoyagerWidget
.
In the aforementioned artile, the author created SlideRightRoute
transition. We can combine that transition with any path from our navigation spec by using code below:
Navigator.push(
context,
SlideRightRoute(widget: VoyagerWidget.fromPath(context, "/path/to/go")),
);
Adding global values #
If you want to expose some global parameters to specs interpolation, you can do so by doing the following:
router.registerGlobalParam("isTablet", false);
NOTE: Because we interpolate String here, only primitve types are allowed.
If you want to make some global entities available via router instance, you can do so by doing the following:
router.registerGlobalEntity("database", someDatabase);
Sample App #
Check out full example here
Strong Typed Paths #
Typing navigation paths by hand is error prone, for this very reason there is there will be companion library that generates you a dart class from your navigation file so you should only need to type path once.
More Resources #
Acknowledgments #
- fluro As their repo says: "The brightest, hippest, coolest router for Flutter." Probably the most know flutter router out there.
- angel-route "A powerful, isomorphic routing library for Dart." Voyager internally was depending on this library till version
0.2.3
. It was a server oriented library and too big dependency for this project - voyager is now using abstract_router.dart which is < 300 LOC. - eyeem/router Protoplast of the voyager library, written in Java, for Android.
- NASA Voyager 2 Interstellar Poster Beautiful artwork I found on NASA page also a base content for the banner - changed colors to flutter ones, cropped the poster, added flutter antenna.