Enforces best practices for Flutter navigation and routing using GoRouter, including type-safe routes, hierarchical structures, and deep links. Use when creating, modifying, or reviewing routes.
npx claudepluginhub verygoodopensource/very_good_claude_code_marketplace --plugin vgv-ai-flutter-pluginThis skill is limited to using the following tools:
Routing and navigation best practices for Flutter applications using GoRouter, the Flutter team's recommended routing package built on the Navigator 2.0 API.
Configures go_router for declarative routing in Flutter apps, enabling deep linking, nested navigation, ShellRoutes, and clean web URLs via path strategy.
Build cross-platform iOS/Android apps with Flutter and Dart. Covers widgets, state management (Provider/BLoC), navigation, API integration, and Material Design best practices.
Guides Flutter cross-platform app development: widget patterns, Riverpod/Bloc state management, GoRouter navigation, const/performance optimization, responsive layouts, testing, DevTools.
Share bugs, ideas, or general feedback.
Routing and navigation best practices for Flutter applications using GoRouter, the Flutter team's recommended routing package built on the Navigator 2.0 API.
Apply these standards to ALL navigation work:
package:go_router for all navigation — never raw Navigator 2.0 or Navigator 1.0 push/pop@TypedGoRoute annotations for type-safe routes — never raw string paths in route definitionsgo() over push() — use push() only when expecting return data from the destinationextra parameter — it breaks deep linking and does not work on the webBuildContext extensions for navigation — prefer context.goNamed() over GoRouter.of(context).goNamed()Structure routes hierarchically with logical parent-child relationships. Sub-routes ensure the app bar back button displays correctly and URLs remain clean.
/flutter
/flutter/news
/flutter/chat
/flutter/articles
/flutter/articles?category=all
/flutter/article/:id
/android
/android/news
/android/chat
/flutter-news
/flutter-chat
/android-news
/android-chat
Hierarchical sub-routes produce proper backward navigation automatically — when a user is on /flutter/news, the back button navigates to /flutter.
Use @TypedGoRoute annotations with GoRouteData classes to eliminate typos and manual parameter casting. The go_router_builder package generates type-safe route helpers at build time.
@TypedGoRoute<CategoriesPageRoute>(
name: 'categories',
path: '/categories',
)
@immutable
class CategoriesPageRoute extends GoRouteData {
const CategoriesPageRoute({
this.size,
this.color,
});
final String? size;
final String? color;
@override
Widget build(BuildContext context, GoRouterState state) {
return CategoriesPage(size: size, color: color);
}
}
@TypedGoRoute<FlutterPageRoute>(
name: 'flutter',
path: '/flutter',
routes: [
TypedGoRoute<FlutterNewsPageRoute>(
name: 'flutterNews',
path: 'news',
),
TypedGoRoute<FlutterArticlesPageRoute>(
name: 'flutterArticles',
path: 'articles',
routes: [
TypedGoRoute<FlutterArticlePageRoute>(
name: 'flutterArticle',
path: 'article/:id',
),
],
),
],
)
@immutable
class FlutterPageRoute extends GoRouteData {
const FlutterPageRoute();
@override
Widget build(BuildContext context, GoRouterState state) {
return const FlutterPage();
}
}
go() vs push()| Method | URL Updates | Back Button | Use Case |
|---|---|---|---|
go() | Yes | App bar | Standard navigation between screens |
push() | No | System | When expecting return data from popped route |
go() (Default)const CategoriesPageRoute(size: 'small', color: 'blue').go(context);
Using go() ensures the back button in the app's AppBar displays when the current route has a parent to navigate back to.
push() (Return Data Only)final result = await DialogPageRoute().push<String>(context);
Use push() only when a route must return data (e.g., a dialog collecting user input). On the web, push() does not update the address bar.
Always use extension methods for cleaner syntax:
// Preferred
context.goNamed('flutterNews');
// Avoid
GoRouter.of(context).goNamed('flutterNews');
Use path parameters (:id) for resource identification and query parameters (?category=all) for optional filtering. Never use extra — it breaks deep linking and does not work on the web.
See references/parameters.md for full examples of path parameters, query parameters, and why extra is prohibited.
Redirects can be applied at the root router level (e.g., authentication guards) and at individual route levels (e.g., authorization guards). Parent redirects execute before child redirects.
See references/redirects.md for root-level and route-level redirect examples.
Mock GoRouter with package:mocktail and wrap widgets in InheritedGoRouter for widget tests. Test redirects by constructing a GoRouter with the target redirect logic and asserting the resulting page.
See references/testing.md for mocking GoRouter and testing redirect examples.
GoRouteData class with @TypedGoRoute annotationdart run build_runner build --delete-conflicting-outputs to regenerate route helpersextra@TypedShellRoute<AppShellRoute>(
routes: [
TypedGoRoute<HomePageRoute>(
name: 'home',
path: '/home',
),
TypedGoRoute<SettingsPageRoute>(
name: 'settings',
path: '/settings',
),
],
)
class AppShellRoute extends ShellRouteData {
@override
Widget builder(BuildContext context, GoRouterState state, Widget navigator) {
return AppShell(child: navigator);
}
}
| Package | Purpose |
|---|---|
go_router | Declarative routing built on Navigator 2.0 |
go_router_builder | Code generation for type-safe route classes |
| Command | Purpose |
|---|---|
dart run build_runner build --delete-conflicting-outputs | Generate type-safe route helpers |
dart run build_runner watch --delete-conflicting-outputs | Watch and regenerate on changes |