feat(apps): 重构 UI 架构为 presentation 层并新增 l10n 国际化支持

This commit is contained in:
qzl
2026-03-27 14:05:03 +08:00
parent b1f0eb8921
commit c592cc7854
178 changed files with 10748 additions and 5764 deletions
+47
View File
@@ -0,0 +1,47 @@
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'di/injection.dart';
import '../core/l10n/l10n.dart';
import '../core/network/i_api_client.dart';
import '../l10n/app_localizations.dart';
import '../features/auth/presentation/bloc/auth_bloc.dart';
import '../features/auth/presentation/bloc/auth_state.dart';
import '../features/chat/presentation/bloc/chat_bloc.dart';
import 'router/app_router.dart';
import '../core/theme/app_theme.dart';
class LinksyApp extends StatelessWidget {
final AuthBloc authBloc;
const LinksyApp({super.key, required this.authBloc});
@override
Widget build(BuildContext context) {
return MultiBlocProvider(
providers: [
BlocProvider<AuthBloc>.value(value: authBloc),
BlocProvider<ChatBloc>(
create: (_) => ChatBloc(apiClient: sl<IApiClient>()),
),
],
child: BlocListener<AuthBloc, AuthState>(
listener: (context, state) {
// Handle auth state changes if needed
},
child: MaterialApp.router(
onGenerateTitle: (context) => AppLocalizations.of(context).appTitle,
debugShowCheckedModeBanner: false,
theme: AppTheme.light,
locale: const Locale('zh'),
supportedLocales: AppLocalizations.supportedLocales,
localizationsDelegates: AppLocalizations.localizationsDelegates,
builder: (context, child) {
L10n.setLocale(Localizations.localeOf(context));
return child ?? const SizedBox.shrink();
},
routerConfig: createAppRouter(authBloc),
),
),
);
}
}
+174
View File
@@ -0,0 +1,174 @@
import 'package:dio/dio.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:get_it/get_it.dart';
import 'package:shared_preferences/shared_preferences.dart';
import '../../core/cache/cache_invalidator.dart';
import '../../core/cache/hybrid_cache_store.dart';
import '../../core/cache/memory_cache_store.dart';
import '../../core/cache/persistent_cache_store.dart';
import '../../core/network/api_client.dart';
import '../../core/network/i_api_client.dart';
import '../../core/storage/token_storage.dart';
import '../../core/config/env.dart';
import '../../features/notification/data/services/local_notification_service.dart';
import '../../features/auth/data/auth_api.dart';
import '../../features/auth/data/auth_repository.dart';
import '../../features/auth/data/auth_repository_impl.dart';
import '../../features/auth/presentation/bloc/auth_bloc.dart';
import '../../features/auth/presentation/bloc/auth_event.dart';
import '../../features/calendar/data/calendar_api.dart';
import '../../features/calendar/data/services/calendar_repository.dart';
import '../../features/calendar/data/services/calendar_service.dart';
import '../../features/notification/domain/services/reminder_action_executor.dart';
import '../../features/calendar/presentation/calendar_state_manager.dart';
import '../../features/contacts/data/friends_api.dart';
import '../../features/messages/data/inbox_api.dart';
import '../../features/settings/data/settings_api.dart';
import '../../features/settings/data/services/automation_jobs_api.dart';
import '../../features/settings/data/services/settings_user_cache.dart';
import '../../features/settings/data/services/user_profile_cache_repository.dart';
import '../../features/settings/data/services/memory_service.dart';
import '../../features/contacts/data/users/users_api.dart';
import '../../features/todo/data/todo_api.dart';
import '../../features/todo/data/todo_repository.dart';
final sl = GetIt.instance;
Future<void> configureDependencies() async {
if (sl.isRegistered<IApiClient>()) {
await sl.reset();
}
final SecureTokenStorage tokenStorage;
final dio = Dio(BaseOptions(baseUrl: Env.apiUrl));
tokenStorage = SecureTokenStorage(
const FlutterSecureStorage(
aOptions: AndroidOptions(encryptedSharedPreferences: true),
iOptions: IOSOptions(
accessibility: KeychainAccessibility.first_unlock_this_device,
),
),
);
final apiClient = ApiClient(
baseUrl: Env.apiUrl,
tokenStorage: tokenStorage,
dio: dio,
);
sl.registerSingleton<IApiClient>(apiClient);
final authApi = AuthApi(apiClient);
sl.registerSingleton<AuthApi>(authApi);
final sharedPreferences = await SharedPreferences.getInstance();
sl.registerSingleton<SharedPreferences>(sharedPreferences);
final memoryCacheStore = MemoryCacheStore();
final persistentCacheStore = PersistentCacheStore();
final hybridCacheStore = HybridCacheStore(
memory: memoryCacheStore,
persistent: persistentCacheStore,
);
sl.registerSingleton<MemoryCacheStore>(memoryCacheStore);
sl.registerSingleton<PersistentCacheStore>(persistentCacheStore);
sl.registerSingleton<HybridCacheStore>(hybridCacheStore);
sl.registerSingleton<CacheInvalidator>(
CacheInvalidator(store: hybridCacheStore),
);
final usersApi = UsersApi(apiClient);
sl.registerSingleton<UsersApi>(usersApi);
final userProfileCacheRepository = UserProfileCacheRepository(
store: hybridCacheStore,
remoteLoader: usersApi.getMe,
);
sl.registerSingleton<UserProfileCacheRepository>(userProfileCacheRepository);
final calendarApi = CalendarApi(apiClient);
sl.registerSingleton<CalendarApi>(calendarApi);
final calendarService = CalendarService(apiClient: apiClient);
sl.registerSingleton<CalendarService>(calendarService);
final calendarRepository = CalendarRepository(
store: hybridCacheStore,
loadDayFromRemote: calendarService.getEventsForDay,
loadMonthFromRemote: calendarService.getEventsForRange,
);
sl.registerSingleton<CalendarRepository>(calendarRepository);
sl.registerSingleton<LocalNotificationService>(LocalNotificationService());
final reminderActionExecutor = ReminderActionExecutor(
calendarService: calendarService,
notificationService: sl<LocalNotificationService>(),
);
sl.registerSingleton<ReminderActionExecutor>(reminderActionExecutor);
final friendsApi = FriendsApi(apiClient);
sl.registerSingleton<FriendsApi>(friendsApi);
final settingsApi = SettingsApi(apiClient);
sl.registerSingleton<SettingsApi>(settingsApi);
final automationJobsApi = AutomationJobsApi(apiClient);
sl.registerSingleton<AutomationJobsApi>(automationJobsApi);
final memoryService = MemoryService(apiClient);
sl.registerSingleton<MemoryService>(memoryService);
sl.registerSingleton<SettingsUserCache>(
SettingsUserCache(userProfileCacheRepository),
);
final inboxApi = InboxApi(apiClient);
sl.registerSingleton<InboxApi>(inboxApi);
final todoApi = TodoApi(apiClient);
sl.registerSingleton<TodoApi>(todoApi);
sl.registerSingleton<TodoRepository>(
TodoRepository(
api: todoApi,
store: hybridCacheStore,
invalidator: sl<CacheInvalidator>(),
),
);
final authRepository = AuthRepositoryImpl(
api: authApi,
tokenStorage: tokenStorage,
onLogout: () async {
apiClient.resetInterceptor();
if (sl.isRegistered<SettingsUserCache>()) {
sl<SettingsUserCache>().invalidate();
}
},
);
sl.registerSingleton<AuthRepository>(authRepository);
final authBloc = AuthBloc(authRepository);
sl.registerSingleton<AuthBloc>(authBloc);
apiClient.setRefreshCallback((token) async {
try {
await authRepository.refreshSession(token);
return true;
} catch (_) {
return false;
}
});
apiClient.setAuthFailureCallback(() async {
if (sl.isRegistered<SettingsUserCache>()) {
sl<SettingsUserCache>().invalidate();
}
authBloc.add(
const AuthSessionInvalidated(
source: AuthInvalidationSource.unauthorized401,
),
);
});
sl.registerSingleton<CalendarStateManager>(CalendarStateManager());
}
@@ -0,0 +1,4 @@
import 'package:flutter/widgets.dart';
final RouteObserver<ModalRoute<void>> appRouteObserver =
RouteObserver<ModalRoute<void>>();
+219
View File
@@ -0,0 +1,219 @@
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_state.dart';
import 'app_routes.dart';
import 'go_router_refresh_stream.dart';
import '../../../features/auth/presentation/screens/auth_boot_screen.dart';
import '../../../features/auth/presentation/screens/login_screen.dart';
import '../../../features/home/presentation/screens/home_screen.dart';
import '../../../features/messages/presentation/screens/message_invite_list_screen.dart';
import '../../../features/messages/presentation/screens/message_invite_detail_screen.dart';
import '../../../features/contacts/presentation/screens/contacts_screen.dart';
import '../../../features/contacts/presentation/screens/add_contact_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_dayweek_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_month_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_event_detail_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_event_create_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_event_edit_screen.dart';
import '../../../features/calendar/presentation/screens/calendar_event_share_screen.dart';
import '../../../features/calendar/presentation/calendar_time_utils.dart';
import '../../../features/todo/presentation/screens/todo_quadrants_screen.dart';
import '../../../features/todo/presentation/screens/todo_detail_screen.dart';
import '../../../features/todo/presentation/screens/todo_edit_screen.dart';
import '../../../features/settings/presentation/screens/settings_screen.dart';
import '../../../features/settings/presentation/screens/features_screen.dart';
import '../../../features/settings/presentation/screens/job_detail_screen.dart';
import '../../../features/settings/presentation/screens/memory_screen.dart';
import '../../../features/settings/presentation/screens/user_memory_view_screen.dart';
import '../../../features/settings/presentation/screens/work_memory_view_screen.dart';
import '../../../features/settings/presentation/screens/user_memory_detail_screen.dart';
import '../../../features/settings/presentation/screens/work_memory_detail_screen.dart';
import '../../../features/settings/presentation/screens/edit_profile_screen.dart';
final _homeSecondLevelRoutes = [
AppRoutes.shellHomeBranch,
AppRoutes.shellCalendarBranch,
AppRoutes.calendarMonth,
AppRoutes.shellTodoBranch,
AppRoutes.settingsMain,
];
final _protectedRoutes = [
..._homeSecondLevelRoutes,
AppRoutes.contactsList,
AppRoutes.contactsAdd,
'/calendar/events',
AppRoutes.settingsFeatures,
AppRoutes.settingsMemory,
AppRoutes.settingsMemoryUser,
AppRoutes.settingsMemoryWork,
AppRoutes.settingsMemoryUserEdit,
AppRoutes.settingsMemoryWorkEdit,
AppRoutes.settingsEditProfile,
AppRoutes.messageInviteList,
];
GoRouter createAppRouter(AuthBloc authBloc) {
return GoRouter(
initialLocation: AppRoutes.authBoot,
observers: [appRouteObserver],
refreshListenable: GoRouterRefreshStream(authBloc.stream),
redirect: (context, state) {
final authState = authBloc.state;
final isAuthenticated = authState is AuthAuthenticated;
final isAuthChecking =
authState is AuthInitial || authState is AuthLoading;
final isBootRoute = state.matchedLocation == AppRoutes.authBoot;
final isAuthRoute =
state.matchedLocation == AppRoutes.authLogin ||
state.matchedLocation.startsWith('/login');
final isProtected = _protectedRoutes.any(
(route) => state.matchedLocation.startsWith(route),
);
if (isAuthChecking && !isBootRoute) {
return AppRoutes.authBoot;
}
if (!isAuthChecking && isBootRoute) {
return isAuthenticated ? AppRoutes.homeMain : AppRoutes.authLogin;
}
if (!isAuthenticated && isProtected) {
return AppRoutes.authLogin;
}
if (isAuthenticated && isAuthRoute) {
return AppRoutes.homeMain;
}
return null;
},
routes: [
GoRoute(
path: AppRoutes.authBoot,
builder: (context, state) => const AuthBootScreen(),
),
GoRoute(
path: AppRoutes.calendarEventCreate,
builder: (context, state) => CalendarEventCreateScreen(
initialDate: parseYmd(state.uri.queryParameters['date']),
),
),
GoRoute(
path: '/calendar/events/:id',
builder: (context, state) =>
CalendarEventDetailScreen(eventId: state.pathParameters['id']!),
),
GoRoute(
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.authLogin,
builder: (context, state) => const LoginScreen(),
),
GoRoute(
path: AppRoutes.homeMain,
builder: (context, state) => const HomeScreen(),
),
GoRoute(
path: AppRoutes.messageInviteList,
builder: (context, state) => const MessageInviteListScreen(),
),
GoRoute(
path: '/messages/invites/:id',
builder: (context, state) =>
MessageInviteDetailScreen(inviteId: state.pathParameters['id']!),
),
GoRoute(
path: AppRoutes.contactsList,
builder: (context, state) => const ContactsScreen(),
),
GoRoute(
path: AppRoutes.contactsAdd,
builder: (context, state) => const AddContactScreen(),
),
GoRoute(
path: AppRoutes.calendarDayWeek,
builder: (context, state) {
final fromHome = state.uri.queryParameters['from'] == 'home';
final initialDate = parseYmd(state.uri.queryParameters['date']);
return CalendarDayWeekScreen(
initialDate: initialDate,
resetToToday: fromHome,
);
},
),
GoRoute(
path: AppRoutes.calendarMonth,
builder: (context, state) {
final fromHome = state.uri.queryParameters['from'] == 'home';
return CalendarMonthScreen(resetToToday: fromHome);
},
),
GoRoute(
path: AppRoutes.todoList,
builder: (context, state) => const TodoQuadrantsScreen(),
),
GoRoute(
path: AppRoutes.todoCreate,
builder: (context, state) => const TodoEditScreen.create(),
),
GoRoute(
path: '/todo/:id',
builder: (context, state) =>
TodoDetailScreen(todoId: state.pathParameters['id']!),
),
GoRoute(
path: '/todo/:id/edit',
builder: (context, state) =>
TodoEditScreen(todoId: state.pathParameters['id']!),
),
GoRoute(
path: AppRoutes.settingsMain,
builder: (context, state) => const SettingsScreen(),
),
GoRoute(
path: AppRoutes.settingsFeatures,
builder: (context, state) => const FeaturesScreen(),
),
GoRoute(
path: AppRoutes.settingsJobNew,
builder: (context, state) => const JobDetailScreen(),
),
GoRoute(
path: '/settings/job/:id',
builder: (context, state) =>
JobDetailScreen(jobId: state.pathParameters['id']),
),
GoRoute(
path: AppRoutes.settingsMemory,
builder: (context, state) => const MemoryScreen(),
),
GoRoute(
path: AppRoutes.settingsMemoryUser,
builder: (context, state) => const UserMemoryViewScreen(),
),
GoRoute(
path: AppRoutes.settingsMemoryWork,
builder: (context, state) => const WorkMemoryViewScreen(),
),
GoRoute(
path: AppRoutes.settingsMemoryUserEdit,
builder: (context, state) => const UserMemoryDetailScreen(),
),
GoRoute(
path: AppRoutes.settingsMemoryWorkEdit,
builder: (context, state) => const WorkMemoryDetailScreen(),
),
GoRoute(
path: AppRoutes.settingsEditProfile,
builder: (context, state) => const EditProfileScreen(),
),
],
);
}
+40
View File
@@ -0,0 +1,40 @@
class AppRoutes {
AppRoutes._();
static const authBoot = '/boot';
static const authLogin = '/';
static const homeMain = '/home';
static const shellHomeBranch = homeMain;
static const shellCalendarBranch = calendarDayWeek;
static const shellTodoBranch = todoList;
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 settingsJobNew = '/settings/job/new';
static String settingsJobDetail(String id) => '/settings/job/$id';
static const settingsMemory = '/settings/memory';
static const settingsMemoryUser = '/settings/memory/user';
static const settingsMemoryWork = '/settings/memory/work';
static const settingsMemoryUserEdit = '/settings/memory/user/edit';
static const settingsMemoryWorkEdit = '/settings/memory/work/edit';
static const settingsEditProfile = '/edit-profile';
}
@@ -0,0 +1,17 @@
import 'dart:async';
import 'package:flutter/foundation.dart';
class GoRouterRefreshStream extends ChangeNotifier {
GoRouterRefreshStream(Stream<dynamic> stream) {
notifyListeners();
_subscription = stream.listen((_) => notifyListeners());
}
late final StreamSubscription<dynamic> _subscription;
@override
void dispose() {
_subscription.cancel();
super.dispose();
}
}
@@ -0,0 +1,38 @@
import '../../features/auth/presentation/bloc/auth_state.dart';
import '../../features/calendar/data/services/calendar_service.dart';
import '../../features/notification/data/services/local_notification_service.dart';
class AuthSessionBootstrapper {
AuthSessionBootstrapper({
required CalendarService calendarService,
required LocalNotificationService notificationService,
}) : _calendarService = calendarService,
_notificationService = notificationService;
final CalendarService _calendarService;
final LocalNotificationService _notificationService;
String? _syncedUserId;
Future<void> syncForAuthState(AuthState state) async {
if (state is! AuthAuthenticated) {
_syncedUserId = null;
return;
}
if (_syncedUserId == state.user.id) {
return;
}
try {
final now = DateTime.now();
final start = now.subtract(const Duration(days: 90));
final end = now.add(const Duration(days: 90));
final events = await _calendarService.getEventsForRange(start, end);
await _notificationService.rebuildUpcomingReminders(events);
_syncedUserId = state.user.id;
} catch (_) {
// ignore reminder bootstrap failures
}
}
}