Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/dennis-krasnov/flutter-deep-link-navigation
Elegant abstraction for complete deep linking navigation in Flutter
https://github.com/dennis-krasnov/flutter-deep-link-navigation
dart deep-links flutter navigation
Last synced: 3 months ago
JSON representation
Elegant abstraction for complete deep linking navigation in Flutter
- Host: GitHub
- URL: https://github.com/dennis-krasnov/flutter-deep-link-navigation
- Owner: Dennis-Krasnov
- License: mit
- Created: 2019-10-25T01:13:22.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2020-02-11T22:37:10.000Z (about 5 years ago)
- Last Synced: 2024-11-16T03:17:39.731Z (3 months ago)
- Topics: dart, deep-links, flutter, navigation
- Language: Dart
- Homepage: https://pub.dev/packages/deep_link_navigation
- Size: 243 KB
- Stars: 68
- Watchers: 3
- Forks: 9
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
README
# Flutter Deep Link Navigation
[data:image/s3,"s3://crabby-images/ada80/ada805838f5158b938f60cd83fd5f41b795401a1" alt="pub.dev package"](https://pub.dev/packages/deep_link_navigation)
[data:image/s3,"s3://crabby-images/ab4f8/ab4f8dec8718f1b728f99c4a4fdf40524b5e33f8" alt="Github stars"](https://github.com/Dennis-Krasnov/Flutter-Deep-Link-Navigation)
[data:image/s3,"s3://crabby-images/5bad4/5bad42aeddca1441bb7e0f2b9f88fc72666283b0" alt="Open source license"](https://github.com/Dennis-Krasnov/Flutter-Deep-Link-Navigation/blob/master/LICENSE)
[data:image/s3,"s3://crabby-images/b571e/b571ee8c3168984b66fb7c78d771959e58b855a1" alt="Awesome Flutter"](https://github.com/Solido/awesome-flutter#navigation)Provides an elegant abstraction for complete deep linking navigation in Flutter.
This package **only provides deep linking for internal navigation**. Any external platform-level deep linking solution can optionally be used in conjuction with this package.
The target audience of the documentation is experienced Flutter developers.
## Motivation
There's nothing wrong with not using deep links for internal navigation.Partially implementing deep links would either have **limited benefits** or be **extremely complicated** (not to mention confusing).
Hence why if you decide to use deep links, it makes sense to exclusively use deep links.
**If you try to implement complete deep linking yourself here's some of the issues you'll run into:**
* Widgets become coupled with their place in the navigation hierarchy
- Becomes an issue if you want to reuse a page
- No amount of fancy iteration or recursion will help
* It's difficult to 'pass down' deep link values used in higher levels
- Given the hierarchy is `Artist` --> `Song`
- Being able to do `(artist) => ArtistPage(artist)` and later `(song) => SongPage(artist, song)`
- I actually published a [rest-like router package](https://pub.dev/packages/rest_router) while trying to solve this issue
* How do you represent deep links internally?
- How to keep configuration as terse as possible?
- How to represent current route in a string-friendly format (eg. for analytics, debugging)
- How to serialize route to be used with a platform-level deep linking solution
- How to handle native navigator pops (eg. back button)?
- How to handle a route that doesn't exist (or any other exception that occurs during dispatch)?
* How do you integrate custom logic for validating deep link navigation?
- Ideally provide context to be able to access state management
- eg. certain subtrees of the navigation hierarchy are available only for subscribed or authenticated users**TL;DR**
I separated the navigation system from [Diet Driven](https://github.com/Dennis-Krasnov/Diet-Driven) (shameless plug, [please hire me](https://denniskrasnov.com/)) into its own package and published it.
This package provides a solution for all the aforementioned difficulties.
## Examples
### [Single base route](https://github.com/Dennis-Krasnov/Flutter-Deep-Link-Navigation/tree/master/examples/single_base_route)
**This example demonstrates:**
* Dispatchers with path-only deep links
* Dispatchers with value deep links (ArtistDL, SongDL)
* Exception handling (RouteNotFoundDL)
* Cross-branch navigation (from favorite's song page to artist page)
data:image/s3,"s3://crabby-images/5bfeb/5bfebc8fab2e5fa225ed104a6c61dcec7e60a54c" alt="Navigation diagram for multiple base routes example"### [Multiple base routes](https://github.com/Dennis-Krasnov/Flutter-Deep-Link-Navigation/tree/master/examples/multiple_base_routes)
**This example demonstrates:**
* Everything from single base route example
* Bottom navigation (library, favorites, user pages) persists across navigation
* Login and error pages are full screen (hide bottom navigation)
* Using the asynchronous result of push (AuthenticationDL from UserDL)
* Custom `Authenticated` mixin ensures user is authenticated (LibraryDL, FavoritesDL, UserDL)
data:image/s3,"s3://crabby-images/a5e10/a5e105851553db5bb88d8ccb10426c637f1a46e3" alt="Navigation diagram for multiple base routes example"## Configuration
#### Deep links
`DeepLink` is base unit of routes, a deep link is mapped to a `Widget` by a `Dispatcher`.Deep links may be reused in different levels of the navigation hierarchy.
A route is the full location of a page, represented by `List`.
`path` is the string representation of the route aka `route.join("/")`.
**Path**
```dart
class LibraryDL extends DeepLink {
LibraryDL() : super("library");
}
```**Value**
Deep links can also store data, value dispatchers are strongly typed.
```dart
class SongDL extends ValueDeepLink {
SongDL(Song song) : super("song", song);
}class SongDL extends ValueDeepLink {
// Override toString
SongDL(Song song) : super("song", song, toString: (song) => song.id);
}
```**Mixin for sake of inheritence**
This could also be achieved by implementing an abstract class.
See use in *Child builder* section.
```dart
mixin FullScreen on DeepLink {}class LoginDL extends DeepLink with FullScreen {
LoginDL() : super("login");
}
```**Mixin with logic**
```dart
mixin Authenticated on DeepLink {
@override
void onDispatch(BuildContext context) {
// Get state from context or global/static variable
final isAuthenticated = Provider.of(context, listen: false).authenticated;// Throw custom exception
if (!isAuthenticated) {
throw Unauthenticated();
}
}
}// ...
navigation: (context) => Dispatcher()
// Unauthenticated login page
..exception((exception, route) => [LoginDL()])
..path((route) => LoginPage()),
```#### Application
Use `DeepLinkMaterialApp` instead of using Flutter's `MaterialApp`.This replaces native navigation options with their deep link counterparts.
At most one deep link of a type can exist on a dispatcher.
**Deep link material app**
```dart
DeepLinkMaterialApp(
navigation: (context) => Dispatcher() // see next section ...
defaultRoute: [LibraryDL()], // if ommited, the splash screen is shown until explicit navigation
splashScreen: SplashPage(),
childBuilder: // see child builder section ...// Non-navigation related fields are still available
themeMode: ThemeMode.light,
// ...
);
```**Path dispatcher**
```dart
..path((route) => LoginPage()),
```**Value dispatcher**
```dart
..value((song, route) => SongPage(song: song)),
```**Sub navigation**
```dart
..path(
(route) => LibraryPage(),
subNavigation: Dispatcher() // ...
),..value(
(song, route) => SongPage(song: song),
subNavigation: (song) => Dispatcher() // song may be used from this point onward
),
```**Exception mapping**
Exceptions that are thrown while running through the navigation hierarchy are mapped to routes.
`..exception` **MUST** be defined on the base-level dispatcher.
If multiple mappings of the same type are found thoughout the hierarchy, the deep-most mapping is used.
```dart
..exception((exception, route) => [ErrorDL(exception)])
```#### Child builder
The widget specified in `childBuilder` is rebuilt when the route in `deepLinkNavigator` changes.
```dart
/// [DeepLink]s associated with the bottom navigation.
final bottomNavigationDeepLinks = [LibraryDL(), FavoritesDL(), UserDL()];/// Current index of bottom navigation based on [currentRoute].
int currentIndex(List currentRoute) {
final index = bottomNavigationDeepLinks.indexOf(currentRoute?.first);
return index != -1 ? index : 0;
}/// ...
childBuilder: (BuildContext context, DeepLinkNavigator deepLinkNavigator, Widget child) => Scaffold(
body: child,
// Don't show bottom navigation while [currentRoute] is null, or any deep list is [FullScreen]
bottomNavigationBar: deepLinkNavigator.currentRoute?.any((dl) => dl is FullScreen) ?? true ? null : BottomNavigationBar(
currentIndex: currentIndex(deepLinkNavigator.currentRoute),
onTap: (int index) => deepLinkNavigator.navigateTo([bottomNavigationDeepLinks[index]]),
items: [
BottomNavigationBarItem(title: Text("Library"), icon: Icon(Icons.queue_music)),
BottomNavigationBarItem(title: Text("Favorites"), icon: Icon(Icons.favorite)),
BottomNavigationBarItem(title: Text("User"), icon: Icon(Icons.person)),
],
),
)
```#### In-app navigation
`DeepLinkNavigator` mirrors `Navigator`'s interface as much as possible (including push and pop futures).All methods internally orchestrate a native flutter navigator.
**Push a deep link**
```dart
await DeepLinkNavigator.of(context).push(ArtistDL(...));
```**Pop a value**
```dart
DeepLinkNavigator.of(context).pop(...);// or
Navigator.of(context).pop(...);
```**Navigate to specific route**
```dart
DeepLinkNavigator.of(context).navigateTo([
LibraryDL(),
ArtistDL(...),
]);
```**Return to default route (if any)**
```dart
DeepLinkNavigator.of(context).replaceWithDefault();
```**TODO: Throw exception to be caught by mapper**
```dart
// TODO DeepLinkNavigator.of(context).throw(Exception(...));
```**TODO: Access deep link navigator from anywhere**
```dart
// TODO DeepLinkNavigator()...
```**TODO: Page transitions**
```dart
// await DeepLinkNavigator.of(context).push(
// ArtistDL(...),
// transition: ...,
// duration: ...,
// );// Possibly:
// await DeepLinkNavigator.of(context).fadeIn(...);
```## Platform deep links
// TODO: serialize `List`## Limitations
* Must **FULLY** specify **ALL** generic types since this is how deep links are matched internally
* Please look very carefully when debugging
* Fails by throwing `RouteNotFound` if the route doesn't exist
* Fails by entering infinite recursion if `RouteNotFound` maps to a route that doesn't exist
* Can't currently define arbitrarily deep navigation hierarchies (think Spotify)
* Can't store separate persisted navigation states for a multi-base route application (think Instagram)## What's left to do
[ ] Custom/predefined page transitions
[ ] Access deep link navigator from anywhere using static method and factory pattern
[ ] Assert `RouteNotFound` dispatcher exists by running through navigation tree
[ ] Unit test deep link navigator logic
[ ] Cupertino and Widget apps
[ ] Explicit exception throwing
[ ] Platform deep links example + documentation
[ ] Route changed callback for analytics, etc