mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-05-15 03:04:43 +08:00
566 lines
15 KiB
Markdown
566 lines
15 KiB
Markdown
---
|
||
name: dart-flutter-patterns
|
||
description: 生产就绪的 Dart 和 Flutter 模式,涵盖空安全、不可变状态、异步组合、Widget 架构、流行的状态管理框架(BLoC、Riverpod、Provider)、GoRouter 导航、Dio 网络请求、Freezed 代码生成和整洁架构。
|
||
origin: ECC
|
||
---
|
||
|
||
# Dart/Flutter 模式
|
||
|
||
## 使用场景
|
||
|
||
在以下情况使用此技能:
|
||
|
||
* 开始新的 Flutter 功能,需要状态管理、导航或数据访问的惯用模式
|
||
* 审查或编写 Dart 代码,需要空安全、密封类型或异步组合的指导
|
||
* 搭建新的 Flutter 项目,在 BLoC、Riverpod 或 Provider 之间做选择
|
||
* 实现安全的 HTTP 客户端、WebView 集成或本地存储
|
||
* 为 Flutter 组件、Cubit 或 Riverpod 提供者编写测试
|
||
* 使用认证守卫配置 GoRouter
|
||
|
||
## 工作原理
|
||
|
||
此技能提供按关注点组织的、可直接复制粘贴的 Dart/Flutter 代码模式:
|
||
|
||
1. **空安全** — 避免 `!`,优先使用 `?.`/`??`/模式匹配
|
||
2. **不可变状态** — 密封类、`freezed`、`copyWith`
|
||
3. **异步组合** — 并发 `Future.wait`、`BuildContext` 后安全使用 `await`
|
||
4. **组件架构** — 提取为类(而非方法)、`const` 传播、作用域重建
|
||
5. **状态管理** — BLoC/Cubit 事件、Riverpod 通知器和派生提供者
|
||
6. **导航** — 通过 `refreshListenable` 实现带响应式认证守卫的 GoRouter
|
||
7. **网络请求** — 带拦截器的 Dio、带一次性重试守卫的令牌刷新
|
||
8. **错误处理** — 全局捕获、`ErrorWidget.builder`、Crashlytics 集成
|
||
9. **测试** — 单元测试(BLoC 测试)、组件测试(ProviderScope 覆盖)、使用假对象而非模拟对象
|
||
|
||
## 示例
|
||
|
||
```dart
|
||
// Sealed state — prevents impossible states
|
||
sealed class AsyncState<T> {}
|
||
final class Loading<T> extends AsyncState<T> {}
|
||
final class Success<T> extends AsyncState<T> { final T data; const Success(this.data); }
|
||
final class Failure<T> extends AsyncState<T> { final Object error; const Failure(this.error); }
|
||
|
||
// GoRouter with reactive auth redirect
|
||
final router = GoRouter(
|
||
refreshListenable: GoRouterRefreshStream(authCubit.stream),
|
||
redirect: (context, state) {
|
||
final authed = context.read<AuthCubit>().state is AuthAuthenticated;
|
||
if (!authed && !state.matchedLocation.startsWith('/login')) return '/login';
|
||
return null;
|
||
},
|
||
routes: [...],
|
||
);
|
||
|
||
// Riverpod derived provider with safe firstWhereOrNull
|
||
@riverpod
|
||
double cartTotal(Ref ref) {
|
||
final cart = ref.watch(cartNotifierProvider);
|
||
final products = ref.watch(productsProvider).valueOrNull ?? [];
|
||
return cart.fold(0.0, (total, item) {
|
||
final product = products.firstWhereOrNull((p) => p.id == item.productId);
|
||
return total + (product?.price ?? 0) * item.quantity;
|
||
});
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
适用于 Dart 和 Flutter 应用程序的实用、生产就绪模式。尽可能保持库无关性,并明确覆盖最常见的生态系统包。
|
||
|
||
***
|
||
|
||
## 1. 空安全基础
|
||
|
||
### 优先使用模式而非感叹号操作符
|
||
|
||
```dart
|
||
// BAD — crashes at runtime if null
|
||
final name = user!.name;
|
||
|
||
// GOOD — provide fallback
|
||
final name = user?.name ?? 'Unknown';
|
||
|
||
// GOOD — Dart 3 pattern matching (preferred for complex cases)
|
||
final display = switch (user) {
|
||
User(:final name, :final email) => '$name <$email>',
|
||
null => 'Guest',
|
||
};
|
||
|
||
// GOOD — guard early return
|
||
String getUserName(User? user) {
|
||
if (user == null) return 'Unknown';
|
||
return user.name; // promoted to non-null after check
|
||
}
|
||
```
|
||
|
||
### 避免过度使用 `late`
|
||
|
||
```dart
|
||
// BAD — defers null error to runtime
|
||
late String userId;
|
||
|
||
// GOOD — nullable with explicit initialization
|
||
String? userId;
|
||
|
||
// OK — use late only when initialization is guaranteed before first access
|
||
// (e.g., in initState() before any widget interaction)
|
||
late final AnimationController _controller;
|
||
|
||
@override
|
||
void initState() {
|
||
super.initState();
|
||
_controller = AnimationController(vsync: this, duration: const Duration(milliseconds: 300));
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 2. 不可变状态
|
||
|
||
### 状态层次结构的密封类
|
||
|
||
```dart
|
||
sealed class UserState {}
|
||
|
||
final class UserInitial extends UserState {}
|
||
|
||
final class UserLoading extends UserState {}
|
||
|
||
final class UserLoaded extends UserState {
|
||
const UserLoaded(this.user);
|
||
final User user;
|
||
}
|
||
|
||
final class UserError extends UserState {
|
||
const UserError(this.message);
|
||
final String message;
|
||
}
|
||
|
||
// Exhaustive switch — compiler enforces all branches
|
||
Widget buildFrom(UserState state) => switch (state) {
|
||
UserInitial() => const SizedBox.shrink(),
|
||
UserLoading() => const CircularProgressIndicator(),
|
||
UserLoaded(:final user) => UserCard(user: user),
|
||
UserError(:final message) => ErrorText(message),
|
||
};
|
||
```
|
||
|
||
### 使用 Freezed 实现无模板代码的不可变性
|
||
|
||
```dart
|
||
import 'package:freezed_annotation/freezed_annotation.dart';
|
||
|
||
part 'user.freezed.dart';
|
||
part 'user.g.dart';
|
||
|
||
@freezed
|
||
class User with _$User {
|
||
const factory User({
|
||
required String id,
|
||
required String name,
|
||
required String email,
|
||
@Default(false) bool isAdmin,
|
||
}) = _User;
|
||
|
||
factory User.fromJson(Map<String, dynamic> json) => _$UserFromJson(json);
|
||
}
|
||
|
||
// Usage
|
||
final user = User(id: '1', name: 'Alice', email: 'alice@example.com');
|
||
final updated = user.copyWith(name: 'Alice Smith'); // immutable update
|
||
final json = user.toJson();
|
||
final fromJson = User.fromJson(json);
|
||
```
|
||
|
||
***
|
||
|
||
## 3. 异步组合
|
||
|
||
### 使用 Future.wait 的结构化并发
|
||
|
||
```dart
|
||
Future<DashboardData> loadDashboard(UserRepository users, OrderRepository orders) async {
|
||
// Run concurrently — don't await sequentially
|
||
final (userList, orderList) = await (
|
||
users.getAll(),
|
||
orders.getRecent(),
|
||
).wait; // Dart 3 record destructuring + Future.wait extension
|
||
|
||
return DashboardData(users: userList, orders: orderList);
|
||
}
|
||
```
|
||
|
||
### 流模式
|
||
|
||
```dart
|
||
// Repository exposes reactive streams for live data
|
||
Stream<List<Item>> watchCartItems() => _db
|
||
.watchTable('cart_items')
|
||
.map((rows) => rows.map(Item.fromRow).toList());
|
||
|
||
// In widget layer — declarative, no manual subscription
|
||
StreamBuilder<List<Item>>(
|
||
stream: cartRepository.watchCartItems(),
|
||
builder: (context, snapshot) => switch (snapshot) {
|
||
AsyncSnapshot(connectionState: ConnectionState.waiting) =>
|
||
const CircularProgressIndicator(),
|
||
AsyncSnapshot(:final error?) => ErrorWidget(error.toString()),
|
||
AsyncSnapshot(:final data?) => CartList(items: data),
|
||
_ => const SizedBox.shrink(),
|
||
},
|
||
)
|
||
```
|
||
|
||
### Await 后的 BuildContext
|
||
|
||
```dart
|
||
// CRITICAL — always check mounted after any await in StatefulWidget
|
||
Future<void> _handleSubmit() async {
|
||
setState(() => _isLoading = true);
|
||
try {
|
||
await authService.login(_email, _password);
|
||
if (!mounted) return; // ← guard before using context
|
||
context.go('/home');
|
||
} on AuthException catch (e) {
|
||
if (!mounted) return;
|
||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(e.message)));
|
||
} finally {
|
||
if (mounted) setState(() => _isLoading = false);
|
||
}
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 4. 组件架构
|
||
|
||
### 提取为类,而非方法
|
||
|
||
```dart
|
||
// BAD — private method returning widget, prevents optimization
|
||
Widget _buildHeader() {
|
||
return Container(
|
||
padding: const EdgeInsets.all(16),
|
||
child: Text(title, style: Theme.of(context).textTheme.headlineMedium),
|
||
);
|
||
}
|
||
|
||
// GOOD — separate widget class, enables const, element reuse
|
||
class _PageHeader extends StatelessWidget {
|
||
const _PageHeader(this.title);
|
||
final String title;
|
||
|
||
@override
|
||
Widget build(BuildContext context) {
|
||
return Container(
|
||
padding: const EdgeInsets.all(16),
|
||
child: Text(title, style: Theme.of(context).textTheme.headlineMedium),
|
||
);
|
||
}
|
||
}
|
||
```
|
||
|
||
### const 传播
|
||
|
||
```dart
|
||
// BAD — new instances every rebuild
|
||
child: Padding(
|
||
padding: EdgeInsets.all(16.0), // not const
|
||
child: Icon(Icons.home, size: 24.0), // not const
|
||
)
|
||
|
||
// GOOD — const stops rebuild propagation
|
||
child: const Padding(
|
||
padding: EdgeInsets.all(16.0),
|
||
child: Icon(Icons.home, size: 24.0),
|
||
)
|
||
```
|
||
|
||
### 作用域重建
|
||
|
||
```dart
|
||
// BAD — entire page rebuilds on every counter change
|
||
class CounterPage extends ConsumerWidget {
|
||
@override
|
||
Widget build(BuildContext context, WidgetRef ref) {
|
||
final count = ref.watch(counterProvider); // rebuilds everything
|
||
return Scaffold(
|
||
body: Column(children: [
|
||
const ExpensiveHeader(), // unnecessarily rebuilt
|
||
Text('$count'),
|
||
const ExpensiveFooter(), // unnecessarily rebuilt
|
||
]),
|
||
);
|
||
}
|
||
}
|
||
|
||
// GOOD — isolate the rebuilding part
|
||
class CounterPage extends StatelessWidget {
|
||
const CounterPage({super.key});
|
||
|
||
@override
|
||
Widget build(BuildContext context) {
|
||
return const Scaffold(
|
||
body: Column(children: [
|
||
ExpensiveHeader(), // never rebuilt (const)
|
||
_CounterDisplay(), // only this rebuilds
|
||
ExpensiveFooter(), // never rebuilt (const)
|
||
]),
|
||
);
|
||
}
|
||
}
|
||
|
||
class _CounterDisplay extends ConsumerWidget {
|
||
const _CounterDisplay();
|
||
|
||
@override
|
||
Widget build(BuildContext context, WidgetRef ref) {
|
||
final count = ref.watch(counterProvider);
|
||
return Text('$count');
|
||
}
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 5. 状态管理:BLoC/Cubit
|
||
|
||
```dart
|
||
// Cubit — synchronous or simple async state
|
||
class AuthCubit extends Cubit<AuthState> {
|
||
AuthCubit(this._authService) : super(const AuthState.initial());
|
||
final AuthService _authService;
|
||
|
||
Future<void> login(String email, String password) async {
|
||
emit(const AuthState.loading());
|
||
try {
|
||
final user = await _authService.login(email, password);
|
||
emit(AuthState.authenticated(user));
|
||
} on AuthException catch (e) {
|
||
emit(AuthState.error(e.message));
|
||
}
|
||
}
|
||
|
||
void logout() {
|
||
_authService.logout();
|
||
emit(const AuthState.initial());
|
||
}
|
||
}
|
||
|
||
// In widget
|
||
BlocBuilder<AuthCubit, AuthState>(
|
||
builder: (context, state) => switch (state) {
|
||
AuthInitial() => const LoginForm(),
|
||
AuthLoading() => const CircularProgressIndicator(),
|
||
AuthAuthenticated(:final user) => HomePage(user: user),
|
||
AuthError(:final message) => ErrorView(message: message),
|
||
},
|
||
)
|
||
```
|
||
|
||
***
|
||
|
||
## 6. 状态管理:Riverpod
|
||
|
||
```dart
|
||
// Auto-dispose async provider
|
||
@riverpod
|
||
Future<List<Product>> products(Ref ref) async {
|
||
final repo = ref.watch(productRepositoryProvider);
|
||
return repo.getAll();
|
||
}
|
||
|
||
// Notifier with complex mutations
|
||
@riverpod
|
||
class CartNotifier extends _$CartNotifier {
|
||
@override
|
||
List<CartItem> build() => [];
|
||
|
||
void add(Product product) {
|
||
final existing = state.where((i) => i.productId == product.id).firstOrNull;
|
||
if (existing != null) {
|
||
state = [
|
||
for (final item in state)
|
||
if (item.productId == product.id) item.copyWith(quantity: item.quantity + 1)
|
||
else item,
|
||
];
|
||
} else {
|
||
state = [...state, CartItem(productId: product.id, quantity: 1)];
|
||
}
|
||
}
|
||
|
||
void remove(String productId) =>
|
||
state = state.where((i) => i.productId != productId).toList();
|
||
|
||
void clear() => state = [];
|
||
}
|
||
|
||
// Derived provider (selector pattern)
|
||
@riverpod
|
||
int cartCount(Ref ref) => ref.watch(cartNotifierProvider).length;
|
||
|
||
@riverpod
|
||
double cartTotal(Ref ref) {
|
||
final cart = ref.watch(cartNotifierProvider);
|
||
final products = ref.watch(productsProvider).valueOrNull ?? [];
|
||
return cart.fold(0.0, (total, item) {
|
||
// firstWhereOrNull (from collection package) avoids StateError when product is missing
|
||
final product = products.firstWhereOrNull((p) => p.id == item.productId);
|
||
return total + (product?.price ?? 0) * item.quantity;
|
||
});
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 7. 使用 GoRouter 的导航
|
||
|
||
```dart
|
||
final router = GoRouter(
|
||
initialLocation: '/',
|
||
// refreshListenable re-evaluates redirect whenever auth state changes
|
||
refreshListenable: GoRouterRefreshStream(authCubit.stream),
|
||
redirect: (context, state) {
|
||
final isLoggedIn = context.read<AuthCubit>().state is AuthAuthenticated;
|
||
final isGoingToLogin = state.matchedLocation == '/login';
|
||
if (!isLoggedIn && !isGoingToLogin) return '/login';
|
||
if (isLoggedIn && isGoingToLogin) return '/';
|
||
return null;
|
||
},
|
||
routes: [
|
||
GoRoute(path: '/login', builder: (_, __) => const LoginPage()),
|
||
ShellRoute(
|
||
builder: (context, state, child) => AppShell(child: child),
|
||
routes: [
|
||
GoRoute(path: '/', builder: (_, __) => const HomePage()),
|
||
GoRoute(
|
||
path: '/products/:id',
|
||
builder: (context, state) =>
|
||
ProductDetailPage(id: state.pathParameters['id']!),
|
||
),
|
||
],
|
||
),
|
||
],
|
||
);
|
||
```
|
||
|
||
***
|
||
|
||
## 8. 使用 Dio 的 HTTP 请求
|
||
|
||
```dart
|
||
final dio = Dio(BaseOptions(
|
||
baseUrl: const String.fromEnvironment('API_URL'),
|
||
connectTimeout: const Duration(seconds: 10),
|
||
receiveTimeout: const Duration(seconds: 30),
|
||
headers: {'Content-Type': 'application/json'},
|
||
));
|
||
|
||
// Add auth interceptor
|
||
dio.interceptors.add(InterceptorsWrapper(
|
||
onRequest: (options, handler) async {
|
||
final token = await secureStorage.read(key: 'auth_token');
|
||
if (token != null) options.headers['Authorization'] = 'Bearer $token';
|
||
handler.next(options);
|
||
},
|
||
onError: (error, handler) async {
|
||
// Guard against infinite retry loops: only attempt refresh once per request
|
||
final isRetry = error.requestOptions.extra['_isRetry'] == true;
|
||
if (!isRetry && error.response?.statusCode == 401) {
|
||
final refreshed = await attemptTokenRefresh();
|
||
if (refreshed) {
|
||
error.requestOptions.extra['_isRetry'] = true;
|
||
return handler.resolve(await dio.fetch(error.requestOptions));
|
||
}
|
||
}
|
||
handler.next(error);
|
||
},
|
||
));
|
||
|
||
// Repository using Dio
|
||
class UserApiDataSource {
|
||
const UserApiDataSource(this._dio);
|
||
final Dio _dio;
|
||
|
||
Future<User> getById(String id) async {
|
||
final response = await _dio.get<Map<String, dynamic>>('/users/$id');
|
||
return User.fromJson(response.data!);
|
||
}
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 9. 错误处理架构
|
||
|
||
```dart
|
||
// Global error capture — set up in main()
|
||
void main() {
|
||
FlutterError.onError = (details) {
|
||
FlutterError.presentError(details);
|
||
crashlytics.recordFlutterFatalError(details);
|
||
};
|
||
|
||
PlatformDispatcher.instance.onError = (error, stack) {
|
||
crashlytics.recordError(error, stack, fatal: true);
|
||
return true;
|
||
};
|
||
|
||
runApp(const App());
|
||
}
|
||
|
||
// Custom ErrorWidget for production
|
||
class App extends StatelessWidget {
|
||
@override
|
||
Widget build(BuildContext context) {
|
||
ErrorWidget.builder = (details) => ProductionErrorWidget(details);
|
||
return MaterialApp.router(routerConfig: router);
|
||
}
|
||
}
|
||
```
|
||
|
||
***
|
||
|
||
## 10. 测试快速参考
|
||
|
||
```dart
|
||
// Unit test — use case
|
||
test('GetUserUseCase returns null for missing user', () async {
|
||
final repo = FakeUserRepository();
|
||
final useCase = GetUserUseCase(repo);
|
||
expect(await useCase('missing-id'), isNull);
|
||
});
|
||
|
||
// BLoC test
|
||
blocTest<AuthCubit, AuthState>(
|
||
'emits loading then error on failed login',
|
||
build: () => AuthCubit(FakeAuthService(throwsOn: 'login')),
|
||
act: (cubit) => cubit.login('user@test.com', 'wrong'),
|
||
expect: () => [const AuthState.loading(), isA<AuthError>()],
|
||
);
|
||
|
||
// Widget test
|
||
testWidgets('CartBadge shows item count', (tester) async {
|
||
await tester.pumpWidget(
|
||
ProviderScope(
|
||
overrides: [cartNotifierProvider.overrideWith(() => FakeCartNotifier(count: 3))],
|
||
child: const MaterialApp(home: CartBadge()),
|
||
),
|
||
);
|
||
expect(find.text('3'), findsOneWidget);
|
||
});
|
||
```
|
||
|
||
***
|
||
|
||
## 参考
|
||
|
||
* [Effective Dart: 设计](https://dart.dev/effective-dart/design)
|
||
* [Flutter 性能最佳实践](https://docs.flutter.dev/perf/best-practices)
|
||
* [Riverpod 文档](https://riverpod.dev/)
|
||
* [BLoC 库](https://bloclibrary.dev/)
|
||
* [GoRouter](https://pub.dev/packages/go_router)
|
||
* [Freezed](https://pub.dev/packages/freezed)
|
||
* 技能:`flutter-dart-code-review` — 全面审查清单
|
||
* 规则:`rules/dart/` — 编码风格、模式、安全性、测试、钩子
|