flutter-routing-and-navigation
在Flutter应用中实现健壮的导航和路由,评估应用需求以选择合适的路由策略。
npx skills add flutter/skills --skill flutter-routing-and-navigationBefore / After 效果对比
1 组1在Flutter应用中,手动管理路由栈、处理深层链接和在路由之间传递数据是一个复杂且容易出错的过程,尤其是在大型应用中。1Flutter路由与导航技能能够根据应用需求,智能选择并实现最佳的路由策略(如命令式Navigator或声明式Router),简化深层链接处理和数据传递,确保导航的健壮性和用户体验。description SKILL.md
flutter-routing-and-navigation
flutter-navigation-routing Goal Implements robust navigation and routing in Flutter applications. Evaluates application requirements to select the appropriate routing strategy (imperative Navigator, declarative Router, or nested navigation), handles deep linking, and manages data passing between routes while adhering to Flutter best practices. Instructions 1. Determine Routing Strategy (Decision Logic) Evaluate the application's navigation requirements using the following decision tree: Condition A: Does the app require complex deep linking, web URL synchronization, or advanced routing logic? Action: Use the declarative Router API (typically via a routing package like go_router). Condition B: Does the app require independent sub-flows (e.g., a multi-step setup wizard or persistent bottom navigation bars)? Action: Implement a Nested Navigator. Condition C: Is it a simple application with basic screen-to-screen transitions and no complex deep linking? Action: Use the imperative Navigator API (Navigator.push and Navigator.pop) with MaterialPageRoute or CupertinoPageRoute. Condition D: Are Named Routes requested? Action: Use MaterialApp.routes or onGenerateRoute, but note the limitations regarding deep link customization and web forward-button support. STOP AND ASK THE USER: "Based on your app's requirements, should we implement simple imperative navigation (Navigator.push), declarative routing (Router/go_router for deep links/web), or a nested navigation flow?" 2. Implement Basic Imperative Navigation If simple navigation is selected, use the Navigator widget to push and pop Route objects. Pushing a new route: Navigator.of(context).push( MaterialPageRoute( builder: (context) => const SecondScreen(), ), ); Popping a route: Navigator.of(context).pop(); 3. Implement Data Passing Between Screens Pass data to new screens using constructor arguments (preferred for imperative navigation) or RouteSettings (for named routes). Passing via Constructor: // Navigating and passing data Navigator.push( context, MaterialPageRoute( builder: (context) => DetailScreen(todo: currentTodo), ), ); // Receiving data class DetailScreen extends StatelessWidget { const DetailScreen({super.key, required this.todo}); final Todo todo; @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar(title: Text(todo.title)), body: Text(todo.description), ); } } Passing via RouteSettings (Named Routes): // Navigating and passing data Navigator.pushNamed( context, '/details', arguments: currentTodo, ); // Extracting data in the destination widget final todo = ModalRoute.of(context)!.settings.arguments as Todo; 4. Implement Named Routes (If Required) If named routes are explicitly required, configure MaterialApp with initialRoute and routes or onGenerateRoute. MaterialApp( title: 'Named Routes App', initialRoute: '/', routes: { '/': (context) => const FirstScreen(), '/second': (context) => const SecondScreen(), }, // OR use onGenerateRoute for dynamic argument extraction onGenerateRoute: (settings) { if (settings.name == '/details') { final args = settings.arguments as Todo; return MaterialPageRoute( builder: (context) => DetailScreen(todo: args), ); } assert(false, 'Need to implement ${settings.name}'); return null; }, ) 5. Implement Nested Navigation For sub-flows, instantiate a new Navigator widget within the widget tree. You MUST assign a GlobalKey to manage the nested stack. class SetupFlowState extends State { final _navigatorKey = GlobalKey(); void _onDiscoveryComplete() { _navigatorKey.currentState!.pushNamed('/select_device'); } @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar(title: const Text('Setup Flow')), body: Navigator( key: _navigatorKey, initialRoute: '/find_devices', onGenerateRoute: _onGenerateRoute, ), ); } Route _onGenerateRoute(RouteSettings settings) { Widget page; switch (settings.name) { case '/find_devices': page = WaitingPage(onWaitComplete: _onDiscoveryComplete); break; case '/select_device': page = const SelectDevicePage(); break; default: throw StateError('Unexpected route name: ${settings.name}!'); } return MaterialPageRoute(builder: (context) => page, settings: settings); } } 6. Validate and Fix Review the implemented routing logic to ensure stability: Verify that Navigator.pop() does not inadvertently close the application if the stack is empty (use Navigator.canPop(context) if necessary). If using initialRoute, verify that the home property is NOT defined in MaterialApp. If extracting arguments via ModalRoute, verify that null checks or type casts are safely handled. Constraints Do NOT use named routes (MaterialApp.routes) for applications requiring complex deep linking or web support; use the Router API instead. Do NOT define a home property in MaterialApp if an initialRoute is provided. You MUST use a GlobalKey when implementing a nested Navigator to ensure the correct navigation stack is targeted. Do NOT include external URLs or links in the generated code or comments. Always cast ModalRoute.of(context)!.settings.arguments to the specific expected type and handle potential nulls if the route can be accessed without arguments. Weekly Installs1.1KRepositoryflutter/skillsGitHub Stars685First Seen13 days agoSecurity AuditsGen Agent Trust HubPassSocketPassSnykPassInstalled oncodex1.0Kcursor1.0Kopencode1.0Kgemini-cli1.0Kgithub-copilot1.0Kkimi-cli1.0K
forum用户评价 (0)
发表评价
暂无评价,来写第一条吧
统计数据
用户评分
为此 Skill 评分