diff --git a/README.md b/README.md index 78e1c7cc..7ec501de 100644 --- a/README.md +++ b/README.md @@ -57,14 +57,13 @@ - [ActiveGuardObserver](#activeguardobserver) - [Examples](#examples) -**Note:** [AutoRoute-Helper] is no longer supported. ## Migration guides +- [Migrating to v9](https://github.com/Milad-Akarie/auto_route_library/blob/master/migrations/migrating_to_v9.md) +- [Migrating to v6](https://github.com/Milad-Akarie/auto_route_library/blob/master/migrations/migrating_to_v6.md) -- [Migrating to v6](#migrating-to-v6) - -## Pre v6 documentation - +## Old documentation +- [Pre v9 documentation](https://github.com/Milad-Akarie/auto_route_library/blob/master/old/pre_v9_README.md) - [Pre v6 documentation](https://github.com/Milad-Akarie/auto_route_library/blob/master/old/pre_v6_README.md) ## Introduction @@ -90,12 +89,12 @@ dev_dependencies: ## Setup And Usage -1. Create a router class and annotate it with `@AutoRouterConfig` then extend "$YourClassName" +1. Create a router class and annotate it with `@AutoRouterConfig` then extend "RootStackRouter" from The auto_route package 2. Override the routes getter and start adding your routes. ```dart @AutoRouterConfig() -class AppRouter extends $AppRouter { +class AppRouter extends RootStackRouter { @override List get routes => [ @@ -106,20 +105,9 @@ class AppRouter extends $AppRouter { ### Using part builder -To generate a part-of file simply add a `part` directive to your `AppRouter` and extend the generated private router. **Note:** The `deferredLoading` functionality does not work with part-file setup. - -```dart -part 'app_router.gr.dart'; +To generate a part-of file simply add a `part` directive to your `AppRouter`. -@AutoRouterConfig() -class AppRouter extends _$AppRouter { - - @override - List get routes => [ - /// routes go here - ]; -} -``` +**Note:** The `deferredLoading` functionality does not work with part-file setup. ### Generating Routable pages @@ -147,15 +135,23 @@ dart run build_runner build #### Add the generated route to your routes list ```dart -@AutoRouterConfig(replaceInRouteName: 'Screen,Route') -class AppRouter extends $AppRouter { +@AutoRouterConfig(replaceInRouteName: 'Screen|Page,Route') +class AppRouter extends RootStackRouter { + @override + RouteType get defaultRouteType => RouteType.material(); //.cupertino, .adaptive ..etc + @override List get routes => [ // HomeScreen is generated as HomeRoute because // of the replaceInRouteName property AutoRoute(page: HomeRoute.page), ]; + + @override + List get guards => [ + // optionally add root guards here + ]; } ``` @@ -187,10 +183,10 @@ A `PageRouteInfo` object will be generated for every declared **AutoRoute**. The class BookListRoute extends PageRouteInfo { const BookListRoute({ List? children, - }) : super(name, path: '/books', initialChildren: children); + }) : super(name, initialChildren: children); static const String name = 'BookListRoute'; - static const PageInfo page = PageInfo(name); + static const PageInfo page = PageInfo(name,builder: (...)); } ``` @@ -302,17 +298,7 @@ then inside of your `LoginPage`, pop with results ```dart router.maybePop(true); ``` - -as you'd notice we did not specify the result type, we're playing with dynamic values here, which can be risky and I personally don't recommend it. - -To avoid working with dynamic values, we specify what type of results we expect our page to return, which is a `bool` value. - -```dart -@RoutePage() -class LoginPage extends StatelessWidget {} -``` - -we push and specify the type of results we're expecting +Specifying the type of the result is optional, but it's recommended to avoid runtime errors. ```dart var result = await router.push(LoginRoute()); @@ -371,7 +357,7 @@ Defining nested routes is as easy as populating the children field of the parent ```dart @AutoRouterConfig(replaceInRouteName: 'Page,Route') -class AppRouter extends $AppRouter { +class AppRouter extends RootStackRouter { @override List get routes => [ @@ -407,14 +393,13 @@ class DashboardPage extends StatelessWidget { ), Expanded( // nested routes will be rendered here - child: AutoRouter(), + child: AutoRouter(), // this is important ), ], ); } } ``` - **Note** NavLink is just a button that calls router.push(destination). Now if we navigate to `/dashboard/users`, we will be taken to the `DashboardPage` and the `UsersPage` will be shown inside of it. What if want to show one of the child pages at `/dashboard`? We can simply do that by giving the child routes an empty path `''` to make initial or by setting initial to true. @@ -430,6 +415,36 @@ AutoRoute( ) ``` +#### Creating Empty Shell routes +Empty shell routes build a screen that contain the `AutoRouter` widget, which is used to render nested routes. +So you can build the widget your self like follows: +```dart +@RoutePage() +class MyShellPage extends StatelessWidget { + const MyShellPage({Key? key}) : super(key: key); + @override + Widget build(BuildContext context) { + /// you can wrap the AutoRouter with any widget you want + return AutoRouter(); + } +} +``` +You can shorten the code above a bit by directly extending the `AutoRouter` Widget. +```dart +@RoutePage() +class MyShellPage extends AutoRouter { + const MyShellPage({Key? key}) : super(key: key); +} +``` + +finally you can create a shell route without code generation using the `EmptyShellRoute` helper + + ```dart + final BooksTab = EmptyShellRoute('BooksTab'); + context.push(BooksTab()); + ``` + + or by using a `RedirectRoute` ```dart @@ -1053,26 +1068,31 @@ AutoRoute( #### Guarding all stack-routes -You can have all your stack-routes (non-tab-routes) go through a global guard by having your router implement an AutoRouteGuard. Lets say you have an app with no publish screens, we'd have a global guard that only allows navigation if the user is authenticated or if we're navigating to the LoginRoute. +You can have all your stack-routes (non-tab-routes) go through a list of global guards by overriding the guards property inside your router class. Lets say you have an app with no public screens, we'd have a global guard that only allows navigation if the user is authenticated or if we're navigating to the LoginRoute. ```dart @AutoRouterConfig() -class AppRouter extends $AppRouter implements AutoRouteGuard { +class AppRouter extends RootStackRouter{ @override - void onNavigation(NavigationResolver resolver, StackRouter router) { - if(isAuthenticated || resolver.route.name == LoginRoute.name) { - // we continue navigation - resolver.next(); - } else { - // else we navigate to the Login page so we get authenticated + late final List guards = [ + AutoRouteGuard.simple((resolver, router) { + if(isAuthenticated || resolver.routeName == LoginRoute.name) { + // we continue navigation + resolver.next(); + } else { + // else we navigate to the Login page so we get authenticated + + // tip: use resolver.redirect to have the redirected route + // automatically removed from the stack when the resolver is completed + resolver.redirect(LoginRoute(onResult: (didLogin) => resolver.next(didLogin))); + } + }, + ), + // add more guards here + ]; - // tip: use resolver.redirect to have the redirected route - // automatically removed from the stack when the resolver is completed - resolver.redirect(LoginRoute(onResult: (didLogin) => resolver.next(didLogin))); - } - } - // ..routes[] +// ..routes[] } ``` @@ -1341,11 +1361,12 @@ CustomRoute( You can use your own custom route by passing a `CustomRouteBuilder` function to `CustomRoute' and implement the builder function the same way we did with the TransitionsBuilder function, the most important part here is passing the page argument to our custom route. +make sure you pass the return type to your custom route builder function. ```dart CustomRoute( page: CustomPage, - customRouteBuilder: (BuildContext context, Widget child, CustomPage page) { - return PageRouteBuilder( + customRouteBuilder: (BuildContext context, Widget child, AutoRoutePage page) { + return PageRouteBuilder( fullscreenDialog: page.fullscreenDialog, // this is important settings: page, @@ -1359,31 +1380,29 @@ CustomRoute( ### Including Micro/External Packages -To include routes inside of a depended-on package, that package needs to generate an `AutoRouterModule` that will be later consumed by the root router. - -To have a package output an `AutoRouterModule` instead of a `RootStackRouter`, we need to use the `AutoRouterConfig.module()` annotation like follows +To include routes inside of a depended-on package, we generated the routes inside the micro package like normal, then either use the generated routes inside your main router individually, +or declare them inside your micro router and merge them with the main router. ```dart -@AutoRouterConfig.module() -class MyPackageModule extends $MyPackageModule {} -``` + final myMicroRouter = MyMicroRouter(); -Then when setting up our root router we need to tell it to include the generated module. - -```dart -@AutoRouterConfig(modules: [MyPackageModule]) -class AppRouter extends $AppRouter {} + @override + List get routes => [ + AutoRoute(page: HomeRoute.page, initial: true), + /// use micro routes individually + AutoRoute(page: RouteFromMicroPackage.page), + /// or merge all routes from micro router + ...myMicroRouter.routes, + ]; ``` -Now you can use `PageRouteInfos` generated inside `MyPackageModule`. - -`Tip:` You can add export `MyPackageModule` to `app_router.dart`, so you only import `app_router.dart` inside of your code. +`Tip:` You can add export `MyMicroRouter` to `app_router.dart`, so you only import `app_router.dart` inside of your code. ```dart // ...imports -export 'package:my_package/my_package_module.dart' -@AutoRouterConfig(modules: [MyPackageModule]) -class AppRouter extends $AppRouter {} +export 'package:my_package/my_micro_router.dart' +@AutoRouterConfig() +class AppRouter extends RootStackRouter {} ``` ## Configuring builders @@ -1519,98 +1538,6 @@ void initState(){ ``` -## Migrating to v6 - -In version 6.0 **AutoRoute** aims for less generated code for more flexibility and less generation time. - -#### 1. Instead of using `MaterialAutoRouter`, `CupertinoAutoRouter`, etc, we now only have one annotation for our router which is `@AutoRouterConfig()` and instead of passing our routes list to the annotation we now pass it to the overridable getter `routes` inside of the generated router class and for the default route type you can override `defaultRouteType` - -#### Before - -```dart -// @CupertinoAutoRouter -// @AdaptiveAutoRouter -// @CustomAutoRouter -@MaterialAutoRouter( - routes: [ - // routes go here - ], -) -class $AppRouter {} -``` - -#### After - - ```dart -@AutoRouterConfig() -class AppRouter extends $AppRouter { - - @override - RouteType get defaultRouteType => RouteType.material(); //.cupertino, .adaptive ..etc - - @override - List get routes => [ - // routes go here - ]; -} -``` - -#### 2. Passing page components as types is changed, now you'd annotate the target page with `@RoutePage()` annotation and pass the generated `result.page` to AutoRoute(): - -#### Before - -```dart -class ProductDetailsPage extends StatelessWidget {} -``` - -```dart -AutoRoute(page: ProductDetailsPage) // as Type -``` - -#### After - -```dart -@RoutePage() // Add this annotation to your routable pages -class ProductDetailsPage extends StatelessWidget {} -``` - -```dart -AutoRoute(page: ProductDetailsRoute.page) // ProductDetailsRoute is generated -``` - -#### 3. `EmptyRoutePage` no longer exists, instead you will now make your own empty pages by extending the `AutoRouter` widget - -#### Before - -```dart -AutoRoute(page: EmptyRoutePage, name: 'ProductsRouter') // as Type -``` - -#### After - -```dart -@RoutePage(name: 'ProductsRouter') -class ProductsRouterPage extends AutoRouter {} -``` - -```dart -AutoRoute(page: ProductsRouter.page) -``` - -#### 4. Passing route guards is also changed now, instead of passing guards as types you now pass instances. - -#### Before - -```dart -AutoRoute(page: ProfilePage, guards:[AuthGuard]) // as Type -``` - -#### After - -```dart -AutoRoute(page: ProfilePage, guards:[AuthGuard()]) // as Instance -``` - ## Examples coming soon diff --git a/art/auto_route_logo.png b/art/auto_route_logo.png new file mode 100644 index 00000000..905ef315 Binary files /dev/null and b/art/auto_route_logo.png differ diff --git a/auto_route/CHANGELOG.md b/auto_route/CHANGELOG.md index 64ed3914..a069ac63 100644 --- a/auto_route/CHANGELOG.md +++ b/auto_route/CHANGELOG.md @@ -1,3 +1,23 @@ +## 9.0.0 [Breaking Changes] + +- **BREAKING CHANGE**: No Router class will be generated anymore. Instead, you + extend `RootStackRouter` from the `auto_route` package. +- **BREAKING CHANGE**: Providing return types inside `@RoutePage()` is no longer needed. you + just provide the type as you push the page. +- **BREAKING CHANGE**: Providing a global route is now done by overriding the `guards` property + inside the router. implementing AutoRouteGuard is no longer supported. +- **BREAKING CHANGE**: `AutoRouterConfig.module` is removed as it's no longer needed. `PageRouteInfos` are now self-contained. +- + For more info read the complete migration guide + [Migrating to v9](https://github.com/Milad-Akarie/auto_route_library/blob/master/migrations/migrating_to_v9.md) + +- **FIX**: Fix Aliased types are not generated correctly. +- **FEAT**: You can now create empty shell routes like follows: + ```dart + final BooksTab = EmptyShellRoute('BooksTab'); + context.push(BooksTab()); + ``` + ## 8.3.0 - **FEAT**: add url#fragment support. diff --git a/auto_route/example/lib/mobile/main.dart b/auto_route/example/lib/mobile/main.dart index 7d8a1233..d8f0a849 100644 --- a/auto_route/example/lib/mobile/main.dart +++ b/auto_route/example/lib/mobile/main.dart @@ -15,7 +15,7 @@ class MyApp extends StatefulWidget { class _MyAppState extends State { final authService = AuthService(); - final _rootRouter = RootRouter(); + final _rootRouter = AppRouter(); @override Widget build(BuildContext context) { diff --git a/auto_route/example/lib/mobile/router/router.dart b/auto_route/example/lib/mobile/router/router.dart index 3ee49d0c..23e19069 100644 --- a/auto_route/example/lib/mobile/router/router.dart +++ b/auto_route/example/lib/mobile/router/router.dart @@ -4,13 +4,12 @@ import 'package:example/mobile/router/router.gr.dart'; import 'package:example/mobile/screens/profile/routes.dart'; @AutoRouterConfig(generateForDir: ['lib/mobile']) -class RootRouter extends $RootRouter { +class AppRouter extends RootStackRouter { @override final List routes = [ - AutoRoute(page: WelcomeRoute.page, initial: true), AutoRoute( page: HomeRoute.page, - path: '/home', + initial: true, children: [ AutoRoute( path: 'books', @@ -38,17 +37,10 @@ class RootRouter extends $RootRouter { ), ], ), - AutoRoute(page: LoginRoute.page, path: '/login'), + AutoRoute(page: BooksTab.page, path: '/login'), RedirectRoute(path: '*', redirectTo: '/'), ]; } -@RoutePage(name: 'BooksTab') -class BooksTabPage extends AutoRouter { - const BooksTabPage({super.key}); -} - -@RoutePage(name: 'ProfileTab') -class ProfileTabPage extends AutoRouter { - const ProfileTabPage({super.key}); -} +final BooksTab = EmptyShellRoute('BooksTab'); +final ProfileTab = EmptyShellRoute('ProfileTab'); diff --git a/auto_route/example/lib/mobile/router/router.gr.dart b/auto_route/example/lib/mobile/router/router.gr.dart index 4437ec73..18ab4f1c 100644 --- a/auto_route/example/lib/mobile/router/router.gr.dart +++ b/auto_route/example/lib/mobile/router/router.gr.dart @@ -8,131 +8,22 @@ // coverage:ignore-file // ignore_for_file: no_leading_underscores_for_library_prefixes -import 'package:auto_route/auto_route.dart' as _i9; -import 'package:example/mobile/router/router.dart' as _i3; +import 'package:auto_route/auto_route.dart' as _i8; import 'package:example/mobile/screens/books/book_details_page.dart' as _i1; import 'package:example/mobile/screens/books/book_list_page.dart' as _i2; -import 'package:example/mobile/screens/home_page.dart' as _i4; -import 'package:example/mobile/screens/login_page.dart' as _i5; -import 'package:example/mobile/screens/profile/my_books_page.dart' as _i6; -import 'package:example/mobile/screens/profile/profile_page.dart' as _i7; -import 'package:example/mobile/screens/settings_page.dart' as _i8; -import 'package:flutter/material.dart' as _i10; - -abstract class $RootRouter extends _i9.RootStackRouter { - $RootRouter({super.navigatorKey}); - - @override - final Map pagesMap = { - BookDetailsRoute.name: (routeData) { - final pathParams = routeData.inheritedPathParams; - final args = routeData.argsAs( - orElse: () => BookDetailsRouteArgs( - id: pathParams.getInt( - 'id', - -3, - ))); - return _i9.AutoRoutePage( - routeData: routeData, - child: _i1.BookDetailsPage(id: args.id), - ); - }, - BookListRoute.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: _i2.BookListScreen(), - ); - }, - BooksTab.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: const _i3.BooksTabPage(), - ); - }, - HomeRoute.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: const _i4.HomePage(), - ); - }, - LoginRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const LoginRouteArgs()); - return _i9.AutoRoutePage( - routeData: routeData, - child: _i5.LoginPage( - key: args.key, - onLoginResult: args.onLoginResult, - showBackButton: args.showBackButton, - ), - ); - }, - MyBooksRoute.name: (routeData) { - final queryParams = routeData.queryParams; - final args = routeData.argsAs( - orElse: () => MyBooksRouteArgs( - filter: queryParams.optString( - 'filter', - 'none', - ))); - return _i9.AutoRoutePage( - routeData: routeData, - child: _i6.MyBooksPage( - key: args.key, - filter: args.filter, - ), - ); - }, - ProfileRoute.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: _i7.ProfilePage(), - ); - }, - ProfileTab.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: const _i3.ProfileTabPage(), - ); - }, - SettingsTab.name: (routeData) { - final pathParams = routeData.inheritedPathParams; - final queryParams = routeData.queryParams; - final args = routeData.argsAs( - orElse: () => SettingsTabArgs( - tab: pathParams.getString( - 'tab', - 'none', - ), - query: queryParams.getString( - 'query', - 'none', - ), - )); - return _i9.AutoRoutePage( - routeData: routeData, - child: _i8.SettingsPage( - key: args.key, - tab: args.tab, - query: args.query, - ), - ); - }, - WelcomeRoute.name: (routeData) { - return _i9.AutoRoutePage( - routeData: routeData, - child: const _i4.WelcomeScreen(), - ); - }, - }; -} +import 'package:example/mobile/screens/home_page.dart' as _i3; +import 'package:example/mobile/screens/login_page.dart' as _i4; +import 'package:example/mobile/screens/profile/my_books_page.dart' as _i5; +import 'package:example/mobile/screens/profile/profile_page.dart' as _i6; +import 'package:example/mobile/screens/settings_page.dart' as _i7; +import 'package:flutter/material.dart' as _i9; /// generated route for /// [_i1.BookDetailsPage] -class BookDetailsRoute extends _i9.PageRouteInfo { +class BookDetailsRoute extends _i8.PageRouteInfo { BookDetailsRoute({ int id = -3, - List<_i9.PageRouteInfo>? children, + List<_i8.PageRouteInfo>? children, }) : super( BookDetailsRoute.name, args: BookDetailsRouteArgs(id: id), @@ -142,8 +33,19 @@ class BookDetailsRoute extends _i9.PageRouteInfo { static const String name = 'BookDetailsRoute'; - static const _i9.PageInfo page = - _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final args = data.argsAs( + orElse: () => BookDetailsRouteArgs( + id: pathParams.getInt( + 'id', + -3, + ))); + return _i1.BookDetailsPage(id: args.id); + }, + ); } class BookDetailsRouteArgs { @@ -159,8 +61,8 @@ class BookDetailsRouteArgs { /// generated route for /// [_i2.BookListScreen] -class BookListRoute extends _i9.PageRouteInfo { - const BookListRoute({List<_i9.PageRouteInfo>? children}) +class BookListRoute extends _i8.PageRouteInfo { + const BookListRoute({List<_i8.PageRouteInfo>? children}) : super( BookListRoute.name, initialChildren: children, @@ -168,27 +70,18 @@ class BookListRoute extends _i9.PageRouteInfo { static const String name = 'BookListRoute'; - static const _i9.PageInfo page = _i9.PageInfo(name); -} - -/// generated route for -/// [_i3.BooksTabPage] -class BooksTab extends _i9.PageRouteInfo { - const BooksTab({List<_i9.PageRouteInfo>? children}) - : super( - BooksTab.name, - initialChildren: children, - ); - - static const String name = 'BooksTab'; - - static const _i9.PageInfo page = _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + return _i2.BookListScreen(); + }, + ); } /// generated route for -/// [_i4.HomePage] -class HomeRoute extends _i9.PageRouteInfo { - const HomeRoute({List<_i9.PageRouteInfo>? children}) +/// [_i3.HomePage] +class HomeRoute extends _i8.PageRouteInfo { + const HomeRoute({List<_i8.PageRouteInfo>? children}) : super( HomeRoute.name, initialChildren: children, @@ -196,17 +89,22 @@ class HomeRoute extends _i9.PageRouteInfo { static const String name = 'HomeRoute'; - static const _i9.PageInfo page = _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + return const _i3.HomePage(); + }, + ); } /// generated route for -/// [_i5.LoginPage] -class LoginRoute extends _i9.PageRouteInfo { +/// [_i4.LoginPage] +class LoginRoute extends _i8.PageRouteInfo { LoginRoute({ - _i10.Key? key, + _i9.Key? key, void Function(bool)? onLoginResult, bool showBackButton = true, - List<_i9.PageRouteInfo>? children, + List<_i8.PageRouteInfo>? children, }) : super( LoginRoute.name, args: LoginRouteArgs( @@ -219,8 +117,18 @@ class LoginRoute extends _i9.PageRouteInfo { static const String name = 'LoginRoute'; - static const _i9.PageInfo page = - _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + final args = + data.argsAs(orElse: () => const LoginRouteArgs()); + return _i4.LoginPage( + key: args.key, + onLoginResult: args.onLoginResult, + showBackButton: args.showBackButton, + ); + }, + ); } class LoginRouteArgs { @@ -230,7 +138,7 @@ class LoginRouteArgs { this.showBackButton = true, }); - final _i10.Key? key; + final _i9.Key? key; final void Function(bool)? onLoginResult; @@ -243,12 +151,12 @@ class LoginRouteArgs { } /// generated route for -/// [_i6.MyBooksPage] -class MyBooksRoute extends _i9.PageRouteInfo { +/// [_i5.MyBooksPage] +class MyBooksRoute extends _i8.PageRouteInfo { MyBooksRoute({ - _i10.Key? key, + _i9.Key? key, String? filter = 'none', - List<_i9.PageRouteInfo>? children, + List<_i8.PageRouteInfo>? children, }) : super( MyBooksRoute.name, args: MyBooksRouteArgs( @@ -261,8 +169,22 @@ class MyBooksRoute extends _i9.PageRouteInfo { static const String name = 'MyBooksRoute'; - static const _i9.PageInfo page = - _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + final queryParams = data.queryParams; + final args = data.argsAs( + orElse: () => MyBooksRouteArgs( + filter: queryParams.optString( + 'filter', + 'none', + ))); + return _i5.MyBooksPage( + key: args.key, + filter: args.filter, + ); + }, + ); } class MyBooksRouteArgs { @@ -271,7 +193,7 @@ class MyBooksRouteArgs { this.filter = 'none', }); - final _i10.Key? key; + final _i9.Key? key; final String? filter; @@ -282,9 +204,9 @@ class MyBooksRouteArgs { } /// generated route for -/// [_i7.ProfilePage] -class ProfileRoute extends _i9.PageRouteInfo { - const ProfileRoute({List<_i9.PageRouteInfo>? children}) +/// [_i6.ProfilePage] +class ProfileRoute extends _i8.PageRouteInfo { + const ProfileRoute({List<_i8.PageRouteInfo>? children}) : super( ProfileRoute.name, initialChildren: children, @@ -292,31 +214,22 @@ class ProfileRoute extends _i9.PageRouteInfo { static const String name = 'ProfileRoute'; - static const _i9.PageInfo page = _i9.PageInfo(name); -} - -/// generated route for -/// [_i3.ProfileTabPage] -class ProfileTab extends _i9.PageRouteInfo { - const ProfileTab({List<_i9.PageRouteInfo>? children}) - : super( - ProfileTab.name, - initialChildren: children, - ); - - static const String name = 'ProfileTab'; - - static const _i9.PageInfo page = _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + return _i6.ProfilePage(); + }, + ); } /// generated route for -/// [_i8.SettingsPage] -class SettingsTab extends _i9.PageRouteInfo { +/// [_i7.SettingsPage] +class SettingsTab extends _i8.PageRouteInfo { SettingsTab({ - _i10.Key? key, + _i9.Key? key, String tab = 'none', String query = 'none', - List<_i9.PageRouteInfo>? children, + List<_i8.PageRouteInfo>? children, }) : super( SettingsTab.name, args: SettingsTabArgs( @@ -331,8 +244,29 @@ class SettingsTab extends _i9.PageRouteInfo { static const String name = 'SettingsTab'; - static const _i9.PageInfo page = - _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final queryParams = data.queryParams; + final args = data.argsAs( + orElse: () => SettingsTabArgs( + tab: pathParams.getString( + 'tab', + 'none', + ), + query: queryParams.getString( + 'query', + 'none', + ), + )); + return _i7.SettingsPage( + key: args.key, + tab: args.tab, + query: args.query, + ); + }, + ); } class SettingsTabArgs { @@ -342,7 +276,7 @@ class SettingsTabArgs { this.query = 'none', }); - final _i10.Key? key; + final _i9.Key? key; final String tab; @@ -355,9 +289,9 @@ class SettingsTabArgs { } /// generated route for -/// [_i4.WelcomeScreen] -class WelcomeRoute extends _i9.PageRouteInfo { - const WelcomeRoute({List<_i9.PageRouteInfo>? children}) +/// [_i3.WelcomeScreen] +class WelcomeRoute extends _i8.PageRouteInfo { + const WelcomeRoute({List<_i8.PageRouteInfo>? children}) : super( WelcomeRoute.name, initialChildren: children, @@ -365,5 +299,10 @@ class WelcomeRoute extends _i9.PageRouteInfo { static const String name = 'WelcomeRoute'; - static const _i9.PageInfo page = _i9.PageInfo(name); + static _i8.PageInfo page = _i8.PageInfo( + name, + builder: (data) { + return const _i3.WelcomeScreen(); + }, + ); } diff --git a/auto_route/example/lib/mobile/screens/home_page.dart b/auto_route/example/lib/mobile/screens/home_page.dart index 65f92144..67e41561 100644 --- a/auto_route/example/lib/mobile/screens/home_page.dart +++ b/auto_route/example/lib/mobile/screens/home_page.dart @@ -1,4 +1,5 @@ import 'package:auto_route/auto_route.dart'; +import 'package:example/mobile/router/router.dart'; import 'package:example/mobile/router/router.gr.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; @@ -65,19 +66,20 @@ class HomePageState extends State with TickerProviderStateMixin { (d) => context.router.isRouteActive(d.route.routeName), ); // there might be no active route until router is mounted - // so we play safe + // so we play it safe if (activeIndex == -1) { activeIndex = 0; } return Row( children: [ NavigationRail( - destinations: destinations - .map((item) => NavigationRailDestination( - icon: Icon(item.icon), - label: Text(item.label), - )) - .toList(), + destinations: [ + for (final d in destinations) + NavigationRailDestination( + icon: Icon(d.icon), + label: Text(d.label), + ) + ], selectedIndex: activeIndex, onDestinationSelected: (index) { // use navigate instead of push so you won't have diff --git a/auto_route/example/lib/mobile/screens/profile/my_books_page.dart b/auto_route/example/lib/mobile/screens/profile/my_books_page.dart index 34bcfa72..7a0c91d6 100644 --- a/auto_route/example/lib/mobile/screens/profile/my_books_page.dart +++ b/auto_route/example/lib/mobile/screens/profile/my_books_page.dart @@ -3,7 +3,7 @@ import 'package:example/mobile/router/router.gr.dart'; import 'package:flutter/material.dart'; //ignore_for_file: public_member_api_docs -@RoutePage() +@RoutePage() class MyBooksPage extends StatelessWidget { final String? filter; diff --git a/auto_route/example/lib/mobile/screens/profile/routes.dart b/auto_route/example/lib/mobile/screens/profile/routes.dart index 7935ab1c..8ab7ff66 100644 --- a/auto_route/example/lib/mobile/screens/profile/routes.dart +++ b/auto_route/example/lib/mobile/screens/profile/routes.dart @@ -1,4 +1,5 @@ import 'package:auto_route/auto_route.dart'; +import 'package:example/mobile/router/router.dart'; import 'package:example/mobile/router/router.gr.dart'; //ignore_for_file: public_member_api_docs diff --git a/auto_route/example/lib/mobile/screens/settings_page.dart b/auto_route/example/lib/mobile/screens/settings_page.dart index 6f16043f..05931231 100644 --- a/auto_route/example/lib/mobile/screens/settings_page.dart +++ b/auto_route/example/lib/mobile/screens/settings_page.dart @@ -1,4 +1,5 @@ import 'package:auto_route/auto_route.dart'; +import 'package:example/mobile/router/router.dart'; import 'package:example/mobile/router/router.gr.dart'; import 'package:flutter/material.dart'; @@ -46,7 +47,7 @@ class _SettingsPageState extends State ElevatedButton( onPressed: () { context.navigateTo(BooksTab( - children: [BookDetailsRoute(id: 2)], + children: [BookDetailsRoute(id: 1)], )); }, child: Text('Navigate to book details/1')) diff --git a/auto_route/example/lib/web_demo/router/web_router.dart b/auto_route/example/lib/web_demo/router/web_router.dart index 8ef17461..651c130a 100644 --- a/auto_route/example/lib/web_demo/router/web_router.dart +++ b/auto_route/example/lib/web_demo/router/web_router.dart @@ -6,24 +6,28 @@ import 'package:flutter/material.dart'; //ignore_for_file: public_member_api_docs @AutoRouterConfig(generateForDir: ['lib/web_demo']) -class WebAppRouter extends $WebAppRouter implements AutoRouteGuard { +class WebAppRouter extends RootStackRouter { AuthService authService; WebAppRouter(this.authService); @override - void onNavigation(NavigationResolver resolver, StackRouter router) async { - if (authService.isAuthenticated || - resolver.routeName == WebLoginRoute.name) { - resolver.next(); - } else { - resolver.redirect( - WebLoginRoute(onResult: (didLogin) { - resolver.resolveNext(didLogin, reevaluateNext: false); - }), - ); - } - } + late final List guards = [ + AutoRouteGuard.simple( + (resolver, scope) { + if (authService.isAuthenticated || + resolver.routeName == WebLoginRoute.name) { + resolver.next(); + } else { + resolver.redirect( + WebLoginRoute(onResult: (didLogin) { + resolver.resolveNext(didLogin, reevaluateNext: false); + }), + ); + } + }, + ) + ]; @override List get routes => [ @@ -95,14 +99,16 @@ class _MainWebPageState extends State { padding: const EdgeInsets.symmetric(vertical: 16), child: ElevatedButton( onPressed: widget.navigate ?? - () { - context.pushRoute( - UserRoute( - id: 2, - query: const ['value1', 'value2'], - fragment: 'frag', - ), - ); + () async { + final x = await MainWebRoute().push(context); + print(x); + // context.pushRoute( + // UserRoute( + // id: 2, + // query: const ['value1', 'value2'], + // fragment: 'frag', + // ), + // ); }, child: Text('Navigate to user/2'), ), @@ -111,7 +117,8 @@ class _MainWebPageState extends State { padding: const EdgeInsets.symmetric(vertical: 16), child: ElevatedButton( onPressed: () { - App.of(context).authService.isAuthenticated = false; + context.maybePop('String'); + // App.of(context).authService.isAuthenticated = false; }, child: Text('Logout'), ), @@ -281,6 +288,7 @@ class UserPage extends StatefulWidget { final int id; final List? query; final String? fragment; + UserPage({ Key? key, @PathParam('userID') this.id = -1, diff --git a/auto_route/example/lib/web_demo/router/web_router.gr.dart b/auto_route/example/lib/web_demo/router/web_router.gr.dart index 18a3fe11..0980aa32 100644 --- a/auto_route/example/lib/web_demo/router/web_router.gr.dart +++ b/auto_route/example/lib/web_demo/router/web_router.gr.dart @@ -15,131 +15,13 @@ import 'package:example/web_demo/router/web_verify_page.dart' as _i3; import 'package:flutter/foundation.dart' as _i5; import 'package:flutter/material.dart' as _i6; -abstract class $WebAppRouter extends _i4.RootStackRouter { - $WebAppRouter({super.navigatorKey}); - - @override - final Map pagesMap = { - MainWebRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const MainWebRouteArgs()); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.MainWebPage( - key: args.key, - navigate: args.navigate, - showUserPosts: args.showUserPosts, - ), - ); - }, - NotFoundRoute.name: (routeData) { - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.NotFoundScreen(), - ); - }, - UserAllPostsRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const UserAllPostsRouteArgs()); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.UserAllPostsPage( - key: args.key, - navigate: args.navigate, - ), - ); - }, - UserFavoritePostsRoute.name: (routeData) { - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.UserFavoritePostsPage(), - ); - }, - UserRoute.name: (routeData) { - final pathParams = routeData.inheritedPathParams; - final queryParams = routeData.queryParams; - final args = routeData.argsAs( - orElse: () => UserRouteArgs( - id: pathParams.getInt( - 'userID', - -1, - ), - query: queryParams.optList('query'), - fragment: routeData.fragment, - )); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.UserPage( - key: args.key, - id: args.id, - query: args.query, - fragment: args.fragment, - ), - ); - }, - UserPostsRoute.name: (routeData) { - final pathParams = routeData.inheritedPathParams; - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.UserPostsPage(id: pathParams.getInt('userID')), - ); - }, - UserProfileRoute.name: (routeData) { - final pathParams = routeData.inheritedPathParams; - final queryParams = routeData.queryParams; - final args = routeData.argsAs( - orElse: () => UserProfileRouteArgs( - userId: pathParams.getInt( - 'userID', - -1, - ), - likes: queryParams.getInt( - 'likes', - 0, - ), - )); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i1.UserProfilePage( - key: args.key, - navigate: args.navigate, - userId: args.userId, - likes: args.likes, - ), - ); - }, - WebLoginRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const WebLoginRouteArgs()); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i2.WebLoginPage( - key: args.key, - onResult: args.onResult, - ), - ); - }, - WebVerifyRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const WebVerifyRouteArgs()); - return _i4.AutoRoutePage( - routeData: routeData, - child: _i3.WebVerifyPage( - key: args.key, - onResult: args.onResult, - ), - ); - }, - }; -} - /// generated route for /// [_i1.MainWebPage] class MainWebRoute extends _i4.PageRouteInfo { MainWebRoute({ _i5.Key? key, - void Function()? navigate, - void Function()? showUserPosts, + _i5.VoidCallback? navigate, + _i5.VoidCallback? showUserPosts, List<_i4.PageRouteInfo>? children, }) : super( MainWebRoute.name, @@ -153,8 +35,18 @@ class MainWebRoute extends _i4.PageRouteInfo { static const String name = 'MainWebRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final args = + data.argsAs(orElse: () => const MainWebRouteArgs()); + return _i1.MainWebPage( + key: args.key, + navigate: args.navigate, + showUserPosts: args.showUserPosts, + ); + }, + ); } class MainWebRouteArgs { @@ -166,9 +58,9 @@ class MainWebRouteArgs { final _i5.Key? key; - final void Function()? navigate; + final _i5.VoidCallback? navigate; - final void Function()? showUserPosts; + final _i5.VoidCallback? showUserPosts; @override String toString() { @@ -187,7 +79,12 @@ class NotFoundRoute extends _i4.PageRouteInfo { static const String name = 'NotFoundRoute'; - static const _i4.PageInfo page = _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + return _i1.NotFoundScreen(); + }, + ); } /// generated route for @@ -195,7 +92,7 @@ class NotFoundRoute extends _i4.PageRouteInfo { class UserAllPostsRoute extends _i4.PageRouteInfo { UserAllPostsRoute({ _i5.Key? key, - void Function()? navigate, + _i5.VoidCallback? navigate, List<_i4.PageRouteInfo>? children, }) : super( UserAllPostsRoute.name, @@ -208,8 +105,17 @@ class UserAllPostsRoute extends _i4.PageRouteInfo { static const String name = 'UserAllPostsRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final args = data.argsAs( + orElse: () => const UserAllPostsRouteArgs()); + return _i1.UserAllPostsPage( + key: args.key, + navigate: args.navigate, + ); + }, + ); } class UserAllPostsRouteArgs { @@ -220,7 +126,7 @@ class UserAllPostsRouteArgs { final _i5.Key? key; - final void Function()? navigate; + final _i5.VoidCallback? navigate; @override String toString() { @@ -239,7 +145,12 @@ class UserFavoritePostsRoute extends _i4.PageRouteInfo { static const String name = 'UserFavoritePostsRoute'; - static const _i4.PageInfo page = _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + return _i1.UserFavoritePostsPage(); + }, + ); } /// generated route for @@ -267,8 +178,28 @@ class UserRoute extends _i4.PageRouteInfo { static const String name = 'UserRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final queryParams = data.queryParams; + final args = data.argsAs( + orElse: () => UserRouteArgs( + id: pathParams.getInt( + 'userID', + -1, + ), + query: queryParams.optList('query'), + fragment: data.fragment, + )); + return _i1.UserPage( + key: args.key, + id: args.id, + query: args.query, + fragment: args.fragment, + ); + }, + ); } class UserRouteArgs { @@ -304,7 +235,13 @@ class UserPostsRoute extends _i4.PageRouteInfo { static const String name = 'UserPostsRoute'; - static const _i4.PageInfo page = _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + return _i1.UserPostsPage(id: pathParams.getInt('userID')); + }, + ); } /// generated route for @@ -312,7 +249,7 @@ class UserPostsRoute extends _i4.PageRouteInfo { class UserProfileRoute extends _i4.PageRouteInfo { UserProfileRoute({ _i5.Key? key, - void Function()? navigate, + _i5.VoidCallback? navigate, int userId = -1, int likes = 0, List<_i4.PageRouteInfo>? children, @@ -331,8 +268,30 @@ class UserProfileRoute extends _i4.PageRouteInfo { static const String name = 'UserProfileRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final queryParams = data.queryParams; + final args = data.argsAs( + orElse: () => UserProfileRouteArgs( + userId: pathParams.getInt( + 'userID', + -1, + ), + likes: queryParams.getInt( + 'likes', + 0, + ), + )); + return _i1.UserProfilePage( + key: args.key, + navigate: args.navigate, + userId: args.userId, + likes: args.likes, + ); + }, + ); } class UserProfileRouteArgs { @@ -345,7 +304,7 @@ class UserProfileRouteArgs { final _i5.Key? key; - final void Function()? navigate; + final _i5.VoidCallback? navigate; final int userId; @@ -362,7 +321,7 @@ class UserProfileRouteArgs { class WebLoginRoute extends _i4.PageRouteInfo { WebLoginRoute({ _i6.Key? key, - void Function(bool)? onResult, + _i6.ValueChanged? onResult, List<_i4.PageRouteInfo>? children, }) : super( WebLoginRoute.name, @@ -375,8 +334,17 @@ class WebLoginRoute extends _i4.PageRouteInfo { static const String name = 'WebLoginRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final args = data.argsAs( + orElse: () => const WebLoginRouteArgs()); + return _i2.WebLoginPage( + key: args.key, + onResult: args.onResult, + ); + }, + ); } class WebLoginRouteArgs { @@ -387,7 +355,7 @@ class WebLoginRouteArgs { final _i6.Key? key; - final void Function(bool)? onResult; + final _i6.ValueChanged? onResult; @override String toString() { @@ -400,7 +368,7 @@ class WebLoginRouteArgs { class WebVerifyRoute extends _i4.PageRouteInfo { WebVerifyRoute({ _i6.Key? key, - void Function(bool)? onResult, + _i6.ValueChanged? onResult, List<_i4.PageRouteInfo>? children, }) : super( WebVerifyRoute.name, @@ -413,8 +381,17 @@ class WebVerifyRoute extends _i4.PageRouteInfo { static const String name = 'WebVerifyRoute'; - static const _i4.PageInfo page = - _i4.PageInfo(name); + static _i4.PageInfo page = _i4.PageInfo( + name, + builder: (data) { + final args = data.argsAs( + orElse: () => const WebVerifyRouteArgs()); + return _i3.WebVerifyPage( + key: args.key, + onResult: args.onResult, + ); + }, + ); } class WebVerifyRouteArgs { @@ -425,7 +402,7 @@ class WebVerifyRouteArgs { final _i6.Key? key; - final void Function(bool)? onResult; + final _i6.ValueChanged? onResult; @override String toString() { diff --git a/auto_route/example/lib/web_demo/web_main.dart b/auto_route/example/lib/web_demo/web_main.dart index ac361abe..61996f22 100644 --- a/auto_route/example/lib/web_demo/web_main.dart +++ b/auto_route/example/lib/web_demo/web_main.dart @@ -1,4 +1,3 @@ -import 'package:auto_route/auto_route.dart'; import 'package:example/web_demo/router/web_router.dart'; import 'package:flutter/material.dart'; @@ -22,16 +21,13 @@ class AppState extends State { setState(() {}); }); - late final _appRouter = WebAppRouter(authService); - - List? urlRoutes; - + late final _router = WebAppRouter(authService); @override Widget build(BuildContext context) { return MaterialApp.router( theme: ThemeData.dark(), debugShowCheckedModeBanner: false, - routerConfig: _appRouter.config( + routerConfig: _router.config( reevaluateListenable: authService, ), ); diff --git a/auto_route/lib/auto_route.dart b/auto_route/lib/auto_route.dart index 898f1500..0a9c0aee 100644 --- a/auto_route/lib/auto_route.dart +++ b/auto_route/lib/auto_route.dart @@ -1,6 +1,5 @@ library auto_route; -export 'src/auto_router_module.dart'; export 'src/common/common.dart'; export 'src/matcher/route_match.dart'; export 'src/navigation_failure.dart'; diff --git a/auto_route/lib/src/auto_router_module.dart b/auto_route/lib/src/auto_router_module.dart deleted file mode 100644 index e08ba730..00000000 --- a/auto_route/lib/src/auto_router_module.dart +++ /dev/null @@ -1,8 +0,0 @@ -import 'package:auto_route/auto_route.dart'; - -/// Module initializers will implement this class -/// to be later used in the root router. -abstract class AutoRouterModule { - /// The map holding the page names and their factories. - Map get pagesMap; -} diff --git a/auto_route/lib/src/common/auto_route_annotations.dart b/auto_route/lib/src/common/auto_route_annotations.dart index 09d281b0..53fe7d56 100644 --- a/auto_route/lib/src/common/auto_route_annotations.dart +++ b/auto_route/lib/src/common/auto_route_annotations.dart @@ -30,41 +30,21 @@ class AutoRouterConfig { /// defaults = const ['lib'] final List generateForDir; - /// Indicates whether the package using the annotation - /// is a module, in that case a different - /// output will be generated. - // ignore: unused_field - final bool _isModule; - - /// A List of modules to be added to the RootRouter. - final List? modules; - /// default constructor const AutoRouterConfig({ this.replaceInRouteName = 'Page|Screen,Route', this.deferredLoading = false, this.generateForDir = const ['lib'], - this.modules, - }) : _isModule = false; - - /// default constructor - const AutoRouterConfig.module({ - this.replaceInRouteName = 'Page|Screen,Route', - this.deferredLoading = false, - this.generateForDir = const ['lib'], - }) : _isModule = true, - modules = null; + }); } /// This annotation is used to mark flutter widgets as routable pages /// by enabling the router to construct them. /// -/// [T] is the results type returned -/// from this page route e.g MaterialPageRoute() /// defaults to dynamic @optionalTypeArgs @Target({TargetKind.classType}) -class RoutePage { +class RoutePage { /// The name of the generated route /// if not provided, a name will be generated from class name /// and maybe altered by [replaceInRouteName] diff --git a/auto_route/lib/src/matcher/route_match.dart b/auto_route/lib/src/matcher/route_match.dart index 1fac8ec3..c8c1923d 100644 --- a/auto_route/lib/src/matcher/route_match.dart +++ b/auto_route/lib/src/matcher/route_match.dart @@ -87,6 +87,9 @@ class RouteMatch { /// Helper to access [AutoRoute.title] TitleBuilder? get titleBuilder => _config.title; + /// Helper to access [AutoRoute.buildPage] + AutoRoutePage buildPage(RouteData data) => _config.buildPage(data); + /// Default constructor const RouteMatch({ required AutoRoute config, diff --git a/auto_route/lib/src/route/auto_route_config.dart b/auto_route/lib/src/route/auto_route_config.dart index c3c405a6..52179d1e 100644 --- a/auto_route/lib/src/route/auto_route_config.dart +++ b/auto_route/lib/src/route/auto_route_config.dart @@ -4,6 +4,9 @@ import 'package:collection/collection.dart'; import 'package:flutter/cupertino.dart'; import 'package:meta/meta.dart'; +/// A Signature for a function that passes the [RouteData] and returns a [PageRoute] +typedef AutoRoutePageBuilder = Widget Function(RouteData data); + /// Signature for a function that builds the route title /// Used in [AutoRoutePage] typedef TitleBuilder = String Function(BuildContext context, RouteData data); @@ -17,7 +20,7 @@ typedef RestorationIdBuilder = String Function(RouteMatch match); @immutable class AutoRoute { /// The name of page this route should map to - final String name; + final PageInfo page; final String? _path; /// Weather to match this route's path as fullMatch @@ -75,9 +78,25 @@ class AutoRoute { /// a RedirectRoute() to that path final bool initial; + /// Helper getter to get the name of the page + String get name => page.name; + + final AutoRoutePageBuilder _pageBuilder; + + /// Returns the Widget builder function of the page + AutoRoutePageBuilder get builder => _pageBuilder; + + /// Builds a [AutoRoutePage] from [RouteData] + AutoRoutePage buildPage(RouteData data) { + return AutoRoutePage( + child: builder(data), + routeData: data, + ); + } + AutoRoute._({ - required this.name, String? path, + required this.page, this.usesPathAsKey = false, this.guards = const [], this.fullMatch = false, @@ -92,12 +111,13 @@ class AutoRoute { this.initial = false, List? children, }) : _path = path, + _pageBuilder = page.builder, _children = children != null && children.isNotEmpty ? RouteCollection.fromList(children) : null; - const AutoRoute._change({ - required this.name, + AutoRoute._change({ + required this.page, required String path, required this.usesPathAsKey, required this.guards, @@ -113,6 +133,7 @@ class AutoRoute { required this.initial, required this.allowSnapshotting, }) : _path = path, + _pageBuilder = page.builder, _children = children; /// Builds a default AutoRoute instance with any [type] @@ -134,7 +155,7 @@ class AutoRoute { bool allowSnapshotting = true, }) { return AutoRoute._( - name: page.name, + page: page, path: path, fullMatch: fullMatch, maintainState: maintainState, @@ -172,7 +193,7 @@ class AutoRoute { bool allowSnapshotting = true, }) { return AutoRoute._( - name: page.name, + page: page, path: path, fullMatch: fullMatch, maintainState: maintainState, @@ -204,7 +225,7 @@ class AutoRoute { @override String toString() { - return 'RouteConfig{name: $name}'; + return 'AutoRoute{name: ${page.name}}'; } /// A simplified copyWith @@ -217,7 +238,7 @@ class AutoRoute { /// Returns a new AutoRoute instance with the provided details overriding. AutoRoute copyWith({ RouteType? type, - String? name, + PageInfo? page, String? path, bool? usesPathAsKey, List? guards, @@ -234,7 +255,7 @@ class AutoRoute { }) { return AutoRoute._change( type: type ?? this.type, - name: name ?? this.name, + page: page ?? this.page, path: path ?? this.path, usesPathAsKey: usesPathAsKey ?? this.usesPathAsKey, guards: guards ?? List.from(this.guards), @@ -268,10 +289,11 @@ class RedirectRoute extends AutoRoute { /// Default constructor RedirectRoute({ - required super.path, + required String path, required this.redirectTo, }) : super._( - name: 'Redirect#$path', + path: path, + page: PageInfo.redirect, fullMatch: true, ); } @@ -281,7 +303,7 @@ class RedirectRoute extends AutoRoute { class MaterialRoute extends AutoRoute { /// default constructor MaterialRoute({ - required PageInfo page, + required super.page, super.path, super.fullscreenDialog, super.maintainState, @@ -296,17 +318,16 @@ class MaterialRoute extends AutoRoute { super.initial, super.allowSnapshotting = true, }) : super._( - name: page.name, type: const RouteType.material(), ); } /// Builds an [AutoRoute] instance with [RouteType.cupertino] type @immutable -class CupertinoRoute extends AutoRoute { +class CupertinoRoute extends AutoRoute { /// Default constructor CupertinoRoute({ - required PageInfo page, + required super.page, super.fullscreenDialog, super.maintainState, super.fullMatch = false, @@ -320,15 +341,15 @@ class CupertinoRoute extends AutoRoute { super.keepHistory, super.initial, super.allowSnapshotting = true, - }) : super._(name: page.name, type: const RouteType.cupertino()); + }) : super._(type: const RouteType.cupertino()); } /// Builds an [AutoRoute] instance with [RouteType.adaptive] type @immutable -class AdaptiveRoute extends AutoRoute { +class AdaptiveRoute extends AutoRoute { /// Default constructor AdaptiveRoute({ - required PageInfo page, + required super.page, super.fullscreenDialog, super.maintainState, super.fullMatch = false, @@ -343,18 +364,15 @@ class AdaptiveRoute extends AutoRoute { bool opaque = true, super.keepHistory, super.allowSnapshotting = true, - }) : super._( - name: page.name, - type: RouteType.adaptive(opaque: opaque), - ); + }) : super._(type: RouteType.adaptive(opaque: opaque)); } /// Builds an [AutoRoute] instance with [RouteType.custom] type @immutable -class CustomRoute extends AutoRoute { +class CustomRoute extends AutoRoute { /// Default constructor CustomRoute({ - required PageInfo page, + required super.page, super.fullscreenDialog, super.maintainState, super.fullMatch = false, @@ -377,7 +395,6 @@ class CustomRoute extends AutoRoute { super.restorationId, Color? barrierColor, }) : super._( - name: page.name, type: RouteType.custom( transitionsBuilder: transitionsBuilder, customRouteBuilder: customRouteBuilder, @@ -402,7 +419,7 @@ class TestRoute extends AutoRoute { super.fullMatch, super.restorationId, super.initial, - }) : super._(name: name); + }) : super._(page: PageInfo(name, builder: (_) => const SizedBox())); } /// Builds a simplified [AutoRoute] instance for internal usage @@ -410,13 +427,12 @@ class TestRoute extends AutoRoute { @internal class DummyRootRoute extends AutoRoute { /// Default constructor - DummyRootRoute( - String name, { + DummyRootRoute({ required String path, super.children, super.fullMatch, super.restorationId, - }) : super._(name: name, path: path); + }) : super._(page: PageInfo.root, path: path); } /// Holds a single set of config-entries diff --git a/auto_route/lib/src/route/page_info.dart b/auto_route/lib/src/route/page_info.dart index 2bc173e1..1f59c2d8 100644 --- a/auto_route/lib/src/route/page_info.dart +++ b/auto_route/lib/src/route/page_info.dart @@ -1,20 +1,44 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/cupertino.dart'; + /// Holds information of the generated [RoutePage] page /// /// Might hold more info in the future -class PageInfo { +class PageInfo { /// The name of the generated [RoutePage] final String name; + /// The builder function of the generated [RoutePage] + final Widget Function(RouteData data) builder; + /// Default constructor - const PageInfo(this.name); + const PageInfo(this.name, {required this.builder}); @override bool operator ==(Object other) => identical(this, other) || other is PageInfo && runtimeType == other.runtimeType && + builder == other.builder && name == other.name; @override - int get hashCode => name.hashCode; + int get hashCode => name.hashCode ^ builder.hashCode; + + /// Dummy [PageInfo] used to represent a redirect route + static final redirect = _NoBuilderPageInfo('#Redirect-Route'); + + /// Dummy [PageInfo] used to represent the root route + static final root = _NoBuilderPageInfo('#Root'); +} + +/// Dummy [PageInfo] used to represent a redirect route +class _NoBuilderPageInfo extends PageInfo { + /// Default constructor + _NoBuilderPageInfo(super.name) + : super( + builder: (data) { + throw FlutterError('RedirectPageInfo does not have a builder'); + }, + ); } diff --git a/auto_route/lib/src/route/page_route_info.dart b/auto_route/lib/src/route/page_route_info.dart index d43fb6a3..695e6502 100644 --- a/auto_route/lib/src/route/page_route_info.dart +++ b/auto_route/lib/src/route/page_route_info.dart @@ -194,3 +194,21 @@ class PageRouteInfo { const MapEquality().hash(rawQueryParams) ^ const ListEquality().hash(initialChildren); } + +/// A proxy Route page that provides a way to create a [PageRouteInfo] +/// without the need for creating a new Page Widget +class EmptyShellRoute { + /// Page name + final String name; + + /// Default constructor + EmptyShellRoute(this.name); + + /// creates [PageInfo] with an just an [AutoRouter] widget + late final page = PageInfo(name, builder: (_) => const AutoRouter()); + + /// Creates a new instance with of [PageRouteInfo] + PageRouteInfo call({List? children}) { + return PageRouteInfo(name, initialChildren: children); + } +} diff --git a/auto_route/lib/src/route/route_data.dart b/auto_route/lib/src/route/route_data.dart index c07f5c05..c7edc270 100644 --- a/auto_route/lib/src/route/route_data.dart +++ b/auto_route/lib/src/route/route_data.dart @@ -185,4 +185,9 @@ class RouteData { } return _match; } + + /// Builds the [AutoRoutePage] page + AutoRoutePage buildPage() { + return _match.buildPage(this); + } } diff --git a/auto_route/lib/src/route/route_type.dart b/auto_route/lib/src/route/route_type.dart index 3122cd9b..834651a2 100644 --- a/auto_route/lib/src/route/route_type.dart +++ b/auto_route/lib/src/route/route_type.dart @@ -3,7 +3,7 @@ import 'package:flutter/material.dart'; /// Signature for custom router builder used by /// [CustomRouteType] -typedef CustomRouteBuilder = Route Function( +typedef CustomRouteBuilder = Route Function( BuildContext context, Widget child, AutoRoutePage page, @@ -92,12 +92,22 @@ class CustomRouteType extends RouteType { /// /// I couldn't type this function from here but it should match /// typedef [CustomRouteBuilder] = Route Function(BuildContext context, CustomPage page); - /// you should only reference the function when passing it so - /// the generator can import it into the generated file /// /// this builder function accepts a BuildContext and a CustomPage /// that has all the other properties assigned to it /// so using them then is totally up to you. + /// + /// Make sure you pass the Return Type to the Route function + /// ex: + /// CustomRoute( + /// path: '/user/:userID', + /// page: UserRoute.page, + /// customRouteBuilder: (context, child, page) { + /// return PageRouteBuilder( + /// settings: page, + /// pageBuilder: (context, _, __) => child, + /// ); + /// }, final CustomRouteBuilder? customRouteBuilder; /// route transition duration in milliseconds diff --git a/auto_route/lib/src/router/auto_route_page.dart b/auto_route/lib/src/router/auto_route_page.dart index 0a1dce28..76551030 100644 --- a/auto_route/lib/src/router/auto_route_page.dart +++ b/auto_route/lib/src/router/auto_route_page.dart @@ -87,7 +87,7 @@ class AutoRoutePage extends Page { } else if (type is CustomRouteType) { final result = buildPage(context); if (type.customRouteBuilder != null) { - return type.customRouteBuilder!(context, result, this) as Route; + return type.customRouteBuilder!.call(context, result, this); } return _CustomPageBasedPageRouteBuilder(page: this, routeType: type); } else if (type is AdaptiveRouteType) { diff --git a/auto_route/lib/src/router/auto_router_x.dart b/auto_route/lib/src/router/auto_router_x.dart index 521fd218..50bed511 100644 --- a/auto_route/lib/src/router/auto_router_x.dart +++ b/auto_route/lib/src/router/auto_router_x.dart @@ -26,12 +26,6 @@ extension AutoRouterX on BuildContext { }) => router.replace(route, onFailure: onFailure); - /// see [StackRouter.maybePop] - @optionalTypeArgs - @Deprecated('Renamed to maybePop to avoid confusion') - Future popRoute([T? result]) => - router.maybePop(result); - /// see [StackRouter.maybePop] @optionalTypeArgs Future maybePop([T? result]) => diff --git a/auto_route/lib/src/router/controller/auto_router_delegate.dart b/auto_route/lib/src/router/controller/auto_router_delegate.dart index bbd6fc22..742a4e3f 100644 --- a/auto_route/lib/src/router/controller/auto_router_delegate.dart +++ b/auto_route/lib/src/router/controller/auto_router_delegate.dart @@ -94,20 +94,6 @@ class AutoRouterDelegate extends RouterDelegate with ChangeNotifier { reevaluateListenable?.addListener(controller.reevaluateGuards); } - /// Builds a [_DeclarativeAutoRouterDelegate] which uses - /// a declarative list of routes to update navigator stack - @Deprecated( - 'Declarative Root routing is not longer supported, Use route guards to conditionally navigate') - factory AutoRouterDelegate.declarative( - RootStackRouter controller, { - required RoutesBuilder routes, - String? navRestorationScopeId, - RoutePopCallBack? onPopRoute, - OnNavigateCallBack? onNavigate, - NavigatorObserversBuilder navigatorObservers, - DeepLinkBuilder? deepLinkBuilder, - }) = _DeclarativeAutoRouterDelegate; - /// Helper to access current urlState UrlState get urlState => controller.navigationHistory.urlState; @@ -274,94 +260,6 @@ class _AutoRootRouterState extends State<_AutoRootRouter> { } } -class _DeclarativeAutoRouterDelegate extends AutoRouterDelegate { - final RoutesBuilder routes; - final RoutePopCallBack? onPopRoute; - final OnNavigateCallBack? onNavigate; - - _DeclarativeAutoRouterDelegate( - RootStackRouter router, { - required this.routes, - String? navRestorationScopeId, - super.deepLinkBuilder, - this.onPopRoute, - this.onNavigate, - NavigatorObserversBuilder navigatorObservers = - AutoRouterDelegate.defaultNavigatorObserversBuilder, - }) : super( - router, - navRestorationScopeId: navRestorationScopeId, - navigatorObservers: navigatorObservers, - ) { - router._managedByWidget = true; - } - - @override - Future setInitialRoutePath(UrlState configuration) async { - final platformDeepLink = PlatformDeepLink._(configuration, true); - if (deepLinkBuilder != null) { - final deepLink = await deepLinkBuilder!(platformDeepLink); - _handleDeclarativeDeepLink(deepLink); - } else if (configuration.hasSegments) { - _handleDeclarativeDeepLink(platformDeepLink); - } - return SynchronousFuture(null); - } - - void _handleDeclarativeDeepLink(DeepLink deepLink) { - if (deepLink is _IgnoredDeepLink) return; - throwIf(!deepLink.isValid, 'Can not resolve initial route'); - List? routes; - if (deepLink is PlatformDeepLink) { - routes = deepLink.matches.map((e) => e.toPageRouteInfo()).toList(); - } else if (deepLink is _PathDeepLink) { - routes = controller.buildPageRoutesStack(deepLink.path); - } else if (deepLink is _RoutesDeepLink) { - routes = deepLink.routes; - } - controller.pendingRoutesHandler._setPendingRoutes(routes); - } - - @override - Future setNewRoutePath(UrlState tree) async { - return _onNavigate(tree); - } - - Future _onNavigate(UrlState tree) { - if (tree.hasSegments) { - controller.navigateAll(tree.segments); - } - if (onNavigate != null) { - onNavigate!(tree); - } - - return SynchronousFuture(null); - } - - @override - Widget build(BuildContext context) { - final stateHash = controller.stateHash; - return RouterScope( - controller: controller, - inheritableObserversBuilder: navigatorObservers, - stateHash: stateHash, - navigatorObservers: _navigatorObservers, - child: StackRouterScope( - controller: controller, - stateHash: stateHash, - child: AutoRouteNavigator( - router: controller, - key: GlobalObjectKey(controller.hashCode), - declarativeRoutesBuilder: routes, - navRestorationScopeId: navRestorationScopeId, - navigatorObservers: _navigatorObservers, - didPop: onPopRoute, - ), - ), - ); - } -} - /// Holds deep-link information abstract class DeepLink { const DeepLink._(); diff --git a/auto_route/lib/src/router/controller/root_stack_router.dart b/auto_route/lib/src/router/controller/root_stack_router.dart index 220b77e3..0cf9d768 100644 --- a/auto_route/lib/src/router/controller/root_stack_router.dart +++ b/auto_route/lib/src/router/controller/root_stack_router.dart @@ -3,17 +3,10 @@ part of 'routing_controller.dart'; /// Signature for a function uses [pagesMap] to build an [AutoRoutePage] typedef PageBuilder = AutoRoutePage Function(RouteData data); -/// Signature for a function that builds an [AutoRoutePage] -/// Used by [RoutingController] -typedef PageFactory = Page Function(RouteData data); - /// An Implementation of [StackRouter] used by [AutoRouterDelegate] abstract class RootStackRouter extends StackRouter { /// Default constructor - RootStackRouter({super.navigatorKey}) - : super( - key: const ValueKey('Root'), - ) { + RootStackRouter({super.navigatorKey}) : super(key: const ValueKey('Root')) { _navigationHistory = NavigationHistory.create(this); } @@ -57,7 +50,7 @@ abstract class RootStackRouter extends StackRouter { type: const RouteType.material(), stackKey: _stackKey, route: RouteMatch( - config: DummyRootRoute('Root', path: ''), + config: DummyRootRoute(path: ''), segments: const [''], stringMatch: '', key: const ValueKey('Root'), @@ -65,12 +58,12 @@ abstract class RootStackRouter extends StackRouter { pendingChildren: const [], ); - /// The map holding the page names and their factories - Map get pagesMap => throw UnimplementedError(); - /// The list of route entries to match against List get routes; + /// A List of Root router guards + List get guards => const []; + /// The default animation RouteType get defaultRouteType => const RouteType.material(); @@ -94,34 +87,8 @@ abstract class RootStackRouter extends StackRouter { ); } - @override - PageBuilder get pageBuilder => _pageBuilder; - AutoRouterDelegate? _lazyRootDelegate; - /// Builds a lazy instance of [AutoRouterDelegate.declarative] - @Deprecated( - 'Declarative Root routing is not longer supported, Use route guards to conditionally navigate') - AutoRouterDelegate declarativeDelegate({ - required RoutesBuilder routes, - String? navRestorationScopeId, - RoutePopCallBack? onPopRoute, - OnNavigateCallBack? onNavigate, - DeepLinkBuilder? deepLinkBuilder, - NavigatorObserversBuilder navigatorObservers = - AutoRouterDelegate.defaultNavigatorObserversBuilder, - }) { - return _lazyRootDelegate ??= AutoRouterDelegate.declarative( - this, - routes: routes, - onNavigate: onNavigate, - onPopRoute: onPopRoute, - navRestorationScopeId: navRestorationScopeId, - navigatorObservers: navigatorObservers, - deepLinkBuilder: deepLinkBuilder, - ); - } - /// Builds a lazy instance of [AutoRouterDelegate] /// _lazyRootDelegate is only built one time AutoRouterDelegate delegate({ @@ -155,12 +122,6 @@ abstract class RootStackRouter extends StackRouter { deepLinkTransformer: deepLinkTransformer ?? (uri) async => uri, ); - AutoRoutePage _pageBuilder(RouteData data) { - var builder = pagesMap[data.name]; - assert(builder != null); - return builder!(data) as AutoRoutePage; - } - @override void updateRouteData(RouteData data) { throw FlutterError('Root RouteData should not update'); diff --git a/auto_route/lib/src/router/controller/routing_controller.dart b/auto_route/lib/src/router/controller/routing_controller.dart index f0aaebb7..fdc65a2b 100644 --- a/auto_route/lib/src/router/controller/routing_controller.dart +++ b/auto_route/lib/src/router/controller/routing_controller.dart @@ -13,8 +13,11 @@ import 'package:flutter/material.dart'; import 'package:meta/meta.dart'; part '../../route/route_data.dart'; + part 'auto_route_guard.dart'; + part 'auto_router_delegate.dart'; + part 'root_stack_router.dart'; // ignore_for_file: deprecated_member_use_from_same_package @@ -501,9 +504,6 @@ abstract class RoutingController with ChangeNotifier { ); } - /// The builder used to build routable pages - PageBuilder get pageBuilder; - /// Clients can either pop their own [_pages] stack /// or defer the call to a parent controller /// @@ -511,20 +511,6 @@ abstract class RoutingController with ChangeNotifier { @optionalTypeArgs Future maybePop([T? result]); - /// Clients can either pop their own [_pages] stack - /// or defer the call to a parent controller - /// - /// see [Navigator.maybePop(context)] for more details - @optionalTypeArgs - @Deprecated( - 'pop was renamed to maybePop to avoid confusion, if you are looking for the implementation of Navigator.pop user popForced') - Future pop([T? result]) => maybePop(result); - - /// Calls [maybePop] on the controller with the top-most visible page - @optionalTypeArgs - @Deprecated('pop was renamed to maybePopTop') - Future popTop([T? result]) => maybePopTop(result); - /// Calls [maybePop] on the controller with the top-most visible page @optionalTypeArgs Future maybePopTop([T? result]) => @@ -644,8 +630,6 @@ class TabsRouter extends RoutingController { @override final RouteCollection routeCollection; @override - final PageBuilder pageBuilder; - @override final RouteMatcher matcher; RouteData _routeData; int _activeIndex = 0; @@ -661,7 +645,6 @@ class TabsRouter extends RoutingController { /// Default constructor TabsRouter( {required this.routeCollection, - required this.pageBuilder, required this.key, required RouteData routeData, this.homeIndex = -1, @@ -786,7 +769,7 @@ class TabsRouter extends RoutingController { for (var route in routes) { var data = _createRouteData(route, routeData); try { - _pages.add(pageBuilder(data)); + _pages.add(data.buildPage()); } on MissingRequiredParameterError catch (e) { if (fromDefault) { throw FlutterError( @@ -867,7 +850,7 @@ class TabsRouter extends RoutingController { } final data = _createRouteData(mayUpdateRoute, routeData); - _pages[pageToUpdateIndex] = pageBuilder(data); + _pages[pageToUpdateIndex] = data.buildPage(); final hasInitCtrl = _innerControllerOf(mayUpdateRoute.key) != null; @@ -1037,9 +1020,6 @@ abstract class StackRouter extends RoutingController { @override RouteCollection get routeCollection; - @override - PageBuilder get pageBuilder; - @override RouteMatcher get matcher; @@ -1522,7 +1502,7 @@ abstract class StackRouter extends RoutingController { } routesToPush.add(match); final data = _createRouteData(match, routeData); - _pages.add(pageBuilder(data)); + _pages.add(data.buildPage()); } navigationHistory.onNewUrlState( @@ -1603,17 +1583,14 @@ abstract class StackRouter extends RoutingController { _removeRoute(topRoute._match, notify: false); } final data = _createRouteData(route, routeData); - final page = pageBuilder(data); + final page = data.buildPage(); _pages.add(page); if (notify) { notifyAll(); } - return (page as AutoRoutePage).popped; + return page.popped; } - late final AutoRouteGuard? _rootGuard = - (root is AutoRouteGuard) ? (root as AutoRouteGuard) : null; - Future _canNavigate( RouteMatch route, { OnNavigationFailure? onFailure, @@ -1621,7 +1598,7 @@ abstract class StackRouter extends RoutingController { bool isReevaluating = false, }) async { final guards = [ - if (_rootGuard != null) _rootGuard!, + ...root.guards, ...route.guards, ]; if (guards.isEmpty) { @@ -1820,8 +1797,7 @@ class NestedStackRouter extends StackRouter { final RouteMatcher matcher; @override final RouteCollection routeCollection; - @override - final PageBuilder pageBuilder; + @override final bool managedByWidget; @@ -1830,7 +1806,6 @@ class NestedStackRouter extends StackRouter { /// Default constructor NestedStackRouter({ required this.routeCollection, - required this.pageBuilder, required super.key, required RouteData routeData, this.managedByWidget = false, diff --git a/auto_route/lib/src/router/widgets/auto_router.dart b/auto_route/lib/src/router/widgets/auto_router.dart index 9b150380..3403b718 100644 --- a/auto_route/lib/src/router/widgets/auto_router.dart +++ b/auto_route/lib/src/router/widgets/auto_router.dart @@ -132,9 +132,7 @@ class AutoRouterState extends State { routeCollection: _parentController.routeCollection.subCollectionOf( parentRouteData.name, ), - pageBuilder: _parentController.pageBuilder, ); - _parentController.attachChildController(_controller!); _controller!.addListener(_rebuildListener); } @@ -239,16 +237,16 @@ class _DeclarativeAutoRouterState extends State<_DeclarativeAutoRouter> { _navigatorObservers = _inheritableObserversBuilder(); _parentController = parentScope.controller; _controller = NestedStackRouter( - parent: _parentController, - key: parentData.key, - routeData: parentData, - managedByWidget: true, - onNavigate: widget.onNavigate, - navigatorKey: widget.navigatorKey, - routeCollection: _parentController.routeCollection.subCollectionOf( - parentData.name, - ), - pageBuilder: _parentController.pageBuilder); + parent: _parentController, + key: parentData.key, + routeData: parentData, + managedByWidget: true, + onNavigate: widget.onNavigate, + navigatorKey: widget.navigatorKey, + routeCollection: _parentController.routeCollection.subCollectionOf( + parentData.name, + ), + ); _parentController.attachChildController(_controller!); } } diff --git a/auto_route/lib/src/router/widgets/auto_tabs_router.dart b/auto_route/lib/src/router/widgets/auto_tabs_router.dart index 22fdd44d..c5cdee9b 100644 --- a/auto_route/lib/src/router/widgets/auto_tabs_router.dart +++ b/auto_route/lib/src/router/widgets/auto_tabs_router.dart @@ -195,7 +195,6 @@ abstract class AutoTabsRouterState extends State { routeCollection: _parentController.routeCollection.subCollectionOf( parentRoute.name, ), - pageBuilder: _parentController.pageBuilder, ); _parentController.attachChildController(_controller!); _setupController(); diff --git a/auto_route/lib/src/router/widgets/custom_cupertino_transitions_builder.dart b/auto_route/lib/src/router/widgets/custom_cupertino_transitions_builder.dart index 87a3b954..bcbd7132 100644 --- a/auto_route/lib/src/router/widgets/custom_cupertino_transitions_builder.dart +++ b/auto_route/lib/src/router/widgets/custom_cupertino_transitions_builder.dart @@ -51,63 +51,6 @@ final Animatable _kBottomUpTween = Tween( end: Offset.zero, ); -/// A modal route that replaces the entire screen with an iOS transition. -/// -/// {@macro flutter.cupertino.cupertinoRouteTransitionMixin} -/// -/// By default, when a modal route is replaced by another, the previous route -/// remains in memory. To free all the resources when this is not necessary, set -/// [maintainState] to false. -/// -/// The type `T` specifies the return type of the route which can be supplied as -/// the route is popped from the stack via [Navigator.pop] when an optional -/// `result` can be provided. -/// -/// See also: -/// -/// * [CustomCupertinoRouteTransitionMixin], for a mixin that provides iOS transition -/// for this modal route. -/// * [MaterialPageRoute], for an adaptive [PageRoute] that uses a -/// platform-appropriate transition. -/// * [CupertinoPageScaffold], for applications that have one page with a fixed -/// navigation bar on top. -/// * [CupertinoTabScaffold], for applications that have a tab bar at the -/// bottom with multiple pages. -/// * [CupertinoPage], for a [Page] version of this class. -class CupertinoPageRoute extends PageRoute - with - CupertinoRouteTransitionMixin, - CupertinoRouteTransitionOverrideMixin { - /// Creates a page route for use in an iOS designed app. - /// - /// The [builder], [maintainState], and [fullscreenDialog] arguments must not - /// be null. - CupertinoPageRoute({ - required this.builder, - this.title, - super.settings, - this.maintainState = true, - super.fullscreenDialog, - }) { - assert(opaque); - } - - /// Builds the primary contents of the route. - final WidgetBuilder builder; - - @override - Widget buildContent(BuildContext context) => builder(context); - - @override - final String? title; - - @override - final bool maintainState; - - @override - String get debugLabel => '${super.debugLabel}(${settings.name})'; -} - /// A mixin that implements methods and/or parameters of a [PageRoute]. /// /// Meant to be used as an override of methods/parameters implemented in @@ -541,30 +484,31 @@ class _CupertinoEdgeShadowDecoration extends Decoration { double t, ) { if (a == null && b == null) return null; + final aColors = a?._colors; + final bColors = b?._colors; if (a == null) { - return b!._colors == null + return bColors == null ? b - : _CupertinoEdgeShadowDecoration._(b._colors! + : _CupertinoEdgeShadowDecoration._(bColors .map((Color color) => Color.lerp(null, color, t)!) .toList()); } if (b == null) { - return a._colors == null + return aColors == null ? a - : _CupertinoEdgeShadowDecoration._(a._colors! + : _CupertinoEdgeShadowDecoration._(aColors .map((Color color) => Color.lerp(null, color, 1.0 - t)!) .toList()); } - assert(b._colors != null || a._colors != null); + assert(bColors != null || aColors != null); // If it ever becomes necessary, we could allow decorations with different // length' here, similarly to how it is handled in [LinearGradient.lerp]. - assert(b._colors == null || - a._colors == null || - a._colors!.length == b._colors!.length); + assert( + bColors == null || aColors == null || aColors.length == bColors.length); return _CupertinoEdgeShadowDecoration._( [ - for (int i = 0; i < b._colors!.length; i += 1) - Color.lerp(a._colors?[i], b._colors?[i], t)!, + for (int i = 0; i < bColors!.length; i += 1) + Color.lerp(a._colors?[i], bColors[i], t)!, ], ); } @@ -679,34 +623,3 @@ class _CupertinoEdgeShadowPainter extends BoxPainter { } } } - -/// A custom cupertino transition builder to fix unwanted shadows in nested navigator -/// -/// This fixes the issue referenced here -/// https://stackoverflow.com/questions/53457772/why-there-is-a-shadow-between-nested-navigator -/// https://stackoverflow.com/questions/68986632/rid-of-elevation-of-nested-flutter-navigator-2-0 -@Deprecated( - 'The issue, this builder fixes, was already fixed in the freamework in v3.0.0' - '(https://github.com/flutter/flutter/pull/95537, ' - 'https://docs.flutter.dev/release/release-notes/release-notes-3.0.0) ' - 'so there is no more reason to use this builder. ' - 'Will be deleted in the next major release. ' - 'If you still need this builder for other reasons, ' - 'use CupertinoPageTransitionsBuilder instead. as it is completely identic', -) -class NoShadowCupertinoPageTransitionsBuilder extends PageTransitionsBuilder { - /// Constructs a page transition animation that matches the iOS transition. - const NoShadowCupertinoPageTransitionsBuilder(); - - @override - Widget buildTransitions( - PageRoute route, - BuildContext context, - Animation animation, - Animation secondaryAnimation, - Widget child, - ) { - return CupertinoRouteTransitionMixin.buildPageTransitions( - route, context, animation, secondaryAnimation, child); - } -} diff --git a/auto_route/pubspec.yaml b/auto_route/pubspec.yaml index 1c351acd..9a94b941 100644 --- a/auto_route/pubspec.yaml +++ b/auto_route/pubspec.yaml @@ -1,7 +1,14 @@ name: auto_route description: AutoRoute is a declarative routing solution, where everything needed for navigation is automatically generated for you. -version: 8.3.0 +version: 9.0.0 homepage: https://github.com/Milad-Akarie/auto_route_library +screenshots: + - description: 'Auto Route Logo' + path: ../art/auto_route_logo.png +topics: + - Navigation + - Router + - Flutter Routes environment: sdk: ">=3.0.0 <4.0.0" @@ -11,7 +18,7 @@ dependencies: flutter: sdk: flutter path: ^1.9.0 - web: ^0.5.1 + web: ^1.0.0 collection: ^1.18.0 meta: ^1.12.0 @@ -21,7 +28,7 @@ dependencies: dev_dependencies: build_runner: mockito: ^5.4.4 - auto_route_generator: ^8.1.0 + auto_route_generator: ^9.0.0 flutter_test: sdk: flutter flutter_lints: ^4.0.0 diff --git a/auto_route/test/main_router.gr.dart b/auto_route/test/main_router.gr.dart index 4576df5c..43bb24f8 100644 --- a/auto_route/test/main_router.gr.dart +++ b/auto_route/test/main_router.gr.dart @@ -12,144 +12,6 @@ part of 'main_router.dart'; abstract class _$MainRouter extends RootStackRouter { // ignore: unused_element _$MainRouter({super.navigatorKey}); - - @override - final Map pagesMap = { - DeclarativeRouterHostRoute.name: (routeData) { - final args = routeData.argsAs(); - return AutoRoutePage( - routeData: routeData, - child: DeclarativeRouterHostScreen( - key: args.key, - pageNotifier: args.pageNotifier, - ), - ); - }, - FirstRoute.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const FirstPage(), - ); - }, - FourthRoute.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const FourthPage(), - ); - }, - NotFoundRoute.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const NotFoundPage(), - ); - }, - SecondHostRoute.name: (routeData) { - final args = routeData.argsAs( - orElse: () => const SecondHostRouteArgs()); - return AutoRoutePage( - routeData: routeData, - child: SecondHostPage( - key: args.key, - useCustomLeading: args.useCustomLeading, - hasDrawer: args.hasDrawer, - ), - ); - }, - SecondNested1Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const SecondNested1Page(), - ); - }, - SecondNested2Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const SecondNested2Page(), - ); - }, - SecondNested3Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const SecondNested3Page(), - ); - }, - SecondRoute.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const SecondPage(), - ); - }, - Tab1Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab1Page(), - ); - }, - Tab2Nested1Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab2Nested1Page(), - ); - }, - Tab2Nested2Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab2Nested2Page(), - ); - }, - Tab2Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab2Page(), - ); - }, - Tab3Nested1Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab3Nested1Page(), - ); - }, - Tab3Nested2Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab3Nested2Page(), - ); - }, - Tab3Route.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const Tab3Page(), - ); - }, - TabsHostRoute.name: (routeData) { - final queryParams = routeData.queryParams; - final args = routeData.argsAs( - orElse: () => TabsHostRouteArgs( - tabsType: queryParams.getString( - 'tabsType', - 'IndexedStack', - ), - useDefaultRoutes: queryParams.getBool( - 'useDefaultRoutes', - false, - ), - )); - return AutoRoutePage( - routeData: routeData, - child: TabsHostPage( - key: args.key, - tabsType: args.tabsType, - useDefaultRoutes: args.useDefaultRoutes, - ), - ); - }, - ThirdRoute.name: (routeData) { - return AutoRoutePage( - routeData: routeData, - child: const ThirdPage(), - ); - }, - }; } /// generated route for @@ -171,8 +33,16 @@ class DeclarativeRouterHostRoute static const String name = 'DeclarativeRouterHostRoute'; - static const PageInfo page = - PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return DeclarativeRouterHostScreen( + key: args.key, + pageNotifier: args.pageNotifier, + ); + }, + ); } class DeclarativeRouterHostRouteArgs { @@ -202,7 +72,12 @@ class FirstRoute extends PageRouteInfo { static const String name = 'FirstRoute'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const FirstPage(); + }, + ); } /// generated route for @@ -216,7 +91,12 @@ class FourthRoute extends PageRouteInfo { static const String name = 'FourthRoute'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const FourthPage(); + }, + ); } /// generated route for @@ -230,7 +110,12 @@ class NotFoundRoute extends PageRouteInfo { static const String name = 'NotFoundRoute'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const NotFoundPage(); + }, + ); } /// generated route for @@ -253,8 +138,18 @@ class SecondHostRoute extends PageRouteInfo { static const String name = 'SecondHostRoute'; - static const PageInfo page = - PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + final args = data.argsAs( + orElse: () => const SecondHostRouteArgs()); + return SecondHostPage( + key: args.key, + useCustomLeading: args.useCustomLeading, + hasDrawer: args.hasDrawer, + ); + }, + ); } class SecondHostRouteArgs { @@ -287,7 +182,12 @@ class SecondNested1Route extends PageRouteInfo { static const String name = 'SecondNested1Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const SecondNested1Page(); + }, + ); } /// generated route for @@ -301,7 +201,12 @@ class SecondNested2Route extends PageRouteInfo { static const String name = 'SecondNested2Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const SecondNested2Page(); + }, + ); } /// generated route for @@ -315,7 +220,12 @@ class SecondNested3Route extends PageRouteInfo { static const String name = 'SecondNested3Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const SecondNested3Page(); + }, + ); } /// generated route for @@ -329,7 +239,12 @@ class SecondRoute extends PageRouteInfo { static const String name = 'SecondRoute'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const SecondPage(); + }, + ); } /// generated route for @@ -343,7 +258,12 @@ class Tab1Route extends PageRouteInfo { static const String name = 'Tab1Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab1Page(); + }, + ); } /// generated route for @@ -357,7 +277,12 @@ class Tab2Nested1Route extends PageRouteInfo { static const String name = 'Tab2Nested1Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab2Nested1Page(); + }, + ); } /// generated route for @@ -371,7 +296,12 @@ class Tab2Nested2Route extends PageRouteInfo { static const String name = 'Tab2Nested2Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab2Nested2Page(); + }, + ); } /// generated route for @@ -385,7 +315,12 @@ class Tab2Route extends PageRouteInfo { static const String name = 'Tab2Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab2Page(); + }, + ); } /// generated route for @@ -399,7 +334,12 @@ class Tab3Nested1Route extends PageRouteInfo { static const String name = 'Tab3Nested1Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab3Nested1Page(); + }, + ); } /// generated route for @@ -413,7 +353,12 @@ class Tab3Nested2Route extends PageRouteInfo { static const String name = 'Tab3Nested2Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab3Nested2Page(); + }, + ); } /// generated route for @@ -427,7 +372,12 @@ class Tab3Route extends PageRouteInfo { static const String name = 'Tab3Route'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const Tab3Page(); + }, + ); } /// generated route for @@ -454,8 +404,28 @@ class TabsHostRoute extends PageRouteInfo { static const String name = 'TabsHostRoute'; - static const PageInfo page = - PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + final queryParams = data.queryParams; + final args = data.argsAs( + orElse: () => TabsHostRouteArgs( + tabsType: queryParams.getString( + 'tabsType', + 'IndexedStack', + ), + useDefaultRoutes: queryParams.getBool( + 'useDefaultRoutes', + false, + ), + )); + return TabsHostPage( + key: args.key, + tabsType: args.tabsType, + useDefaultRoutes: args.useDefaultRoutes, + ); + }, + ); } class TabsHostRouteArgs { @@ -488,5 +458,10 @@ class ThirdRoute extends PageRouteInfo { static const String name = 'ThirdRoute'; - static const PageInfo page = PageInfo(name); + static PageInfo page = PageInfo( + name, + builder: (data) { + return const ThirdPage(); + }, + ); } diff --git a/auto_route/test/page_info_test.dart b/auto_route/test/page_info_test.dart deleted file mode 100644 index 4db4cc4c..00000000 --- a/auto_route/test/page_info_test.dart +++ /dev/null @@ -1,8 +0,0 @@ -import 'package:auto_route/auto_route.dart'; -import 'package:flutter_test/flutter_test.dart'; - -void main() { - test("PageInfo equality test", () { - expect(const PageInfo('Name') == const PageInfo('Name'), true); - }); -} diff --git a/auto_route/test/parameters_test.dart b/auto_route/test/parameters_test.dart index 785c4377..76c01621 100644 --- a/auto_route/test/parameters_test.dart +++ b/auto_route/test/parameters_test.dart @@ -126,9 +126,10 @@ void main() { expect(params.getBool('key2'), false); }); - test('Calling Parameters.getList on a list value should return a ', - () { - const params = Parameters({'key': ['1', '2']}); + test('Calling Parameters.getList on a list value should return a ', () { + const params = Parameters({ + 'key': ['1', '2'] + }); expect(params.getList('key'), isA()); expect(params.optList('key2'), null); }); diff --git a/auto_route_generator/CHANGELOG.md b/auto_route_generator/CHANGELOG.md index c0856dcd..9d46096c 100644 --- a/auto_route_generator/CHANGELOG.md +++ b/auto_route_generator/CHANGELOG.md @@ -1,3 +1,24 @@ +## 9.0.0 [Breaking Changes] + +- **BREAKING CHANGE**: No Router class will be generated anymore. Instead, you + extend `RootStackRouter` from the `auto_route` package. +- **BREAKING CHANGE**: Providing return types inside `@RoutePage()` is no longer needed. you + just provide the type as you push the page. +- **BREAKING CHANGE**: Providing a global route is now done by overriding the `guards` property + inside the router. implementing AutoRouteGuard is no longer supported. +- **BREAKING CHANGE**: `AutoRouterConfig.module` is removed as it's no longer needed. `PageRouteInfos` are now self-contained. +- +For more info read the complete migration guide +[Migrating to v9](https://github.com/Milad-Akarie/auto_route_library/blob/master/migrations/migrating_to_v9.md) + +- **FIX**: Fix Aliased types are not generated correctly. +- **FEAT**: You can now create empty shell routes like follows: + ```dart + final BooksTab = EmptyShellRoute('BooksTab'); + context.push(BooksTab()); + ``` + + ## 8.1.0 - **FEAT**: add url#fragment support. - **CHORE**: update auto_route_generator dependencies diff --git a/auto_route_generator/lib/src/builders/auto_router_builder_base.dart b/auto_route_generator/lib/src/builders/auto_router_builder_base.dart index 0d4d504f..181b01b6 100644 --- a/auto_route_generator/lib/src/builders/auto_router_builder_base.dart +++ b/auto_route_generator/lib/src/builders/auto_router_builder_base.dart @@ -16,7 +16,6 @@ import 'package:source_gen/source_gen.dart'; import '../../utils.dart'; import '../resolvers/router_config_resolver.dart'; -import '../resolvers/type_resolver.dart'; const _typeChecker = TypeChecker.fromRuntime(AutoRouterConfig); @@ -51,11 +50,15 @@ abstract class AutoRouterBuilderBase extends CacheAwareBuilder { for (final clazz in unit.declarations .whereType() .where((e) => e.metadata.any((e) => e.name.name == annotationName))) { - final routerAnnotation = clazz.metadata.firstWhere((e) => e.name.name == annotationName); - final partDirectives = - unit.directives.whereType().fold(0, (acc, a) => acc ^ a.toSource().hashCode); - calculatedHash = - calculatedHash ^ clazz.name.toString().hashCode ^ routerAnnotation.toSource().hashCode ^ partDirectives; + final routerAnnotation = + clazz.metadata.firstWhere((e) => e.name.name == annotationName); + final partDirectives = unit.directives + .whereType() + .fold(0, (acc, a) => acc ^ a.toSource().hashCode); + calculatedHash = calculatedHash ^ + clazz.name.toString().hashCode ^ + routerAnnotation.toSource().hashCode ^ + partDirectives; } return calculatedHash; } @@ -69,7 +72,8 @@ abstract class AutoRouterBuilderBase extends CacheAwareBuilder { } @override - Future onGenerateContent(BuildStep buildStep, RouterConfig item) async { + Future onGenerateContent( + BuildStep buildStep, RouterConfig item) async { final generateForDir = item.generateForDir; final generatedResults = []; final routes = []; @@ -104,7 +108,8 @@ abstract class AutoRouterBuilderBase extends CacheAwareBuilder { } @override - Future onResolve(LibraryReader library, BuildStep buildStep, int stepHash) async { + Future onResolve( + LibraryReader library, BuildStep buildStep, int stepHash) async { final annotatedElements = library.annotatedWith(_typeChecker); if (annotatedElements.isEmpty) return null; final element = annotatedElements.first.element; @@ -119,9 +124,7 @@ abstract class AutoRouterBuilderBase extends CacheAwareBuilder { final usesPartBuilder = _hasPartDirective(clazz); - final router = RouterConfigResolver( - TypeResolver(await buildStep.resolver.libraries.toList()), - ).resolve( + final router = RouterConfigResolver().resolve( annotation, buildStep.inputId, clazz, @@ -132,7 +135,8 @@ abstract class AutoRouterBuilderBase extends CacheAwareBuilder { return router; } - void _writeRouterFile(BuildStep buildStep, RouterConfig router, {bool toCache = false}) { + void _writeRouterFile(BuildStep buildStep, RouterConfig router, + {bool toCache = false}) { final routerFile = _buildRouterFile(buildStep, toCache: toCache); if (!routerFile.existsSync()) { routerFile.createSync(recursive: true); diff --git a/auto_route_generator/lib/src/builders/cache_aware_builder.dart b/auto_route_generator/lib/src/builders/cache_aware_builder.dart index e02017b0..3c84bc67 100644 --- a/auto_route_generator/lib/src/builders/cache_aware_builder.dart +++ b/auto_route_generator/lib/src/builders/cache_aware_builder.dart @@ -22,7 +22,8 @@ abstract class CacheAwareBuilder extends Builder { bool get cacheEnabled; /// Custom ignore for file rules passed from the options - Set get ignoreForFile => options?.config['ignore_for_file']?.cast()?.toSet() ?? {}; + Set get ignoreForFile => + options?.config['ignore_for_file']?.cast()?.toSet() ?? {}; @override final Map> buildExtensions; diff --git a/auto_route_generator/lib/src/code_builder/deferred_pages_allocator.dart b/auto_route_generator/lib/src/code_builder/deferred_pages_allocator.dart index da174f00..64084d94 100644 --- a/auto_route_generator/lib/src/code_builder/deferred_pages_allocator.dart +++ b/auto_route_generator/lib/src/code_builder/deferred_pages_allocator.dart @@ -33,9 +33,11 @@ class DeferredPagesAllocator implements Allocator { Iterable get imports => _imports.keys.map( (importPath) { if (routes.isDeferred(importPath, defaultDeferredLoading)) { - return Directive.importDeferredAs(importPath, '_i${_imports[importPath]}'); + return Directive.importDeferredAs( + importPath, '_i${_imports[importPath]}'); } else { - return Directive.import(importPath, as: '_i${_imports[importPath]}'); + return Directive.import(importPath, + as: '_i${_imports[importPath]}'); } }, ); @@ -43,10 +45,12 @@ class DeferredPagesAllocator implements Allocator { extension _RouteConfigList on List { bool isDeferred(String importPath, bool defaultDeferredLoading) { - return mapRouteToDeferredType(importPath, defaultDeferredLoading) == _DeferredStatus.Deferred; + return mapRouteToDeferredType(importPath, defaultDeferredLoading) == + _DeferredStatus.Deferred; } - _DeferredStatus mapRouteToDeferredType(String importPath, bool defaultDeferredLoading) { + _DeferredStatus mapRouteToDeferredType( + String importPath, bool defaultDeferredLoading) { for (RouteConfig routeConfig in this) { if (routeConfig.pageType?.import == importPath) { return (routeConfig.deferredLoading ?? defaultDeferredLoading) diff --git a/auto_route_generator/lib/src/code_builder/library_builder.dart b/auto_route_generator/lib/src/code_builder/library_builder.dart index ee4cfe4e..7cb9230e 100644 --- a/auto_route_generator/lib/src/code_builder/library_builder.dart +++ b/auto_route_generator/lib/src/code_builder/library_builder.dart @@ -7,7 +7,6 @@ import '../models/route_config.dart'; import '../models/router_config.dart'; import 'deferred_pages_allocator.dart'; import 'route_info_builder.dart'; -import 'router_config_builder.dart'; /// AutoRoute imports const autoRouteImport = 'package:auto_route/auto_route.dart'; @@ -22,10 +21,11 @@ const Reference stringRefer = Reference('String'); const Reference pageRouteType = Reference('PageRouteInfo', autoRouteImport); /// Builds a list type reference of type [reference] -TypeReference listRefer(Reference reference, {bool nullable = false}) => TypeReference((b) => b - ..symbol = "List" - ..isNullable = nullable - ..types.add(reference)); +TypeReference listRefer(Reference reference, {bool nullable = false}) => + TypeReference((b) => b + ..symbol = "List" + ..isNullable = nullable + ..types.add(reference)); /// Generates the router library output String generateLibrary( @@ -39,7 +39,9 @@ String generateLibrary( ); final emitter = DartEmitter( - allocator: router.usesPartBuilder ? Allocator.none : DeferredPagesAllocator(routes, router.deferredLoading), + allocator: router.usesPartBuilder + ? Allocator.none + : DeferredPagesAllocator(routes, router.deferredLoading), orderDirectives: true, useNullSafetySyntax: true, ); @@ -52,7 +54,8 @@ String generateLibrary( for (var i = 0; i < routes.length; i++) { final route = routes[i]; - if (deferredRoutes.any((e) => e.pageType == route.pageType && route.deferredLoading != true)) { + if (deferredRoutes.any( + (e) => e.pageType == route.pageType && route.deferredLoading != true)) { routes[i] = route.copyWith(deferredLoading: true); } } @@ -68,7 +71,6 @@ String generateLibrary( for (final ignore in ignoreForFile) "ignore_for_file: $ignore", ]) ..body.addAll([ - buildRouterConfig(router, routes), if (routes.isNotEmpty) ...routes .distinctBy((e) => e.getName(router.replaceInRouteName)) diff --git a/auto_route_generator/lib/src/code_builder/route_info_builder.dart b/auto_route_generator/lib/src/code_builder/route_info_builder.dart index 41818a22..ef1b15fa 100644 --- a/auto_route_generator/lib/src/code_builder/route_info_builder.dart +++ b/auto_route_generator/lib/src/code_builder/route_info_builder.dart @@ -11,16 +11,10 @@ List buildRouteInfoAndArgs( RouteConfig r, RouterConfig router, DartEmitter emitter) { final argsClassRefer = refer('${r.getName(router.replaceInRouteName)}Args'); final parameters = r.parameters; - final fragmentParam = parameters.firstWhereOrNull((e)=> e.isUrlFragment); + final fragmentParam = parameters.firstWhereOrNull((e) => e.isUrlFragment); final nonInheritedParameters = parameters.where((p) => !p.isInheritedPathParam).toList(); - final pageInfoRefer = TypeReference( - (b) => b - ..url = autoRouteImport - ..symbol = 'PageInfo' - ..types.add( - (nonInheritedParameters.isNotEmpty) ? argsClassRefer : refer('void')), - ); + final pageInfoRefer = refer('PageInfo', autoRouteImport); return [ Class( (b) => b @@ -50,11 +44,11 @@ List buildRouteInfoAndArgs( ), Field( (b) => b - ..modifier = FieldModifier.constant ..name = 'page' ..static = true ..type = pageInfoRefer - ..assignment = pageInfoRefer.newInstance([refer('name')]).code, + ..assignment = pageInfoRefer.newInstance( + [refer('name')], {'builder': _buildMethod(r, router)}).code, ), ]) ..constructors.add( @@ -107,7 +101,7 @@ List buildRouteInfoAndArgs( ), ), ), - if(fragmentParam != null) + if (fragmentParam != null) 'fragment': refer(fragmentParam.name), 'initialChildren': refer('children'), }).code); @@ -178,3 +172,122 @@ Iterable buildArgParams( ), ); } + +Expression _buildMethod(RouteConfig r, RouterConfig router) { + final useConsConstructor = + r.hasConstConstructor && !(r.deferredLoading ?? router.deferredLoading); + var constructedPage = useConsConstructor + ? r.pageType!.refer.constInstance([]) + : _getPageInstance(r); + + if (r.hasWrappedRoute == true) { + constructedPage = refer('WrappedRoute', autoRouteImport).newInstance( + [], + {'child': constructedPage}, + ); + } + + if ((r.deferredLoading ?? router.deferredLoading) && r.pageType != null) { + constructedPage = _getDeferredBuilder(r, constructedPage); + } + final inheritedParameters = r.parameters.where((p) => p.isInheritedPathParam); + + final nonInheritedParameters = + r.parameters.where((p) => !p.isInheritedPathParam); + return Method( + (b) => b + ..requiredParameters.add( + Parameter((b) => b.name = 'data'), + ) + ..body = Block((b) => b.statements.addAll([ + if ((!r.hasUnparsableRequiredArgs) && + r.parameters.any((p) => p.isPathParam) || + inheritedParameters.isNotEmpty) + declareFinal('pathParams') + .assign(refer('data').property('inheritedPathParams')) + .statement, + if (!r.hasUnparsableRequiredArgs && + r.parameters.any((p) => p.isQueryParam)) + declareFinal('queryParams') + .assign(refer('data').property('queryParams')) + .statement, + if (nonInheritedParameters.isNotEmpty) + declareFinal('args') + .assign( + refer('data').property('argsAs').call([], { + if (!r.hasUnparsableRequiredArgs) + 'orElse': Method( + (b) => b + ..lambda = true + ..body = r.pathQueryParams.isEmpty + ? refer('${r.getName(router.replaceInRouteName)}Args') + .constInstance([]).code + : refer('${r.getName(router.replaceInRouteName)}Args') + .newInstance( + [], + Map.fromEntries( + nonInheritedParameters + .where((p) => (p.isPathParam || + p.isQueryParam || + p.isUrlFragment)) + .map( + (p) => MapEntry( + p.name, + _getUrlPartAssignment(p), + ), + ), + ), + ).code, + ).closure + }, [ + refer('${r.getName(router.replaceInRouteName)}Args'), + ]), + ) + .statement, + constructedPage.returned.statement + ])), + ).closure; +} + +Expression _getDeferredBuilder(RouteConfig r, Expression page) { + return TypeReference((b) => b + ..symbol = 'DeferredWidget' + ..url = autoRouteImport).newInstance([ + TypeReference((b) => b + ..symbol = 'loadLibrary' + ..url = r.pageType!.refer.url), + Method((b) => b..body = page.code).closure + ]); +} + +Expression _getPageInstance(RouteConfig r) { + return r.pageType!.refer.newInstance( + r.positionalParams.map((p) { + return refer('args').property(p.name); + }), + Map.fromEntries(r.namedParams.map( + (p) => MapEntry( + p.name, + p.isInheritedPathParam + ? _getUrlPartAssignment(p) + : refer('args').property(p.name), + ), + )), + ); +} + +Expression _getUrlPartAssignment(ParamConfig p) { + if (p.isPathParam) { + return refer('pathParams').property(p.getterMethodName).call([ + literalString(p.paramName), + if (p.defaultValueCode != null) refer(p.defaultValueCode!), + ]); + } else if (p.isQueryParam) { + return refer('queryParams').property(p.getterMethodName).call([ + literalString(p.paramName), + if (p.defaultValueCode != null) refer(p.defaultValueCode!), + ]); + } else { + return refer('data').property('fragment'); + } +} diff --git a/auto_route_generator/lib/src/code_builder/router_config_builder.dart b/auto_route_generator/lib/src/code_builder/router_config_builder.dart deleted file mode 100644 index 2cbb8bf7..00000000 --- a/auto_route_generator/lib/src/code_builder/router_config_builder.dart +++ /dev/null @@ -1,218 +0,0 @@ -import 'package:code_builder/code_builder.dart'; - -import '../../utils.dart'; -import '../models/route_config.dart'; -import '../models/route_parameter_config.dart'; -import '../models/router_config.dart'; -import 'library_builder.dart'; - -/// Builds a router config class for the given [RouterConfig] -Class buildRouterConfig(RouterConfig router, List routes) => Class( - (b) => b - ..name = - '${router.usesPartBuilder ? '_' : ''}\$${router.routerClassName}' - ..abstract = true - ..extend = refer( - router.isModule ? 'AutoRouterModule' : 'RootStackRouter', - autoRouteImport, - ) - ..fields.addAll([_buildPagesMap(routes, router)]) - ..constructors.addAll(router.isModule - ? [] - : [ - Constructor((b) => b - ..docs.addAll( - [if (router.usesPartBuilder) '// ignore: unused_element']) - ..optionalParameters.add( - Parameter((b) => b - ..name = 'navigatorKey' - ..named = true - ..toSuper = true), - )), - ]), - ); - -Field _buildPagesMap(List routes, RouterConfig router) { - return Field( - (b) => b - ..name = "pagesMap" - ..modifier = FieldModifier.final$ - ..annotations.add(refer('override')) - ..type = TypeReference( - (b) => b - ..symbol = 'Map' - ..types.addAll([ - stringRefer, - refer('PageFactory', autoRouteImport), - ]), - ) - ..assignment = _buildAssignment(routes, router), - ); -} - -/// Builds the map assignment for the pagesMap field -Code _buildAssignment(List routes, RouterConfig router) { - final effectiveRoutes = - routes.distinctBy((e) => e.getName(router.replaceInRouteName)).map( - (r) => MapEntry( - refer(r.getName(router.replaceInRouteName)).property('name'), - _buildMethod(r, router), - ), - ); - final modules = router.modules; - - if (modules.isEmpty) { - return literalMap(Map.fromEntries(effectiveRoutes)).code; - } - - return Block.of([ - Code('{'), - for (final route in effectiveRoutes) ...[ - route.key.code, - Code(':'), - route.value.code, - Code(',') - ], - for (final module in modules) ...[ - refer(module.name, module.import) - .newInstance([]) - .property('pagesMap') - .spread - .code, - Code(',') - ], - Code('}'), - ]); -} - -Expression _buildMethod(RouteConfig r, RouterConfig router) { - final useConsConstructor = - r.hasConstConstructor && !(r.deferredLoading ?? router.deferredLoading); - var constructedPage = useConsConstructor - ? r.pageType!.refer.constInstance([]) - : _getPageInstance(r); - - if (r.hasWrappedRoute == true) { - constructedPage = refer('WrappedRoute', autoRouteImport).newInstance( - [], - {'child': constructedPage}, - ); - } - - if ((r.deferredLoading ?? router.deferredLoading) && r.pageType != null) { - constructedPage = _getDeferredBuilder(r, constructedPage); - } - final inheritedParameters = r.parameters.where((p) => p.isInheritedPathParam); - - final nonInheritedParameters = - r.parameters.where((p) => !p.isInheritedPathParam); - return Method( - (b) => b - ..requiredParameters.add( - Parameter((b) => b.name = 'routeData'), - ) - ..body = Block((b) => b.statements.addAll([ - if ((!r.hasUnparsableRequiredArgs) && - r.parameters.any((p) => p.isPathParam) || - inheritedParameters.isNotEmpty) - declareFinal('pathParams') - .assign(refer('routeData').property('inheritedPathParams')) - .statement, - if (!r.hasUnparsableRequiredArgs && - r.parameters.any((p) => p.isQueryParam)) - declareFinal('queryParams') - .assign(refer('routeData').property('queryParams')) - .statement, - if (nonInheritedParameters.isNotEmpty) - declareFinal('args') - .assign( - refer('routeData').property('argsAs').call([], { - if (!r.hasUnparsableRequiredArgs) - 'orElse': Method( - (b) => b - ..lambda = true - ..body = r.pathQueryParams.isEmpty - ? refer('${r.getName(router.replaceInRouteName)}Args') - .constInstance([]).code - : refer('${r.getName(router.replaceInRouteName)}Args') - .newInstance( - [], - Map.fromEntries( - nonInheritedParameters - .where((p) => - (p.isPathParam || p.isQueryParam || p.isUrlFragment)) - .map( - (p) => MapEntry( - p.name, - _getUrlPartAssignment(p), - ), - ), - ), - ).code, - ).closure - }, [ - refer('${r.getName(router.replaceInRouteName)}Args'), - ]), - ) - .statement, - TypeReference( - (b) => b - ..symbol = 'AutoRoutePage' - ..url = autoRouteImport - ..types.add(r.returnType?.refer ?? refer('dynamic')), - ) - .newInstance( - [], - { - 'routeData': refer('routeData'), - 'child': constructedPage, - }, - ) - .returned - .statement - ])), - ).closure; -} - -Expression _getDeferredBuilder(RouteConfig r, Expression page) { - return TypeReference((b) => b - ..symbol = 'DeferredWidget' - ..url = autoRouteImport).newInstance([ - TypeReference((b) => b - ..symbol = 'loadLibrary' - ..url = r.pageType!.refer.url), - Method((b) => b..body = page.code).closure - ]); -} - -Expression _getPageInstance(RouteConfig r) { - return r.pageType!.refer.newInstance( - r.positionalParams.map((p) { - return refer('args').property(p.name); - }), - Map.fromEntries(r.namedParams.map( - (p) => MapEntry( - p.name, - p.isInheritedPathParam - ? _getUrlPartAssignment(p) - : refer('args').property(p.name), - ), - )), - ); -} - -Expression _getUrlPartAssignment(ParamConfig p) { - if (p.isPathParam) { - return refer('pathParams').property(p.getterMethodName).call([ - literalString(p.paramName), - if (p.defaultValueCode != null) refer(p.defaultValueCode!), - ]); - } else if(p.isQueryParam) { - return refer('queryParams').property(p.getterMethodName).call([ - literalString(p.paramName), - if (p.defaultValueCode != null) refer(p.defaultValueCode!), - ]); - }else{ - return refer('routeData').property('fragment'); - } -} diff --git a/auto_route_generator/lib/src/models/resolved_type.dart b/auto_route_generator/lib/src/models/resolved_type.dart index 239ff97a..f614203f 100644 --- a/auto_route_generator/lib/src/models/resolved_type.dart +++ b/auto_route_generator/lib/src/models/resolved_type.dart @@ -1,4 +1,5 @@ -import 'package:code_builder/code_builder.dart' show TypeReference, RecordType, Reference; +import 'package:code_builder/code_builder.dart' + show TypeReference, RecordType, Reference; /// A class that represents a resolved type. /// @@ -61,10 +62,15 @@ class ResolvedType { ..url = import ..isNullable = isNullable ..positionalFieldTypes.addAll( - typeArguments.where((e) => !e.isNamedRecordField).map((e) => e.refer), + typeArguments + .where((e) => !e.isNamedRecordField) + .map((e) => e.refer), ) ..namedFieldTypes.addAll({ - for (final entry in [...typeArguments.where((e) => e.isNamedRecordField)]) entry.nameInRecord!: entry.refer + for (final entry in [ + ...typeArguments.where((e) => e.isNamedRecordField) + ]) + entry.nameInRecord!: entry.refer }), ); } @@ -82,7 +88,10 @@ class ResolvedType { @override bool operator ==(Object other) => - identical(this, other) || other is ResolvedType && runtimeType == other.runtimeType && identity == other.identity; + identical(this, other) || + other is ResolvedType && + runtimeType == other.runtimeType && + identity == other.identity; @override int get hashCode => import.hashCode ^ name.hashCode; @@ -95,7 +104,8 @@ class ResolvedType { 'isNullable': this.isNullable, 'isRecordType': this._isRecordType, if (nameInRecord != null) 'nameInRecord': this.nameInRecord, - if (typeArguments.isNotEmpty) 'typeArguments': this.typeArguments.map((e) => e.toJson()).toList(), + if (typeArguments.isNotEmpty) + 'typeArguments': this.typeArguments.map((e) => e.toJson()).toList(), }; } diff --git a/auto_route_generator/lib/src/models/route_config.dart b/auto_route_generator/lib/src/models/route_config.dart index a4c3c748..92f83a0e 100644 --- a/auto_route_generator/lib/src/models/route_config.dart +++ b/auto_route_generator/lib/src/models/route_config.dart @@ -17,9 +17,6 @@ class RouteConfig { /// the class name of the route final String className; - /// the return type of the route - final ResolvedType? returnType; - /// the parameters of the route final List parameters; @@ -40,7 +37,6 @@ class RouteConfig { required this.className, this.parameters = const [], this.hasWrappedRoute, - this.returnType, this.hasConstConstructor = false, this.deferredLoading, }); @@ -61,10 +57,12 @@ class RouteConfig { } /// Returns all the required params - Iterable get requiredParams => parameters.where((p) => p.isPositional && !p.isOptional); + Iterable get requiredParams => + parameters.where((p) => p.isPositional && !p.isOptional); /// Returns all the optional params - Iterable get positionalParams => parameters.where((p) => p.isPositional); + Iterable get positionalParams => + parameters.where((p) => p.isPositional); /// Returns all the named params Iterable get namedParams => parameters.where((p) => p.isNamed); @@ -74,7 +72,8 @@ class RouteConfig { var nameToUse; if (name != null) { nameToUse = name; - } else if (replacementInRouteName != null && replacementInRouteName.split(',').length == 2) { + } else if (replacementInRouteName != null && + replacementInRouteName.split(',').length == 2) { var parts = replacementInRouteName.split(','); nameToUse = className.replaceAll(RegExp(parts[0]), parts[1]); } else { @@ -84,8 +83,8 @@ class RouteConfig { } /// Whether this route has arguments that can't be parsed - bool get hasUnparsableRequiredArgs => - parameters.any((p) => (p.isRequired || p.isPositional) && !p.isPathParam && !p.isQueryParam); + bool get hasUnparsableRequiredArgs => parameters.any((p) => + (p.isRequired || p.isPositional) && !p.isPathParam && !p.isQueryParam); /// Clones the route config with the given parameters RouteConfig copyWith({ @@ -96,7 +95,6 @@ class RouteConfig { bool? fullMatch, ResolvedType? pageType, String? className, - ResolvedType? returnType, List? parameters, String? redirectTo, bool? hasWrappedRoute, @@ -109,7 +107,6 @@ class RouteConfig { pathParams: pathParams ?? this.pathParams, pageType: pageType ?? this.pageType, className: className ?? this.className, - returnType: returnType ?? this.returnType, parameters: parameters ?? this.parameters, hasWrappedRoute: hasWrappedRoute ?? this.hasWrappedRoute, hasConstConstructor: hasConstConstructor ?? this.hasConstConstructor, @@ -124,7 +121,6 @@ class RouteConfig { 'pathParams': this.pathParams.map((e) => e.toJson()).toList(), 'pageType': this.pageType?.toJson(), 'className': this.className, - 'returnType': this.returnType?.toJson(), 'parameters': this.parameters.map((e) => e.toJson()).toList(), 'hasWrappedRoute': this.hasWrappedRoute, 'hasConstConstructor': this.hasConstConstructor, @@ -151,9 +147,10 @@ class RouteConfig { return RouteConfig( name: map['name'] as String?, pathParams: pathParams, - pageType: map['pageType'] == null ? null : ResolvedType.fromJson(map['pageType']), + pageType: map['pageType'] == null + ? null + : ResolvedType.fromJson(map['pageType']), className: map['className'] as String, - returnType: map['returnType'] == null ? null : ResolvedType.fromJson(map['returnType']), parameters: parameters, hasWrappedRoute: map['hasWrappedRoute'] as bool?, hasConstConstructor: map['hasConstConstructor'] as bool, diff --git a/auto_route_generator/lib/src/models/route_parameter_config.dart b/auto_route_generator/lib/src/models/route_parameter_config.dart index 7033e827..cdd8829a 100644 --- a/auto_route_generator/lib/src/models/route_parameter_config.dart +++ b/auto_route_generator/lib/src/models/route_parameter_config.dart @@ -47,6 +47,7 @@ class ParamConfig { /// whether the parameter is a url fragment final bool isUrlFragment; + /// whether the parameter is an inherited path param final bool isInheritedPathParam; @@ -102,7 +103,7 @@ class ParamConfig { case 'bool': return type.isNullable ? 'optBool' : 'getBool'; - case 'List' : + case 'List': return type.isNullable ? 'optList' : 'getList'; default: return 'get'; @@ -151,7 +152,6 @@ class ParamConfig { isQueryParam: map['isQueryParam'] as bool, isUrlFragment: map['isUrlFragment'] as bool, defaultValueCode: map['defaultValueCode'] as String?, - ); } } diff --git a/auto_route_generator/lib/src/models/router_config.dart b/auto_route_generator/lib/src/models/router_config.dart index 8e8240df..854f3074 100644 --- a/auto_route_generator/lib/src/models/router_config.dart +++ b/auto_route_generator/lib/src/models/router_config.dart @@ -1,5 +1,3 @@ -import 'resolved_type.dart'; - /// RouterConfig class RouterConfig { /// The name of the router class @@ -23,12 +21,6 @@ class RouterConfig { /// The list of directories to generate for final List generateForDir; - /// Whether the router is a module - final bool isModule; - - /// The list of modules - final List modules; - /// Default constructor const RouterConfig({ required this.routerClassName, @@ -38,8 +30,6 @@ class RouterConfig { required this.path, required this.cacheHash, required this.generateForDir, - this.isModule = false, - this.modules = const [], }); /// Serializes this instance to a map @@ -52,8 +42,6 @@ class RouterConfig { 'path': this.path, 'cacheHash': this.cacheHash, 'generateForDir': this.generateForDir, - 'isModule': this.isModule, - 'modules': this.modules, }; } @@ -67,10 +55,6 @@ class RouterConfig { path: map['path'] as String, cacheHash: map['cacheHash'] as int?, generateForDir: (map['generateForDir'] as List).cast(), - isModule: map['isModule'] as bool, - modules: (map['modules'] as List) - .map((e) => ResolvedType.fromJson(e)) - .toList(), ); } } diff --git a/auto_route_generator/lib/src/resolvers/route_config_resolver.dart b/auto_route_generator/lib/src/resolvers/route_config_resolver.dart index 54382abb..c4cedc11 100644 --- a/auto_route_generator/lib/src/resolvers/route_config_resolver.dart +++ b/auto_route_generator/lib/src/resolvers/route_config_resolver.dart @@ -1,9 +1,7 @@ import 'package:analyzer/dart/element/element.dart'; -import 'package:analyzer/dart/element/type.dart'; import 'package:source_gen/source_gen.dart'; import '../../utils.dart'; -import '../models/resolved_type.dart'; import '../models/route_config.dart'; import '../models/route_parameter_config.dart'; import '../resolvers/route_parameter_resolver.dart'; @@ -27,17 +25,11 @@ class RouteConfigResolver { final classElement = element as ClassElement; final page = classElement.thisType; - final hasWrappedRoute = - classElement.allSupertypes.any((e) => e.getDisplayString(withNullability: false) == 'AutoRouteWrapper'); + final hasWrappedRoute = classElement.allSupertypes.any((e) => + e.getDisplayString(withNullability: false) == 'AutoRouteWrapper'); var pageType = _typeResolver.resolveType(page); var className = page.getDisplayString(withNullability: false); - var returnType = ResolvedType(name: 'dynamic'); - var dartType = routePage.objectValue.type; - if (dartType is InterfaceType) { - returnType = _typeResolver.resolveType(dartType.typeArguments.first); - } - var name = routePage.peek('name')?.stringValue; final constructor = classElement.unnamedConstructor; throwIf( @@ -69,8 +61,10 @@ class RouteConfigResolver { var pathParameters = parameters.where((element) => element.isPathParam); if (parameters.any((p) => p.isPathParam || p.isQueryParam)) { - var unParsableRequiredArgs = - parameters.where((p) => (p.isRequired || p.isPositional) && !p.isPathParam && !p.isQueryParam); + var unParsableRequiredArgs = parameters.where((p) => + (p.isRequired || p.isPositional) && + !p.isPathParam && + !p.isQueryParam); if (unParsableRequiredArgs.isNotEmpty) { print( '\nWARNING => Because [$className] has required parameters ${unParsableRequiredArgs.map((e) => e.paramName)} ' @@ -93,7 +87,6 @@ class RouteConfigResolver { hasWrappedRoute: hasWrappedRoute, parameters: parameters, hasConstConstructor: hasConstConstructor, - returnType: returnType, pageType: pageType, deferredLoading: isDeferred, ); diff --git a/auto_route_generator/lib/src/resolvers/route_parameter_resolver.dart b/auto_route_generator/lib/src/resolvers/route_parameter_resolver.dart index 9e95c812..a054b7db 100644 --- a/auto_route_generator/lib/src/resolvers/route_parameter_resolver.dart +++ b/auto_route_generator/lib/src/resolvers/route_parameter_resolver.dart @@ -21,7 +21,7 @@ class RouteParameterResolver { /// Resolves a ParameterElement into a consumable [ParamConfig] ParamConfig resolve(ParameterElement parameterElement) { final paramType = parameterElement.type; - if (paramType is FunctionType) { + if (paramType is FunctionType && paramType.alias == null) { return _resolveFunctionType(parameterElement); } var type = _typeResolver.resolveType(paramType); @@ -31,7 +31,8 @@ class RouteParameterResolver { var nameOrAlias = paramName; var isInheritedPathParam = false; - final isUrlFragment = _urlFragmentChecker.hasAnnotationOfExact(parameterElement); + final isUrlFragment = + _urlFragmentChecker.hasAnnotationOfExact(parameterElement); if (pathParamAnnotation != null) { isInheritedPathParam = @@ -56,12 +57,15 @@ class RouteParameterResolver { } throwIf( - [isUrlFragment, pathParamAnnotation != null, queryParamAnnotation != null].where((e) => e).length > 1, + [isUrlFragment, pathParamAnnotation != null, queryParamAnnotation != null] + .where((e) => e) + .length > + 1, '${parameterElement.name} can only be annotated with one of @PathParam, @QueryParam or @urlFragment', element: parameterElement, ); - if(isUrlFragment){ + if (isUrlFragment) { throwIf( type.name != 'String', 'UrlFragments must be of type String', @@ -88,7 +92,6 @@ class RouteParameterResolver { isQueryParam: queryParamAnnotation != null, isUrlFragment: isUrlFragment, defaultValueCode: parameterElement.defaultValueCode, - ); } diff --git a/auto_route_generator/lib/src/resolvers/router_config_resolver.dart b/auto_route_generator/lib/src/resolvers/router_config_resolver.dart index 4680d030..dfdc6e1d 100644 --- a/auto_route_generator/lib/src/resolvers/router_config_resolver.dart +++ b/auto_route_generator/lib/src/resolvers/router_config_resolver.dart @@ -3,14 +3,11 @@ import 'package:build/build.dart'; import 'package:source_gen/source_gen.dart'; import '../models/router_config.dart'; -import '../resolvers/type_resolver.dart'; /// Extracts and holds router configs class RouterConfigResolver { - final TypeResolver _typeResolver; - /// Default constructor - RouterConfigResolver(this._typeResolver); + RouterConfigResolver(); /// Resolves a [ClassElement] into a consumable [RouterConfig] RouterConfig resolve( @@ -20,14 +17,6 @@ class RouterConfigResolver { bool usesPartBuilder = false, int? cacheHash, }) { - // /// ensure router config classes are prefixed with $ - // /// to use the stripped name for the generated class - // throwIf( - // !usesPartBuilder && !clazz.displayName.startsWith(r'$'), - // 'Router class name must be prefixed with \$', - // element: clazz, - // ); - final deferredLoading = autoRouter.peek('deferredLoading')?.boolValue ?? false; var replaceInRouteName = autoRouter.peek('replaceInRouteName')?.stringValue; @@ -35,11 +24,6 @@ class RouterConfigResolver { .read('generateForDir') .listValue .map((e) => e.toStringValue()!); - var isModule = autoRouter.read('_isModule').boolValue; - final modules = autoRouter - .peek('modules') - ?.listValue - .map((e) => _typeResolver.resolveType(e.toTypeValue()!)); return RouterConfig( routerClassName: clazz.displayName, @@ -49,8 +33,6 @@ class RouterConfigResolver { path: input.path, cacheHash: cacheHash, generateForDir: List.of(generateForDir), - isModule: isModule, - modules: List.of(modules ?? []), ); } } diff --git a/auto_route_generator/lib/src/resolvers/type_resolver.dart b/auto_route_generator/lib/src/resolvers/type_resolver.dart index b5329e58..f9b8faf8 100644 --- a/auto_route_generator/lib/src/resolvers/type_resolver.dart +++ b/auto_route_generator/lib/src/resolvers/type_resolver.dart @@ -1,9 +1,13 @@ import 'package:analyzer/dart/element/element.dart'; -import 'package:analyzer/dart/element/nullability_suffix.dart' show NullabilitySuffix; -import 'package:analyzer/dart/element/type.dart' show DartType, ParameterizedType, RecordType; +import 'package:analyzer/dart/element/nullability_suffix.dart' + show NullabilitySuffix; +import 'package:analyzer/dart/element/type.dart' + show DartType, ParameterizedType, RecordType; import 'package:auto_route_generator/src/models/resolved_type.dart'; import 'package:path/path.dart' as p; +const _unPreferredImports = {'dart:ui'}; + /// A Helper class that resolves types class TypeResolver { /// The list of resolved libraries in [BuildStep] @@ -21,16 +25,25 @@ class TypeResolver { if (libs.isEmpty || element?.source == null || _isCoreDartType(element!)) { return null; } + + final fallBackImports = {}; for (var lib in libs) { - if (!_isCoreDartType(lib) && lib.exportNamespace.definedNames.values.contains(element)) { + if (!_isCoreDartType(lib) && + lib.exportNamespace.definedNames.values.contains(element)) { final uri = lib.source.uri; + if (_unPreferredImports.contains(uri.toString())) { + fallBackImports.add(uri.toString()); + continue; + } if (uri.scheme == 'asset') { return _assetToPackage(lib.source.uri); } - return targetFile == null ? lib.identifier : _relative(uri, targetFile!); + return targetFile == null + ? lib.identifier + : _relative(uri, targetFile!); } } - return null; + return fallBackImports.firstOrNull; } String _assetToPackage(Uri uri) { @@ -52,12 +65,16 @@ class TypeResolver { String _relative(Uri fileUri, Uri to) { var libName = to.pathSegments.first; - if ((to.scheme == 'package' && fileUri.scheme == 'package' && fileUri.pathSegments.first == libName) || + if ((to.scheme == 'package' && + fileUri.scheme == 'package' && + fileUri.pathSegments.first == libName) || (to.scheme == 'asset' && fileUri.scheme != 'package')) { if (fileUri.path == to.path) { return fileUri.pathSegments.last; } else { - return p.posix.relative(fileUri.path, from: to.path).replaceFirst('../', ''); + return p.posix + .relative(fileUri.path, from: to.path) + .replaceFirst('../', ''); } } else { return fileUri.toString(); @@ -75,7 +92,8 @@ class TypeResolver { types.add(ResolvedType( name: recordField.type.element?.name ?? 'void', import: resolveImport(recordField.type.element), - isNullable: recordField.type.nullabilitySuffix == NullabilitySuffix.question, + isNullable: + recordField.type.nullabilitySuffix == NullabilitySuffix.question, typeArguments: _resolveTypeArguments(recordField.type), )); } @@ -83,7 +101,8 @@ class TypeResolver { types.add(ResolvedType( name: recordField.type.element?.name ?? 'void', import: resolveImport(recordField.type.element), - isNullable: recordField.type.nullabilitySuffix == NullabilitySuffix.question, + isNullable: + recordField.type.nullabilitySuffix == NullabilitySuffix.question, typeArguments: _resolveTypeArguments(recordField.type), nameInRecord: recordField.name, )); @@ -114,19 +133,30 @@ class TypeResolver { /// Resolves the given [type] to a [ResolvedType] ResolvedType resolveType(DartType type) { - if (type is RecordType) { + final effectiveElement = type.alias?.element ?? type.element; + final import = resolveImport(effectiveElement); + final typeArgs = []; + final alias = type.alias; + if (alias != null) { + typeArgs.addAll(alias.typeArguments.map(resolveType)); + } else { + typeArgs.addAll(_resolveTypeArguments(type)); + } + if (type is RecordType && type.alias == null) { return ResolvedType.record( - name: type.element?.name ?? 'void', - import: resolveImport(type.element), + name: effectiveElement?.displayName ?? 'void', + import: import, isNullable: type.nullabilitySuffix == NullabilitySuffix.question, - typeArguments: _resolveTypeArguments(type), + typeArguments: typeArgs, ); } + return ResolvedType( - name: type.element?.name ?? type.getDisplayString(withNullability: false), + name: effectiveElement?.displayName ?? + type.getDisplayString(withNullability: false), isNullable: type.nullabilitySuffix == NullabilitySuffix.question, - import: resolveImport(type.element), - typeArguments: _resolveTypeArguments(type), + import: import, + typeArguments: typeArgs, ); } } diff --git a/auto_route_generator/pubspec.yaml b/auto_route_generator/pubspec.yaml index 65c92933..61aa4e9e 100644 --- a/auto_route_generator/pubspec.yaml +++ b/auto_route_generator/pubspec.yaml @@ -1,6 +1,6 @@ name: auto_route_generator description: AutoRoute is a declarative routing solution, where everything needed for navigation is automatically generated for you. -version: 8.1.0 +version: 9.0.0 homepage: https://github.com/Milad-Akarie/auto_route_library environment: sdk: ">=3.3.0 <4.0.0" @@ -19,7 +19,7 @@ dependencies: args: ^2.5.0 glob: ^2.1.2 package_config: ^2.1.0 - auto_route: ^8.3.0 + auto_route: ^9.0.0 diff --git a/migrations/migrating_to_v6.md b/migrations/migrating_to_v6.md new file mode 100644 index 00000000..3b26b7ce --- /dev/null +++ b/migrations/migrating_to_v6.md @@ -0,0 +1,91 @@ +## Migrating to v6 + +In version 6.0 **AutoRoute** aims for less generated code for more flexibility and less generation time. + +#### 1. Instead of using `MaterialAutoRouter`, `CupertinoAutoRouter`, etc, we now only have one annotation for our router which is `@AutoRouterConfig()` and instead of passing our routes list to the annotation we now pass it to the overridable getter `routes` inside of the generated router class and for the default route type you can override `defaultRouteType` + +#### Before + +```dart +// @CupertinoAutoRouter +// @AdaptiveAutoRouter +// @CustomAutoRouter +@MaterialAutoRouter( + routes: [ + // routes go here + ], +) +class $AppRouter {} +``` + +#### After + + ```dart +@AutoRouterConfig() +class AppRouter extends $AppRouter { + + @override + RouteType get defaultRouteType => RouteType.material(); //.cupertino, .adaptive ..etc + + @override + List get routes => [ + // routes go here + ]; +} +``` + +#### 2. Passing page components as types is changed, now you'd annotate the target page with `@RoutePage()` annotation and pass the generated `result.page` to AutoRoute(): + +#### Before + +```dart +class ProductDetailsPage extends StatelessWidget {} +``` + +```dart +AutoRoute(page: ProductDetailsPage) // as Type +``` + +#### After + +```dart +@RoutePage() // Add this annotation to your routable pages +class ProductDetailsPage extends StatelessWidget {} +``` + +```dart +AutoRoute(page: ProductDetailsRoute.page) // ProductDetailsRoute is generated +``` + +#### 3. `EmptyRoutePage` no longer exists, instead you will now make your own empty pages by extending the `AutoRouter` widget + +#### Before + +```dart +AutoRoute(page: EmptyRoutePage, name: 'ProductsRouter') // as Type +``` + +#### After + +```dart +@RoutePage(name: 'ProductsRouter') +class ProductsRouterPage extends AutoRouter {} +``` + +```dart +AutoRoute(page: ProductsRouter.page) +``` + +#### 4. Passing route guards is also changed now, instead of passing guards as types you now pass instances. + +#### Before + +```dart +AutoRoute(page: ProfilePage, guards:[AuthGuard]) // as Type +``` + +#### After + +```dart +AutoRoute(page: ProfilePage, guards:[AuthGuard()]) // as Instance +``` \ No newline at end of file diff --git a/migrations/migrating_to_v9.md b/migrations/migrating_to_v9.md new file mode 100644 index 00000000..41d41507 --- /dev/null +++ b/migrations/migrating_to_v9.md @@ -0,0 +1,123 @@ +## Migrating to v9 + +#### 1. You now need to extend `RootStackRouter` from the `auto_route` package instead of the generated `$YOUR_APP_NAME`. + +#### Before + +```dart + +@AutoRouterConfig() +class AppRouter extends $AppRouter { + + @override + List get routes => [ + /// routes go here + ]; +} +``` +#### After + +```dart + +@AutoRouterConfig() +class AppRouter extends RootStackRouter { + + @override + List get routes => [ + /// routes go here + ]; +} +``` + +#### 2. You no longer need to provide the return type of a page inside `@RoutePage()` instead provide the return type as you push your page. + +#### Before + +```dart +@RoutePage() +class LoginPage extends StatelessWidget {} + +``` + +```dart + /// pushing the route +bool didLogin = await context.pushRoute(); +``` + +#### After + +```dart +/// provide the return type as you push your page + bool didLogin = await context.pushRoute(); +``` + +#### 3. Global guards are now provided as a list of `AutoRouteGuard` instead implementing `AutoRouteGuard` directly. + +#### Before + +```dart +@AutoRouterConfig() +class AppRouter extends $AppRouter implements AutoRouteGuard { + + @override + void onNavigation(NavigationResolver resolver, StackRouter router) { + /// guard logic + } + +} +``` +#### After + +```dart +@AutoRouterConfig() +class AppRouter extends RootStackRouter{ + + final authGuard = AuthGuard(); + + @override + late final List guards = [ + authGuard, ///add guard instance + /// + /// or use a simple guard wrapper + AutoRouteGuard.simple((resolver, router) { + /// guard logic + ), + + ]; +} +``` + +#### 4. AutoRouterConfig.module is no longer used, generated `PageRouteInfos` are now self-sufficient. they contain the page builder inside `PageRouteInfo.page`. +What you do now is generated the routes inside the micro package like normal, then either use the generated routes inside your main router individually, +or declare them inside your micro router then merge them with the main router. + +#### Before +```dart +@AutoRouterConfig.module() +class MyPackageModule extends $MyPackageModule {} +``` +```dart +@AutoRouterConfig(modules: [MyPackageModule]) +class AppRouter extends $AppRouter {} +``` + +#### After + +```dart +/// normal auto router config +@AutoRouterConfig() +class MyMicroRouter extends RootStackRouter{} +``` + +```dart + final myMicroRouter = MyMicroRouter(); + + @override + List get routes => [ + AutoRoute(page: HomeRoute.page, initial: true), + /// use micro routes individually + AutoRoute(page: RouteFromMicroPackage.page), + /// or merge all routes from micro router + ...myMicroRouter.routes, + ]; +``` \ No newline at end of file diff --git a/old/pre_v9_README.md b/old/pre_v9_README.md new file mode 100644 index 00000000..3628ea19 --- /dev/null +++ b/old/pre_v9_README.md @@ -0,0 +1,1622 @@ +

+ auto_route_logo +

+ +

+ + MIT License + + + stars + + + pub version + + + Discord Badge + +

+ +

+ + Buy Me A Coffee + +

+ +--- + +- [Introduction](#introduction) + - [Installation](#installation) + - [Setup and Usage](#setup-and-usage) +- [Generated routes](#generated-routes) +- [Navigation](#navigating-between-screens) + - [Navigating Between Screens](#navigating-between-screens) + - [Passing Arguments](#passing-arguments) + - [Returning Results](#returning-results) + - [Nested navigation](#nested-navigation) + - [Tab Navigation](#tab-navigation) + - [Using PageView](#using-pageview) + - [Using TabBar](#using-tabbar) + - [Finding The Right Router](#finding-the-right-router) + - [Navigating Without Context](#navigating-without-context) +- [Deep Linking](#deep-linking) +- [Declarative Navigation](#declarative-navigation) +- [Working with Paths](#working-with-paths) +- [Route guards](#route-guards) +- [Wrapping routes](#wrapping-routes) +- [Navigation Observers](#navigation-observers) +- [Customization](#customizations) + - [Custom Route Transitions](#custom-route-transitions) + - [Custom Route Builder](#custom-route-builder) +- [Others](#others) + - [Including Micro/External Packages](#including-microexternal-packages) + - [Configuring builders](#configuring-builders) + - [Optimizing Generation Time](#optimizing-generation-time) + - [Enabling Cached Builds (Experimental)](#enabling-cached-builds) + - [AutoLeadingButton-BackButton](#autoleadingbutton-backbutton) + - [ActiveGuardObserver](#activeguardobserver) +- [Examples](#examples) + +**Note:** [AutoRoute-Helper] is no longer supported. + +## Migration guides + +- [Migrating to v6](#migrating-to-v6) + +## Pre v6 documentation + +- [Pre v6 documentation](https://github.com/Milad-Akarie/auto_route_library/blob/master/old/pre_v6_README.md) + +## Introduction + +#### What is AutoRoute? + +It’s a Flutter navigation package, it allows for strongly-typed arguments passing, effortless deep-linking and it uses code generation to simplify routes setup. With that being said, it requires a minimal amount of code to generate everything needed for navigation inside of your App. + +#### Why AutoRoute? + +If your App requires deep-linking or guarded routes or just a clean routing setup, you'll need to use named/generated routes and you’ll end up writing a lot of boilerplate code for mediator argument classes, checking for required arguments, extracting arguments and a bunch of other stuff. **AutoRoute** does all that for you and much more. + +## Installation + + ```yaml +dependencies: + auto_route: [latest-version] + +dev_dependencies: + auto_route_generator: [latest-version] + build_runner: +``` + +## Setup And Usage + +1. Create a router class and annotate it with `@AutoRouterConfig` then extend "$YourClassName" +2. Override the routes getter and start adding your routes. + + ```dart +@AutoRouterConfig() +class AppRouter extends $AppRouter { + + @override + List get routes => [ + /// routes go here + ]; +} +``` + +### Using part builder + +To generate a part-of file simply add a `part` directive to your `AppRouter` and extend the generated private router. **Note:** The `deferredLoading` functionality does not work with part-file setup. + +```dart +part 'app_router.gr.dart'; + +@AutoRouterConfig() +class AppRouter extends _$AppRouter { + + @override + List get routes => [ + /// routes go here + ]; +} +``` + +### Generating Routable pages + +Routable pages are just simple everyday widgets annotated with `@RoutePage()` which allows them to be constructed by the router. + +```dart +@RoutePage() +class HomeScreen extends StatefulWidget {} +``` + +#### Now simply run the generator + +Use the [watch] flag to watch the files' system for edits and rebuild as necessary. + +```terminal +dart run build_runner watch +``` + +If you want the generator to run one time and exit, use + +```terminal +dart run build_runner build +``` + +#### Add the generated route to your routes list + +```dart +@AutoRouterConfig(replaceInRouteName: 'Screen,Route') +class AppRouter extends $AppRouter { + + @override + List get routes => [ + // HomeScreen is generated as HomeRoute because + // of the replaceInRouteName property + AutoRoute(page: HomeRoute.page), + ]; +} +``` + +#### Finalize the setup + +After you run the generator, your router class will be generated. Then simply hook it up with your MaterialApp. + +```dart +// assuming this is the root widget of your App +class App extends StatelessWidget { + // make sure you don't initiate your router + // inside of the build function. + final _appRouter = AppRouter(); + + @override + Widget build(BuildContext context){ + return MaterialApp.router( + routerConfig: _appRouter.config(), + ); + } +} +``` + +## Generated Routes + +A `PageRouteInfo` object will be generated for every declared **AutoRoute**. These objects hold strongly-typed page arguments which are extracted from the page's default constructor. Think of them as string path segments on steroids. + +```dart +class BookListRoute extends PageRouteInfo { + const BookListRoute({ + List? children, + }) : super(name, path: '/books', initialChildren: children); + + static const String name = 'BookListRoute'; + static const PageInfo page = PageInfo(name); +} +``` + +## Navigating Between Screens + +`AutoRouter` offers the same known push, pop and friends methods to manipulate the pages stack using both the generated `PageRouteInfo` objects and paths. + +```dart +// get the scoped router by calling +AutoRouter.of(context); +// or using the extension +context.router; +// adds a new entry to the pages stack +router.push(const BooksListRoute()); +// or by using paths +router.pushNamed('/books'); +// removes last entry in stack and pushes provided route +// if last entry == provided route page will just be updated +router.replace(const BooksListRoute()); +// or by using paths +router.replaceNamed('/books'); +// pops until provided route, if it already exists in stack +// else adds it to the stack (good for web Apps). +router.navigate(const BooksListRoute()); +// or by using paths +router.navigateNamed('/books'); +// on Web it calls window.history.back(); +// on Native it navigates you back +// to the previous location +router.back(); +// adds a list of routes to the pages stack at once +router.pushAll([ + BooksListRoute(), + BookDetailsRoute(id: 1), +]); +// This is like providing a completely new stack as it rebuilds the stack +// with the list of passed routes +// entries might just update if already exist +router.replaceAll([ + LoginRoute(), +]); +// pops the last page unless blocked or stack has only 1 entry +context.router.maybePop(); +// pops the most top page of the most top router unless blocked +// or stack has only 1 entry +context.router.maybePopTop(); +// keeps popping routes until predicate is satisfied +context.router.popUntil((route) => route.settings.name == 'HomeRoute'); +// a simplified version of the above line +context.router.popUntilRouteWithName('HomeRoute'); +// keeps popping routes until route with provided path is found +context.router.popUntilRouteWithPath('/some-path'); +// pops all routes down to the root +context.router.popUntilRoot(); +// removes the top most page in stack even if it's the last +// remove != pop, it doesn't respect WillPopScopes it just +// removes the entry. +context.router.removeLast(); +// removes any route in stack that satisfies the predicate +// this works exactly like removing items from a regular List +// [...].removeWhere((r)=>) +context.router.removeWhere((route) => ); +// you can also use the common helper methods from context extension to navigate +context.pushRoute(const BooksListRoute()); +context.replaceRoute(const BooksListRoute()); +context.navigateTo(const BooksListRoute()); +context.navigateNamedTo('/books'); +context.back(); +context.maybePop(); +``` + +## Passing Arguments + +That's the fun part! **AutoRoute** automatically detects and handles your page arguments for you, the generated route object will deliver all the arguments your page needs including path/query params. + +e.g. The following page widget will take an argument of type `Book`. + +```dart +@RoutePage() +class BookDetailsPage extends StatelessWidget { + const BookDetailsPage({required this.book}); + + final Book book; + ... +``` + +**Note:** Default values are respected. Required fields are also respected and handled properly. + +The generated `BookDetailsRoute` will deliver the same arguments to its corresponding page. + +```dart +router.push(BookDetailsRoute(book: book)); +``` + +**Note:** All arguments are generated as named parameters regardless of their original type. + +## Returning Results + +You can return results by either using the pop completer or by passing a callback function as an argument the same way you'd pass an object. + +#### 1. Using the `pop` completer + +```dart +var result = await router.push(LoginRoute()); +``` + +then inside of your `LoginPage`, pop with results + +```dart +router.maybePop(true); +``` + +as you'd notice we did not specify the result type, we're playing with dynamic values here, which can be risky and I personally don't recommend it. + +To avoid working with dynamic values, we specify what type of results we expect our page to return, which is a `bool` value. + +```dart +@RoutePage() +class LoginPage extends StatelessWidget {} +``` + +we push and specify the type of results we're expecting + +```dart +var result = await router.push(LoginRoute()); +``` + +and of course we pop with the same type + +```dart +router.maybePop(true); +``` + +#### 2. Passing a callback function as an argument. +We only have to add a callback function as a parameter to our page constructor like follows: + +```dart +@RoutePage() +class BookDetailsPage extends StatelessWidget { + const BookDetailsRoute({this.book, required this.onRateBook}); + + final Book book; + final void Function(int) onRateBook; + ... +``` + +The generated `BookDetailsRoute` will deliver the same arguments to its corresponding page. + +```dart +context.pushRoute( + BookDetailsRoute( + book: book, + onRateBook: (rating) { + // handle result + }, + ), +); +``` + +If you're finishing with results, make sure you call the callback function as you pop the page + +```dart +onRateBook(RESULT); +context.maybePop(); +``` + +**Note:** Default values are respected. Required fields are also respected and handled properly. + +## Nested Navigation + +Nested navigation means building an inner router inside of a page of another router, for example in the below diagram users page is built inside of dashboard page. + +

+ nested-router-demo +

+ +Defining nested routes is as easy as populating the children field of the parent route. In the following example `UsersPage`, `PostsPage` and `SettingsPage` are nested children of `DashboardPage`. + +```dart +@AutoRouterConfig(replaceInRouteName: 'Page,Route') +class AppRouter extends $AppRouter { + +@override +List get routes => [ + AutoRoute( + path: '/dashboard', + page: DashboardRoute.page, + children: [ + AutoRoute(path: 'users', page: UsersRoute.page), + AutoRoute(path: 'posts', page: PostsRoute.page), + AutoRoute(path: 'settings', page: SettingsRoute.page), + ], + ), + AutoRoute(path: '/login', page: LoginRoute.page), + ]; +} +``` + +To render/build nested routes we need an `AutoRouter` widget that works as an outlet or a nested router-view inside of our dashboard page. + +```dart +class DashboardPage extends StatelessWidget { + + @override + Widget build(BuildContext context) { + return Row( + children: [ + Column( + children: [ + NavLink(label: 'Users', destination: const UsersRoute()), + NavLink(label: 'Posts', destination: const PostsRoute()), + NavLink(label: 'Settings', destination: const SettingsRoute()), + ], + ), + Expanded( + // nested routes will be rendered here + child: AutoRouter(), + ), + ], + ); + } +} +``` + +**Note** NavLink is just a button that calls router.push(destination). Now if we navigate to `/dashboard/users`, we will be taken to the `DashboardPage` and the `UsersPage` will be shown inside of it. + +What if want to show one of the child pages at `/dashboard`? We can simply do that by giving the child routes an empty path `''` to make initial or by setting initial to true. + +```dart +AutoRoute( + path: '/dashboard', + page: DashboardRoute.page, + children: [ + AutoRoute(path: '', page: UsersRoute.page), + AutoRoute(path: 'posts', page: PostsRoute.page), + ], +) +``` + +or by using a `RedirectRoute` + +```dart +AutoRoute( + path: '/dashboard', + page: DashboardRoute.page, + children: [ + RedirectRoute(path: '', redirectTo: 'users'), + AutoRoute(path: 'users', page: UsersRoute.page), + AutoRoute(path: 'posts', page: PostsRoute.page), + ], +) +``` + +### Things to keep in mind when implementing nested navigation + +1. Each router manages its own pages stack. +2. Navigation actions like push, pop and friends are handled by the topmost router and bubble up if it couldn't be handled. + +## Tab Navigation + +If you're working with flutter mobile, you're most likely to implement tabs navigation, that's why `auto_route` makes tabs navigation as easy and straightforward as possible. + +In the previous example we used an `AutoRouter` widget to render nested child routes, `AutoRouter` is just a shortcut for `AutoStackRouter`. `StackRouters` manage a stack of pages inside of them, where the active/visible page is always the one on top and you'd need to pop it to see the page beneath it. + +Now we can try to implement our tabs using an `AutoRouter` (StackRouter) by pushing or replacing a nested route every time the tab changes and that might work, but our tabs state will be lost, not to mention the transition between tabs issue, luckily auto_route comes equipped with an `AutoTabsRouter`, which is especially made to handle tab navigation. + +`AutoTabsRouter` lets you switch between different routes while preserving offstage-routes state, tab routes are lazily loaded by default (can be disabled) and it finally allows to create whatever transition animation you want. + +Let's change the previous example to use tab navigation. + +Notice that we're not going to change anything in our routes declaration map, we still have a dashboard page that has three nested children: users, posts and settings. + +```dart +class DashboardPage extends StatelessWidget { + + @override + Widget build(BuildContext context) { + return AutoTabsRouter( + // list of your tab routes + // routes used here must be declared as children + // routes of /dashboard + routes: const [ + UsersRoute(), + PostsRoute(), + SettingsRoute(), + ], + transitionBuilder: (context,child,animation) => FadeTransition( + opacity: animation, + // the passed child is technically our animated selected-tab page + child: child, + ), + builder: (context, child) { + // obtain the scoped TabsRouter controller using context + final tabsRouter = AutoTabsRouter.of(context); + // Here we're building our Scaffold inside of AutoTabsRouter + // to access the tabsRouter controller provided in this context + // + // alternatively, you could use a global key + return Scaffold( + body: child, + bottomNavigationBar: BottomNavigationBar( + currentIndex: tabsRouter.activeIndex, + onTap: (index) { + // here we switch between tabs + tabsRouter.setActiveIndex(index); + }, + items: [ + BottomNavigationBarItem(label: 'Users', ...), + BottomNavigationBarItem(label: 'Posts', ...), + BottomNavigationBarItem(label: 'Settings', ...), + ], + ), + ); + }, + ); + } +} +``` + +If you think the above setup is a bit messy you could use the shipped-in `AutoTabsScaffold` that makes things much cleaner. + +```dart +class DashboardPage extends StatelessWidget { + @override + Widget build(BuildContext context) { + return AutoTabsScaffold( + routes: const [ + UsersRoute(), + PostsRoute(), + SettingsRoute(), + ], + bottomNavigationBuilder: (_, tabsRouter) { + return BottomNavigationBar( + currentIndex: tabsRouter.activeIndex, + onTap: tabsRouter.setActiveIndex, + items: const [ + BottomNavigationBarItem(label: 'Users', ...), + BottomNavigationBarItem(label: 'Posts', ...), + BottomNavigationBarItem(label: 'Settings', ...), + ], + ); + }, + ); + } +} +``` + +### Using PageView + +Use the `AutoTabsRouter.pageView` constructor to implement tabs using PageView + +```dart +AutoTabsRouter.pageView( + routes: [ + BooksTab(), + ProfileTab(), + SettingsTab(), + ], + builder: (context, child, _) { + final tabsRouter = AutoTabsRouter.of(context); + return Scaffold( + appBar: AppBar( + title: Text(context.topRoute.name), + leading: AutoLeadingButton(), + ), + body: child, + bottomNavigationBar: BottomNavigationBar( + currentIndex: tabsRouter.activeIndex, + onTap: tabsRouter.setActiveIndex, + items: [ + BottomNavigationBarItem(label: 'Books', ...), + BottomNavigationBarItem(label: 'Profile', ...), + BottomNavigationBarItem(label: 'Settings', ...), + ], + ), + ); + }, +); +``` + +### Using TabBar + +Use the `AutoTabsRouter.tabBar` constructor to implement tabs using TabBar + +```dart +AutoTabsRouter.tabBar( + routes: [ + BooksTab(), + ProfileTab(), + SettingsTab(), + ], + builder: (context, child, controller) { + final tabsRouter = AutoTabsRouter.of(context); + return Scaffold( + appBar: AppBar( + title: Text(context.topRoute.name), + leading: AutoLeadingButton(), + bottom: TabBar( + controller: controller, + tabs: const [ + Tab(text: '1', icon: Icon(Icons.abc)), + Tab(text: '2', icon: Icon(Icons.abc)), + Tab(text: '3', icon: Icon(Icons.abc)), + ], + ), + ), + body: child, + bottomNavigationBar: BottomNavigationBar( + currentIndex: tabsRouter.activeIndex, + onTap: tabsRouter.setActiveIndex, + items: [ + BottomNavigationBarItem(label: 'Books',...), + BottomNavigationBarItem(label: 'Profile',...), + BottomNavigationBarItem(label: 'Settings',...), + ], + ), + ); + }, +); +``` + +## Finding The Right Router + +Every nested `AutoRouter` has its own routing controller to manage the stack inside of it and the easiest way to obtain a scoped controller is by using the `BuildContext`. + +In the previous example, `DashboardPage` is a root level stack entry so calling `AutoRouter.of(context)` anywhere inside of it will get us the root routing controller. + +`AutoRouter` widgets that are used to render nested routes, insert a new router scope into the widgets tree, so when a nested route calls for the scoped controller, they will get the closest parent controller in the widgets tree; not the root controller. + +```dart +class Dashboard extends StatelessWidget { + + @override + Widget build(BuildContext context) { + // this will get us the root routing controller + AutoRouter.of(context); + return Scaffold( + appBar: AppBar(title: Text('Dashboard page')), + // this inserts a new router scope into the widgets tree + body: AutoRouter() + ); + } +} +``` + +Here's a simple diagram to help visualize this + +

+ scoped-router-demo +

+ +As you can tell from the above diagram it's possible to access parent routing controllers by calling `router.parent()`, we're using a generic function because we have two different routing controllers: `StackRouter` and `TabsRouter`, one of them could be the parent controller of the current router and that's why we need to specify a type. + +```dart +router.parent() // this returns the parent router as a Stack Routing controller +router.parent() // this returns the parent router as a Tabs Routing controller +``` + +On the other hand, obtaining the root controller does not require type casting because it's always a `StackRouter`. + +```dart +router.root // this returns the root router as a Stack Routing controller +``` + +You can obtain access to inner-routers from outside their scope using a global key + +```dart +class DashboardPage extends StatefulWidget { + @override + _DashboardPageState createState() => _DashboardPageState(); +} + +class _DashboardPageState extends State { + final _innerRouterKey = GlobalKey(); + + @override + Widget build(BuildContext context) { + return Row( + children: [ + Column( + children: [ + NavLink( + label: 'Users', + onTap: () { + final router = _innerRouterKey.currentState?.controller; + router?.push(const UsersRoute()); + }, + ), + ... + ], + ), + Expanded( + child: AutoRouter(key: _innerRouterKey), + ), + ], + ); + } +} +``` + +You could also obtain access to inner-routers from outside their scope without a global key, as long as they're initiated. + +```dart +// assuming this is the root router +context.innerRouterOf(UserRoute.name); +// or if we're using an AutoTabsRouter inside of DashboardPage +context.innerRouterOf(UserRoute.name); +``` + +Accessing the `DashboardPage` inner router from the previous example. + +```dart +class Dashboard extends StatelessWidget { + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar( + title: Text('Dashboard'), + actions: [ + IconButton( + icon: Icon(Icons.person), + onPressed: () { + // accessing the inner router from + // outside the scope + final router = context.innerRouterOf(DashboardRoute.name) + router?.push(const UsersRoute()); + }, + ), + ], + ), + body: AutoRouter(), // we're trying to get access to this + ); + } +} +``` + +## Navigating Without Context + +To navigate without context you can simply assign your generated router to a global variable + +```dart +// declare your route as a global variable +final appRouter = AppRouter(); + +class MyApp extends StatefulWidget { + + @override + Widget build(BuildContext context) { + return MaterialApp.router( + routerConfig: appRouter.config(), + ); + } +} +``` + +**Note:** Using global variable is not recommended and is considered bad practice and most of the times you should use dependency injection instead. + +Here's an example using `get_it` (which is just a personal favorite). You can use any dependency injection package you like. + +```dart +void main(){ + // make sure you register it as a Singleton or a lazySingleton + getIt.registerSingleton(AppRouter()); + runApp(MyApp()); +} + +class MyApp extends StatefulWidget { + @override + Widget build(BuildContext context) { + final appRouter = getIt(); + + return MaterialApp.router( + routerConfig: appRouter.config(), + ); + } +} +``` + +Now you can access your router anywhere inside of your app without using context. + +```dart +getIt().push(...); +``` + +**Note:** Navigating without context is not recommended in nested navigation unless you use `navigate` instead of `push` and you provide a full hierarchy, e.g `router.navigate(SecondRoute(children: [SubChild2Route()]))` + +## Deep Linking + +**AutoRoute** will automatically handle deep-links coming from the platform, but native platforms require some setup, see [Deep linking topic](https://docs.flutter.dev/ui/navigation/deep-linking) in flutter documentation. + +### Using Deep-link Transformer + +Deep link transformer intercepts deep-links before they're processed by the matcher, it's useful for stripping or modifying deep-links before they're matched. + +In the following example we will strip a prefix from the deep-link before it's matched. + +```dart +MaterialApp.router( + routerConfig: _appRouter.config( + deepLinkTransformer: (uri) { + if (uri.path.startsWith('/prefix')) { + return SynchronousFuture( + uri.replace(path: uri.path.replaceFirst('/prefix', '')), + ); + } + return SynchronousFuture(uri); + } + ), +); +``` +**Note** for prefix stripping use the shipped-in `DeepLink.prefixStripper('prefix')` + +```dart +MaterialApp.router( + routerConfig: _appRouter.config( + deepLinkTransformer: DeepLink.prefixStripper('prefix'), + ), +); +``` + +```dart +### Using Deep-link Builder + +Deep link builder is an interceptor for deep-links where you can validate or override deep-links coming from the platform. + +In the following example we will only allow deep-links starting with `/products` + +```dart +MaterialApp.router( + routerConfig: _appRouter.config( + deepLinkBuilder: (deepLink) { + if (deepLink.path.startsWith('/products')) { + // continue with the platform link + return deepLink; + } else { + return DeepLink.defaultPath; + // or DeepLink.path('/') + // or DeepLink([HomeRoute()]) + } + } + ), +); +``` + +### Deep Linking to non-nested Routes + +**AutoRoute** can build a stack from a linear route list as long as they're ordered properly and can be matched as prefix, e.g `/` is a prefix match of `/products`, and `/products` is prefix match of `/products/:id`. Then we have a setup that looks something like this: +- `/` +- `/products` +- `/products/:id` + +Now, receiving this deep-link `/products/123` will add all above routes to the stack. This of course requires `includePrefixMatches` to be true in the root config (default is `!kWeb`) or when using `pushNamed`, `navigateNamed` and `replaceNamed`. + +**Things to keep in mind**: + +- If a full match can not finally be found, no prefix matches will be included. +- Paths that require a full path match => `AutoRoute(path:'path', fullMatch: true)` will not be + included as prefix matches. +- In the above example, if `/products/:id` comes before `/products`, `/products` will not be + included. + +## Declarative Navigation + +To use declarative navigation with auto_route, you simply use the `AutoRouter.declarative` constructor and return a list of routes based on state. + +```dart +AutoRouter.declarative( + routes: (handler) => [ + BookListRoute(), + if(_selectedBook != null) { + BookDetailsRoute(id: _selectedBook.id), + } + ], +); +``` + +**Note:** The handler contains a temp-list of pending initial routes which can be read only once. + +## Working with Paths + +Working with paths in **AutoRoute** is optional because `PageRouteInfo` objects are matched by name unless pushed as a string using the `deepLinkBuilder` property in root delegate or `pushNamed`, `replaceNamed` `navigateNamed` methods. + +If you don’t specify a path it’s going to be generated from the page name e.g. `BookListPage` will have ‘book-list-page’ as a path, if initial arg is set to true the path will be `/`, unless it's relative then it will be an empty string `''`. + +When developing a web application or a native app that requires deep-linking, you'd probably need to define paths with clear memorable names, and that's done using the `path` argument in `AutoRoute`. + +```dart +AutoRoute(path: '/books', page: BookListPage), +``` + +### Path Parameters (dynamic segments) + +You can define a dynamic segment by prefixing it with a colon + +```dart +AutoRoute(path: '/books/:id', page: BookDetailsPage), +``` + +The simplest way to extract path parameters from path and gain access to them is by annotating constructor params with `@PathParam('optional-alias')` with the same alias/name of the segment. + +```dart +class BookDetailsPage extends StatelessWidget { + const BookDetailsPage({@PathParam('id') this.bookId}); + + final int bookId; + ... +} +``` + +Now writing `/books/1` in the browser will navigate you to `BookDetailsPage` and automatically extract the `bookId` argument from path and inject it to your widget. + +#### Inherited Path Parameters + +To inherit a path-parameter from a parent route's path, we need to use `@PathParam.inherit` annotation in the child route's constructor. Let's say we have the following setup: + +```dart +AutoRoute( + path: '/product/:id', + page: ProductRoute.page, + children: [ + AutoRoute(path: 'review',page: ProductReviewRoute.page), + ], +) +``` + +Now `ProductReviewScreen` expects a path-param named `id` but, from the above snippet we know that the path corresponding with it. `review` has no path parameters, but we can inherit 'id' from the parent `/product/:id` like follows: + +```dart +@RoutePage() +class ProductReviewScreen extends StatelessWidget { + // the path-param 'id' will be inherited and it can not be passed + // as a route arg by user + const ProductReviewScreen({super.key, @PathParam.inherit('id') required String id}); +} +``` + +### Query Parameters + +Query parameters are accessed the same way, simply annotate the constructor parameter to hold the value of the query param with `@QueryParam('optional-alias')` and let **AutoRoute** do the rest. + +You could also access path/query parameters using the scoped `RouteData` object. + +```dart +RouteData.of(context).pathParams; +// or using the extension +context.routeData.queryParams; +``` + +`Tip`: if your parameter name is the same as the path/query parameter, you could use the const `@pathParam` or `@queryParam` and not pass a slug/alias. + +```dart +@RoutePage() +class BookDetailsPage extends StatelessWidget { + const BookDetailsPage({@pathParam this.id}); + + final int id; + ... +} +``` + +### Redirecting Paths + +Paths can be redirected using `RedirectRoute`. The following setup will navigate us to `/books` when `/` is matched. + +```dart + [ + RedirectRoute(path: '/', redirectTo: '/books'), + AutoRoute(path: '/books', page: BookListRoute.page), +] +``` + +When redirecting initial routes the above setup can be simplified by setting the `/books` path as initial and **AutoRoute** will automatically generate the required redirect code for you. + +```dart + [ + AutoRoute(path: '/books', page: BookListRoute.page, initial: true), +] +``` + +You can also redirect paths with params like follows: + +```dart + [ + RedirectRoute(path: 'books/:id', redirectTo: '/books/:id/details'), + AutoRoute(path: '/books/:id/details', page: BookDetailsRoute.page), +] +``` + +**Note**: `RedirectRoutes` are fully matched. + +### Wildcards + +**AutoRoute** supports wildcard matching to handle invalid or undefined paths. + +```dart +AutoRoute( + path: '*', + page: UnknownRoute.page, +) +// it could be used with defined prefixes +AutoRoute( + path: '/profile/*', + page: ProfileRoute.page, +) +// or it could be used with RedirectRoute +RedirectRoute( + path: '*', + redirectTo: '/', +) +``` + +**Note:** Be sure to always add your wildcards at the end of your route list because routes are matched in order. + +## Route Guards + +Think of route guards as middleware or interceptors, routes can not be added to the stack without going through their assigned guards. Guards are useful for restricting access to certain routes. + +We create a route guard by extending `AutoRouteGuard` from the **AutoRoute** package and implementing our logic inside of the onNavigation method. + +```dart +class AuthGuard extends AutoRouteGuard { + + @override + void onNavigation(NavigationResolver resolver, StackRouter router) { + // the navigation is paused until resolver.next() is called with either + // true to resume/continue navigation or false to abort navigation + if(authenticated) { + // if user is authenticated we continue + resolver.next(true); + } else { + // we redirect the user to our login page + // tip: use resolver.redirect to have the redirected route + // automatically removed from the stack when the resolver is completed + resolver.redirect( + LoginRoute(onResult: (success) { + // if success == true the navigation will be resumed + // else it will be aborted + resolver.next(success); + }, + ); + ); + } + } +} +``` + +**Important**: `resolver.next()` should only be called once. + +The `NavigationResolver` object contains the guarded route which you can access by calling the property `resolver.route` and a list of pending routes (if there are any) accessed by calling `resolver.pendingRoutes`. + +Now we assign our guard to the routes we want to protect. + +```dart +AutoRoute( + page: ProfileRoute.page, + guards: [AuthGuard()], +); +``` + +#### Guarding all stack-routes + +You can have all your stack-routes (non-tab-routes) go through a global guard by having your router implement an AutoRouteGuard. Lets say you have an app with no publish screens, we'd have a global guard that only allows navigation if the user is authenticated or if we're navigating to the LoginRoute. + +```dart +@AutoRouterConfig() +class AppRouter extends $AppRouter implements AutoRouteGuard { + + @override + void onNavigation(NavigationResolver resolver, StackRouter router) { + if(isAuthenticated || resolver.route.name == LoginRoute.name) { + // we continue navigation + resolver.next(); + } else { + // else we navigate to the Login page so we get authenticated + + // tip: use resolver.redirect to have the redirected route + // automatically removed from the stack when the resolver is completed + resolver.redirect(LoginRoute(onResult: (didLogin) => resolver.next(didLogin))); + } + } + // ..routes[] +} +``` + + + +### Using a Reevaluate Listenable + +Route guards can prevent users from accessing private pages until they're logged in for example, but auth state may change when the user is already navigated to the private page, to make sure private pages are only accessed by logged-in users all the time, we need a listenable that tells the router that the auth state has changed and you need to re-evaluate your stack. + +The following auth provider mock will act as our re-valuate listenable + +```dart +class AuthProvider extends ChangeNotifier { + bool _isLoggedIn = false; + + bool get isLoggedIn => _isLoggedIn; + + void login() { + _isLoggedIn = true; + notifyListeners(); + } + + void logout() { + _isLoggedIn = false; + notifyListeners(); + } +} +``` + +We simply pass an instance of our `AuthProvider` to `reevaluateListenable` inside of `router.config` + +```dart +MaterialApp.router( + routerConfig: _appRouter.config( + reevaluateListenable: authProvider, + ), +); +``` + +Now, every time `AutoProvider` notifies listeners, the stack will be re-evaluated and `AutoRouteGuard.onNavigation()`. Methods will be re-called on all guards + +In the above example, we assigned our `AuthProvider` to `reevaluateListenable` directly, that's because `reevaluateListenable` takes a `Listenable` and AuthProvider extends `ChangeNotifier` which is a `Listenable`, if your auth provider is a stream you can use `reevaluateListenable: ReevaluateListenable.stream(YOUR-STREAM)` + +**Note**: When the Stack is re-evaluated, the whole existing hierarchy will be re-pushed, so if you want to stop re-evaluating routes at some point, use `resolver.resolveNext()` which is like `resolver.next()` but with more options. + +```dart +@override +void onNavigation(NavigationResolver resolver, StackRouter router) async { + if (authProvider.isAuthenticated) { + resolver.next(); + } else { + resolver.redirect( + WebLoginRoute( + onResult: (didLogin) { + // stop re-pushing any pending routes after current + resolver.resolveNext(didLogin, reevaluateNext: false); + }, + ), + ); + } +} +``` + +## Wrapping Routes + +In some cases we want to wrap our screen with a parent widget, usually to provide some values through context, e.g wrapping your route with a custom `Theme` or a `Provider`. To do that, simply implement `AutoRouteWrapper`, and have wrappedRoute(context) method return (this) as the child of your wrapper widget. + +```dart +@RoutePage() +class ProductsScreen extends StatelessWidget implements AutoRouteWrapper { + + @override + Widget wrappedRoute(BuildContext context) { + return Provider(create: (ctx) => ProductsBloc(), child: this); + } + ... +} +``` + + + +## Navigation Observers + +Navigation observers are used to observe when routes are pushed ,replaced or popped ..etc. + +We implement an AutoRouter observer by extending an `AutoRouterObserver` which is just a `NavigatorObserver` with tab route support. + +```dart +class MyObserver extends AutoRouterObserver { + + @override + void didPush(Route route, Route? previousRoute) { + print('New route pushed: ${route.settings.name}'); + } + + // only override to observer tab routes + @override + void didInitTabRoute(TabPageRoute route, TabPageRoute? previousRoute) { + print('Tab route visited: ${route.name}'); + } + + @override + void didChangeTabRoute(TabPageRoute route, TabPageRoute previousRoute) { + print('Tab route re-visited: ${route.name}'); + } +} +``` + +Then we pass our observer to the `.config().` **Important:** Notice that `navigatorObservers` property is a builder function that returns a list of observers and the reason for that is a navigator observer instance can only be used by a single router, so unless you're using a single router or you don't want your nested routers to inherit observers, make sure navigatorObservers builder always returns fresh observer instances. + +```dart +return MaterialApp.router( + routerConfig: _appRouter.config( + navigatorObservers: () => [MyObserver()], + ), +); +``` + +The following approach **won't** work if you have nested routers unless they don't inherit the observers. + +```dart +final _observer = MyObserver(); +return MaterialApp.router( + routerConfig: _appRouter.config( + // this should always return new instances + navigatorObservers: () => [_observer], + ), +); +``` + +Every nested router can have it's own observers and inherit it's parent's. + +```dart +AutoRouter( + inheritNavigatorObservers: true, // true by default + navigatorObservers:() => [list of observers], +); + +AutoTabsRouter( + inheritNavigatorObservers: true, // true by default + navigatorObservers:() => [list of observers], +); +``` + +We can also make a certain screen **route** aware by subscribing to an `AutoRouteObserver` (route not router). + +First we provide our `AutoRouteObserver` instance + +```dart +return MaterialApp.router( + routerConfig: _appRouter.config( + navigatorObservers: () => [AutoRouteObserver()], + ), +); +``` + +Next, we use an `AutoRouteAware` mixin which is a `RouteAware` mixin with tab support to provide the needed listeners, then subscribe to our `AutoRouteObserver`. + +```dart +class BooksListPage extends State with AutoRouteAware { + AutoRouteObserver? _observer; + + @override + void didChangeDependencies() { + super.didChangeDependencies(); + // RouterScope exposes the list of provided observers + // including inherited observers + _observer = RouterScope.of(context).firstObserverOfType(); + if (_observer != null) { + // we subscribe to the observer by passing our + // AutoRouteAware state and the scoped routeData + _observer.subscribe(this, context.routeData); + } + } + + @override + void dispose() { + super.dispose(); + // don't forget to unsubscribe from the + // observer on dispose + _observer.unsubscribe(this); + } + + // only override if this is a tab page + @override + void didInitTabRoute(TabPageRoute? previousRoute) {} + + // only override if this is a tab page + @override + void didChangeTabRoute(TabPageRoute previousRoute) {} + + @override + void didPopNext() {} + + @override + void didPushNext() {} + + @override + void didPush() {} + + @override + void didPop() {} +} +``` + +#### AutoRouteAwareStateMixin + +The above code can be simplified using `AutoRouteAwareStateMixin` + +```dart +class BooksListPage extends State with AutoRouteAwareStateMixin { + // only override if this is a tab page + @override + void didInitTabRoute(TabPageRoute? previousRoute) {} + + // only override if this is a tab page + @override + void didChangeTabRoute(TabPageRoute previousRoute) {} + + // only override if this is a stack page + @override + void didPopNext() {} + + // only override if this is a stack page + @override + void didPushNext() {} +} +``` + +## Customizations + +##### MaterialAutoRouter | CupertinoAutoRouter | AdaptiveAutoRouter + +| Property | Default value | Definition | +|-----------------------------|-----------------------|-----------------------------------------------------------------------------------| +| replaceInRouteName [String] | Page|Screen,Route | Used to replace conventional words in generated route name (pattern, replacement) | + +## Custom Route Transitions + +To use custom route transitions use a `CustomRoute` and pass in your preferences. The `TransitionsBuilder` function needs to be passed as a static/const reference that has the same signature as the `TransitionsBuilder` function of the `PageRouteBuilder` class. + +```dart +CustomRoute( + page: LoginRoute.page, + // TransitionsBuilders class contains a preset of common transitions builders. + transitionsBuilder: TransitionsBuilders.slideBottom, + durationInMilliseconds: 400, +) +``` + +`Tip:` Override `defaultRouteType` in generated router to define global custom route transitions. + +You can of course use your own transitionsBuilder function, as long as it has the same function signature. The function has to take in exactly one `BuildContext`, `Animation`, `Animation` and a child `Widget` and it needs to return a `Widget`. Typically, you would wrap your child with one of Flutter's transition widgets as follows: + +```dart +CustomRoute( + page: ZoomInScreen, + transitionsBuilder: + (BuildContext context, Animation animation, Animation secondaryAnimation, Widget child) { + // you get an animation object and a widget + // make your own transition + return ScaleTransition(scale: animation, child: child); + }, +) +``` + +## Custom Route Builder + +You can use your own custom route by passing a `CustomRouteBuilder` function to `CustomRoute' and implement the builder function the same way we did with the TransitionsBuilder function, the most important part here is passing the page argument to our custom route. + +```dart +CustomRoute( + page: CustomPage, + customRouteBuilder: (BuildContext context, Widget child, CustomPage page) { + return PageRouteBuilder( + fullscreenDialog: page.fullscreenDialog, + // this is important + settings: page, + pageBuilder: (_,__,___) => child, + ); + }, +) +``` + +## Others + +### Including Micro/External Packages + +To include routes inside of a depended-on package, that package needs to generate an `AutoRouterModule` that will be later consumed by the root router. + +To have a package output an `AutoRouterModule` instead of a `RootStackRouter`, we need to use the `AutoRouterConfig.module()` annotation like follows + +```dart +@AutoRouterConfig.module() +class MyPackageModule extends $MyPackageModule {} +``` + +Then when setting up our root router we need to tell it to include the generated module. + +```dart +@AutoRouterConfig(modules: [MyPackageModule]) +class AppRouter extends $AppRouter {} +``` + +Now you can use `PageRouteInfos` generated inside `MyPackageModule`. + +`Tip:` You can add export `MyPackageModule` to `app_router.dart`, so you only import `app_router.dart` inside of your code. + +```dart +// ...imports +export 'package:my_package/my_package_module.dart' +@AutoRouterConfig(modules: [MyPackageModule]) +class AppRouter extends $AppRouter {} +``` + +## Configuring builders +To pass builder configuration to `auto_route_generator` we need to add `build.yaml` file next to `pubspec.yaml` if not already added. + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_route_generator: + # configs for @RoutePage() generator ... + auto_route_generator:auto_router_generator: + # configs for @AutoRouterConfig() generator ... +``` + +### Passing custom ignore_for_file rules +You can pass custom ignore_for_file rules to the generated router by adding the following: + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_router_generator: + options: + ignore_fore_file: + - custom_rule_1 + - custom_rule_2 +``` + +### Optimizing generation time +The first thing you want to do to reduce generation time, is specifying the files build_runner should process and we do that by using [globs](https://pub.dev/packages/glob). Globs are kind of regex patterns with little differences that's used to match file names. **Note:** for this to work on file level you need to follow a naming convention + +``` +let's say we have the following files tree +├── lib +│ ├── none_widget_file.dart +│ ├── none_widget_file2.dart +│ └── ui +│ ├── products_screen.dart +│ ├── products_details_screen.dart +``` + +By default, the builder will process all of these files to check for a page with `@RoutePage()` +annotation, we can help by letting it know what files we need processed, e.g only process the files +inside the ui folder: +**Note** (**) matches everything including '/'; + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_route_generator: + generate_for: + - lib/ui/**.dart +``` + +Let's say you have widget files inside of the ui folder, but we only need to process files ending with `_screen.dart` + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_route_generator: + generate_for: + - lib/ui/**_screen.dart +``` + +Now only `products_screen.dart`, `products_details_screen.dart` will be processed + +The same goes for `@AutoRouterConfig` builder + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_route_generator: # this for @RoutePage + generate_for: + - lib/ui/**_screen.dart + auto_route_generator:auto_router_generator: # this for @AutoRouterConfig + generate_for: + - lib/ui/router.dart +``` + +## Enabling cached builds + +**This is still experimental** +When cached builds are enabled, **AutoRoute** will try to prevent redundant re-builds by analyzing whether the file changes has any effect on the extracted route info, e.g any changes inside of the build method should be ignored. + +**Note** Enable cached builds on both generators + +```yaml +targets: + $default: + builders: + auto_route_generator:auto_route_generator: # this for @RoutePage + options: + enable_cached_builds: true + generate_for: + - lib/ui/**_screen.dart + auto_route_generator:auto_router_generator: # this for @AutoRouterConfig + options: + enable_cached_builds: true + generate_for: + - lib/ui/router.dart +``` + +### AutoLeadingButton-BackButton + +`AutoLeadingButton` is **AutoRoute**'s replacement to the default BackButton to handle nested or parent stack popping. To use it, simply assign it to the `leading` property inside of `AppBar` + +```dart +AppBar( + title: Text(context.topRoute.name), + leading: AutoLeadingButton(), +) +``` + +### ActiveGuardObserver + +`ActiveGuardObserver` can notify you when a guard is being checked and what guard it is. This can be used to implement a loading indicator for example. + +```dart +var isLoading = false; +void initState(){ + final guardObserver = context.router.activeGuardObserver; + + guardObserver.addListener(() { + setState((){ + isLoading = guardObserver.guardInProgress; + }); + }); +} +``` + + +## Migrating to v6 + +In version 6.0 **AutoRoute** aims for less generated code for more flexibility and less generation time. + +#### 1. Instead of using `MaterialAutoRouter`, `CupertinoAutoRouter`, etc, we now only have one annotation for our router which is `@AutoRouterConfig()` and instead of passing our routes list to the annotation we now pass it to the overridable getter `routes` inside of the generated router class and for the default route type you can override `defaultRouteType` + +#### Before + +```dart +// @CupertinoAutoRouter +// @AdaptiveAutoRouter +// @CustomAutoRouter +@MaterialAutoRouter( + routes: [ + // routes go here + ], +) +class $AppRouter {} +``` + +#### After + + ```dart +@AutoRouterConfig() +class AppRouter extends $AppRouter { + + @override + RouteType get defaultRouteType => RouteType.material(); //.cupertino, .adaptive ..etc + + @override + List get routes => [ + // routes go here + ]; +} +``` + +#### 2. Passing page components as types is changed, now you'd annotate the target page with `@RoutePage()` annotation and pass the generated `result.page` to AutoRoute(): + +#### Before + +```dart +class ProductDetailsPage extends StatelessWidget {} +``` + +```dart +AutoRoute(page: ProductDetailsPage) // as Type +``` + +#### After + +```dart +@RoutePage() // Add this annotation to your routable pages +class ProductDetailsPage extends StatelessWidget {} +``` + +```dart +AutoRoute(page: ProductDetailsRoute.page) // ProductDetailsRoute is generated +``` + +#### 3. `EmptyRoutePage` no longer exists, instead you will now make your own empty pages by extending the `AutoRouter` widget + +#### Before + +```dart +AutoRoute(page: EmptyRoutePage, name: 'ProductsRouter') // as Type +``` + +#### After + +```dart +@RoutePage(name: 'ProductsRouter') +class ProductsRouterPage extends AutoRouter {} +``` + +```dart +AutoRoute(page: ProductsRouter.page) +``` + +#### 4. Passing route guards is also changed now, instead of passing guards as types you now pass instances. + +#### Before + +```dart +AutoRoute(page: ProfilePage, guards:[AuthGuard]) // as Type +``` + +#### After + +```dart +AutoRoute(page: ProfilePage, guards:[AuthGuard()]) // as Instance +``` + +## Examples + +coming soon + +### Support auto_route + +You can support auto_route by liking it on Pub and staring it on Github, sharing ideas on how we can enhance a certain functionality or by reporting any problems you encounter and of course buying a couple coffees will help speed up the development process \ No newline at end of file