Provides best practices for Flutter navigation using GoRouter, including type-safe routes with @TypedGoRoute, hierarchical sub-route structure, and proper use of go() vs push().
How this skill is triggered — by the user, by Claude, or both
Slash command
/vgv-ai-flutter-plugin:navigationWhen to use
Use when creating, modifying, or reviewing routes, deep links, redirects, or navigation logic that uses package:go_router or package:go_router_builder.
sonnetThis skill is limited to the following tools:
The summary Claude sees in its skill listing — used to decide when to auto-load this skill
Routing and navigation best practices for Flutter applications using GoRouter, the Flutter team's recommended routing package built on the Navigator 2.0 API.
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 |
npx claudepluginhub verygoodopensource/very-good-claude-code-marketplace --plugin vgv-ai-flutter-pluginProvides production-ready Dart/Flutter patterns for null safety, state management (BLoC, Riverpod, Provider), GoRouter navigation, Dio networking, Freezed code generation, and testing.
Guides Flutter cross-platform app development: widget patterns, Riverpod/Bloc state management, GoRouter navigation, const/performance optimization, responsive layouts, testing, DevTools.
Provides expert Flutter/Dart patterns for cross-platform mobile apps including feature-first project structure, const widget best practices, and Riverpod/Bloc state management.