feat(apps/router): 重构路由系统使用 AppRoutes 常量

This commit is contained in:
zl-q
2026-03-19 00:51:45 +08:00
parent 00f37d7e19
commit 5def0358ba
4 changed files with 134 additions and 39 deletions
@@ -0,0 +1,4 @@
import 'package:flutter/widgets.dart';
final RouteObserver<ModalRoute<void>> appRouteObserver =
RouteObserver<ModalRoute<void>>();
+78 -39
View File
@@ -1,6 +1,8 @@
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'app_route_observer.dart';
import '../../features/auth/presentation/bloc/auth_bloc.dart'; import '../../features/auth/presentation/bloc/auth_bloc.dart';
import '../../features/auth/presentation/bloc/auth_state.dart'; import '../../features/auth/presentation/bloc/auth_state.dart';
import 'app_routes.dart';
import 'go_router_refresh_stream.dart'; import 'go_router_refresh_stream.dart';
import '../../features/auth/ui/screens/login_screen.dart'; import '../../features/auth/ui/screens/login_screen.dart';
import '../../features/auth/ui/screens/auth_boot_screen.dart'; import '../../features/auth/ui/screens/auth_boot_screen.dart';
@@ -15,9 +17,13 @@ import '../../features/contacts/ui/screens/add_contact_screen.dart';
import '../../features/calendar/ui/screens/calendar_dayweek_screen.dart'; import '../../features/calendar/ui/screens/calendar_dayweek_screen.dart';
import '../../features/calendar/ui/screens/calendar_month_screen.dart'; import '../../features/calendar/ui/screens/calendar_month_screen.dart';
import '../../features/calendar/ui/screens/calendar_event_detail_screen.dart'; import '../../features/calendar/ui/screens/calendar_event_detail_screen.dart';
import '../../features/calendar/ui/screens/calendar_event_create_screen.dart';
import '../../features/calendar/ui/screens/calendar_event_edit_screen.dart';
import '../../features/calendar/ui/screens/calendar_event_share_screen.dart';
import '../../features/calendar/ui/calendar_time_utils.dart'; import '../../features/calendar/ui/calendar_time_utils.dart';
import '../../features/todo/ui/screens/todo_quadrants_screen.dart'; import '../../features/todo/ui/screens/todo_quadrants_screen.dart';
import '../../features/todo/ui/screens/todo_detail_screen.dart'; import '../../features/todo/ui/screens/todo_detail_screen.dart';
import '../../features/todo/ui/screens/todo_edit_screen.dart';
import '../../features/settings/ui/screens/settings_screen.dart'; import '../../features/settings/ui/screens/settings_screen.dart';
import '../../features/settings/ui/screens/features_screen.dart'; import '../../features/settings/ui/screens/features_screen.dart';
import '../../features/settings/ui/screens/memory_screen.dart'; import '../../features/settings/ui/screens/memory_screen.dart';
@@ -26,34 +32,35 @@ import '../../features/settings/ui/screens/change_password_screen.dart';
import '../../features/settings/ui/screens/edit_profile_screen.dart'; import '../../features/settings/ui/screens/edit_profile_screen.dart';
final _protectedRoutes = [ final _protectedRoutes = [
'/home', AppRoutes.homeMain,
'/contacts', AppRoutes.contactsList,
'/contacts/add', AppRoutes.contactsAdd,
'/calendar/dayweek', AppRoutes.calendarDayWeek,
'/calendar/month', AppRoutes.calendarMonth,
'/calendar/events', '/calendar/events',
'/todo', AppRoutes.todoList,
'/settings', AppRoutes.settingsMain,
'/settings/features', AppRoutes.settingsFeatures,
'/settings/memory', AppRoutes.settingsMemory,
'/settings/account', AppRoutes.settingsAccount,
'/change-password', AppRoutes.settingsChangePassword,
'/edit-profile', AppRoutes.settingsEditProfile,
'/messages/invites', AppRoutes.messageInviteList,
]; ];
GoRouter createAppRouter(AuthBloc authBloc) { GoRouter createAppRouter(AuthBloc authBloc) {
return GoRouter( return GoRouter(
initialLocation: '/boot', initialLocation: AppRoutes.authBoot,
observers: [appRouteObserver],
refreshListenable: GoRouterRefreshStream(authBloc.stream), refreshListenable: GoRouterRefreshStream(authBloc.stream),
redirect: (context, state) { redirect: (context, state) {
final authState = authBloc.state; final authState = authBloc.state;
final isAuthenticated = authState is AuthAuthenticated; final isAuthenticated = authState is AuthAuthenticated;
final isAuthChecking = final isAuthChecking =
authState is AuthInitial || authState is AuthLoading; authState is AuthInitial || authState is AuthLoading;
final isBootRoute = state.matchedLocation == '/boot'; final isBootRoute = state.matchedLocation == AppRoutes.authBoot;
final isAuthRoute = final isAuthRoute =
state.matchedLocation == '/' || state.matchedLocation == AppRoutes.authLogin ||
state.matchedLocation.startsWith('/login') || state.matchedLocation.startsWith('/login') ||
state.matchedLocation.startsWith('/register'); state.matchedLocation.startsWith('/register');
final isProtected = _protectedRoutes.any( final isProtected = _protectedRoutes.any(
@@ -61,61 +68,84 @@ GoRouter createAppRouter(AuthBloc authBloc) {
); );
if (isAuthChecking && !isBootRoute) { if (isAuthChecking && !isBootRoute) {
return '/boot'; return AppRoutes.authBoot;
} }
if (!isAuthChecking && isBootRoute) { if (!isAuthChecking && isBootRoute) {
return isAuthenticated ? '/home' : '/'; return isAuthenticated ? AppRoutes.homeMain : AppRoutes.authLogin;
} }
if (!isAuthenticated && isProtected) { if (!isAuthenticated && isProtected) {
return '/'; return AppRoutes.authLogin;
} }
if (isAuthenticated && isAuthRoute) { if (isAuthenticated && isAuthRoute) {
return '/home'; return AppRoutes.homeMain;
} }
return null; return null;
}, },
routes: [ routes: [
GoRoute( GoRoute(
path: '/boot', path: AppRoutes.authBoot,
builder: (context, state) => const AuthBootScreen(), builder: (context, state) => const AuthBootScreen(),
), ),
GoRoute(path: '/', builder: (context, state) => const LoginScreen()), GoRoute(
path: AppRoutes.authLogin,
builder: (context, state) => const LoginScreen(),
),
GoRoute(
path: AppRoutes.calendarEventCreate,
builder: (context, state) => CalendarEventCreateScreen(
initialDate: parseYmd(state.uri.queryParameters['date']),
),
),
GoRoute( GoRoute(
path: '/calendar/events/:id', path: '/calendar/events/:id',
builder: (context, state) => builder: (context, state) =>
CalendarEventDetailScreen(eventId: state.pathParameters['id']!), CalendarEventDetailScreen(eventId: state.pathParameters['id']!),
), ),
GoRoute( GoRoute(
path: '/register', path: '/calendar/events/:id/edit',
builder: (context, state) =>
CalendarEventEditScreen(eventId: state.pathParameters['id']!),
),
GoRoute(
path: '/calendar/events/:id/share',
builder: (context, state) =>
CalendarEventShareScreen(eventId: state.pathParameters['id']!),
),
GoRoute(
path: AppRoutes.authRegister,
builder: (context, state) => const RegisterScreen(), builder: (context, state) => const RegisterScreen(),
), ),
GoRoute( GoRoute(
path: '/register/verification', path: AppRoutes.authRegisterVerification,
builder: (context, state) => const RegisterVerificationScreen(), builder: (context, state) => const RegisterVerificationScreen(),
), ),
GoRoute( GoRoute(
path: '/reset-password', path: AppRoutes.authResetPassword,
builder: (context, state) => const ResetPasswordScreen(), builder: (context, state) => const ResetPasswordScreen(),
), ),
GoRoute(path: '/home', builder: (context, state) => const HomeScreen()),
GoRoute( GoRoute(
path: '/messages/invites', path: AppRoutes.homeMain,
builder: (context, state) => const HomeScreen(),
),
GoRoute(
path: AppRoutes.messageInviteList,
builder: (context, state) => const MessageInviteListScreen(), builder: (context, state) => const MessageInviteListScreen(),
), ),
GoRoute( GoRoute(
path: '/messages/invites/:id', path: '/messages/invites/:id',
builder: (context, state) => const MessageInviteDetailScreen(), builder: (context, state) =>
MessageInviteDetailScreen(inviteId: state.pathParameters['id']!),
), ),
GoRoute( GoRoute(
path: '/contacts', path: AppRoutes.contactsList,
builder: (context, state) => const ContactsScreen(), builder: (context, state) => const ContactsScreen(),
), ),
GoRoute( GoRoute(
path: '/contacts/add', path: AppRoutes.contactsAdd,
builder: (context, state) => const AddContactScreen(), builder: (context, state) => const AddContactScreen(),
), ),
GoRoute( GoRoute(
path: '/calendar/dayweek', path: AppRoutes.calendarDayWeek,
builder: (context, state) { builder: (context, state) {
final fromHome = state.uri.queryParameters['from'] == 'home'; final fromHome = state.uri.queryParameters['from'] == 'home';
final initialDate = parseYmd(state.uri.queryParameters['date']); final initialDate = parseYmd(state.uri.queryParameters['date']);
@@ -126,43 +156,52 @@ GoRouter createAppRouter(AuthBloc authBloc) {
}, },
), ),
GoRoute( GoRoute(
path: '/calendar/month', path: AppRoutes.calendarMonth,
builder: (context, state) { builder: (context, state) {
final fromHome = state.uri.queryParameters['from'] == 'home'; final fromHome = state.uri.queryParameters['from'] == 'home';
return CalendarMonthScreen(resetToToday: fromHome); return CalendarMonthScreen(resetToToday: fromHome);
}, },
), ),
GoRoute( GoRoute(
path: '/todo', path: AppRoutes.todoList,
builder: (context, state) => const TodoQuadrantsScreen(), builder: (context, state) => const TodoQuadrantsScreen(),
), ),
GoRoute(
path: AppRoutes.todoCreate,
builder: (context, state) => const TodoEditScreen.create(),
),
GoRoute( GoRoute(
path: '/todo/:id', path: '/todo/:id',
builder: (context, state) => builder: (context, state) =>
TodoDetailScreen(todoId: state.pathParameters['id']!), TodoDetailScreen(todoId: state.pathParameters['id']!),
), ),
GoRoute( GoRoute(
path: '/settings', path: '/todo/:id/edit',
builder: (context, state) =>
TodoEditScreen(todoId: state.pathParameters['id']!),
),
GoRoute(
path: AppRoutes.settingsMain,
builder: (context, state) => const SettingsScreen(), builder: (context, state) => const SettingsScreen(),
), ),
GoRoute( GoRoute(
path: '/settings/features', path: AppRoutes.settingsFeatures,
builder: (context, state) => const FeaturesScreen(), builder: (context, state) => const FeaturesScreen(),
), ),
GoRoute( GoRoute(
path: '/settings/memory', path: AppRoutes.settingsMemory,
builder: (context, state) => const MemoryScreen(), builder: (context, state) => const MemoryScreen(),
), ),
GoRoute( GoRoute(
path: '/settings/account', path: AppRoutes.settingsAccount,
builder: (context, state) => const AccountScreen(), builder: (context, state) => const AccountScreen(),
), ),
GoRoute( GoRoute(
path: '/change-password', path: AppRoutes.settingsChangePassword,
builder: (context, state) => const ChangePasswordScreen(), builder: (context, state) => const ChangePasswordScreen(),
), ),
GoRoute( GoRoute(
path: '/edit-profile', path: AppRoutes.settingsEditProfile,
builder: (context, state) => const EditProfileScreen(), builder: (context, state) => const EditProfileScreen(),
), ),
], ],
+36
View File
@@ -0,0 +1,36 @@
class AppRoutes {
AppRoutes._();
static const authBoot = '/boot';
static const authLogin = '/';
static const authRegister = '/register';
static const authRegisterVerification = '/register/verification';
static const authResetPassword = '/reset-password';
static const homeMain = '/home';
static const messageInviteList = '/messages/invites';
static String messageInviteDetail(String id) => '/messages/invites/$id';
static const contactsList = '/contacts';
static const contactsAdd = '/contacts/add';
static const calendarDayWeek = '/calendar/dayweek';
static const calendarMonth = '/calendar/month';
static String calendarEventDetail(String id) => '/calendar/events/$id';
static const calendarEventCreate = '/calendar/events/new';
static String calendarEventEdit(String id) => '/calendar/events/$id/edit';
static String calendarEventShare(String id) => '/calendar/events/$id/share';
static const todoList = '/todo';
static String todoDetail(String id) => '/todo/$id';
static const todoCreate = '/todo/new';
static String todoEdit(String id) => '/todo/$id/edit';
static const settingsMain = '/settings';
static const settingsFeatures = '/settings/features';
static const settingsMemory = '/settings/memory';
static const settingsAccount = '/settings/account';
static const settingsChangePassword = '/change-password';
static const settingsEditProfile = '/edit-profile';
}
@@ -0,0 +1,16 @@
import 'package:flutter_test/flutter_test.dart';
import 'package:social_app/core/router/app_routes.dart';
void main() {
test('calendar and todo route builders generate concrete paths', () {
expect(
AppRoutes.calendarEventEdit('evt_123'),
'/calendar/events/evt_123/edit',
);
expect(
AppRoutes.calendarEventShare('evt_123'),
'/calendar/events/evt_123/share',
);
expect(AppRoutes.todoEdit('todo_123'), '/todo/todo_123/edit');
});
}