refactor(apps): 主题系统迁移至 ColorScheme + 扩展架构并支持 Dark Mode

This commit is contained in:
qzl
2026-03-27 19:07:39 +08:00
parent ecc1ec6ce4
commit ae29a8209b
146 changed files with 4301 additions and 3200 deletions
+100 -14
View File
@@ -1,37 +1,72 @@
import 'dart:async';
import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:go_router/go_router.dart';
import 'di/injection.dart';
import '../data/models/reminder_payload.dart';
import '../data/services/calendar_service.dart';
import '../data/services/local_notification_service.dart';
import '../data/services/reminder_notification_callbacks.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_event.dart';
import '../features/auth/presentation/bloc/auth_state.dart';
import '../features/chat/presentation/bloc/chat_bloc.dart';
import '../features/notification/domain/models/reminder_action.dart';
import '../features/notification/domain/services/reminder_action_executor.dart';
import 'router/app_router.dart';
import '../core/theme/app_theme.dart';
class LinksyApp extends StatelessWidget {
final AuthBloc authBloc;
class LinksyApp extends StatefulWidget {
const LinksyApp({super.key});
const LinksyApp({super.key, required this.authBloc});
@override
State<LinksyApp> createState() => _LinksyAppState();
}
class _LinksyAppState extends State<LinksyApp> {
late final AuthBloc _authBloc;
late final GoRouter _router;
String? _reminderBootstrapUserId;
@override
void initState() {
super.initState();
_authBloc = sl<AuthBloc>();
_authBloc.add(AuthStarted());
_router = createAppRouter(_authBloc);
unawaited(_bindNotificationResponseHandler());
}
@override
void dispose() {
_router.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return MultiBlocProvider(
providers: [
BlocProvider<AuthBloc>.value(value: authBloc),
BlocProvider<ChatBloc>(
create: (_) => ChatBloc(apiClient: sl<IApiClient>()),
),
],
return BlocProvider<AuthBloc>.value(
value: _authBloc,
child: BlocListener<AuthBloc, AuthState>(
listener: (context, state) {
// Handle auth state changes if needed
if (state is AuthAuthenticated &&
state.user.id != _reminderBootstrapUserId) {
_reminderBootstrapUserId = state.user.id;
unawaited(_rebuildUpcomingReminders());
}
if (state is AuthUnauthenticated) {
_reminderBootstrapUserId = null;
}
},
child: MaterialApp.router(
onGenerateTitle: (context) => AppLocalizations.of(context).appTitle,
debugShowCheckedModeBanner: false,
theme: AppTheme.light,
darkTheme: AppTheme.dark,
themeMode: ThemeMode.system,
locale: const Locale('zh'),
supportedLocales: AppLocalizations.supportedLocales,
localizationsDelegates: AppLocalizations.localizationsDelegates,
@@ -39,9 +74,60 @@ class LinksyApp extends StatelessWidget {
L10n.setLocale(Localizations.localeOf(context));
return child ?? const SizedBox.shrink();
},
routerConfig: createAppRouter(authBloc),
routerConfig: _router,
),
),
);
}
Future<void> _rebuildUpcomingReminders() async {
final now = DateTime.now();
final start = now.subtract(const Duration(days: 90));
final end = now.add(const Duration(days: 90));
try {
final events = await sl<CalendarService>().getEventsForRange(start, end);
await sl<LocalNotificationService>().rebuildUpcomingReminders(events);
} catch (error) {
debugPrint('reminder bootstrap skipped: $error');
}
}
Future<void> _bindNotificationResponseHandler() async {
await ReminderNotificationCallbacks.bindResponseHandler((response) async {
final payloadRaw = response.payload;
if (payloadRaw == null || payloadRaw.isEmpty) {
return;
}
ReminderPayload payload;
try {
payload = ReminderPayload.fromJson(
Map<String, dynamic>.from(jsonDecode(payloadRaw) as Map),
);
} catch (_) {
return;
}
final actionId = response.actionId;
ReminderAction? action;
if (actionId != null) {
try {
action = ReminderAction.fromValue(actionId);
} catch (_) {
action = null;
}
}
if (action == null) {
ReminderNotificationCallbacks.onNotificationPayloadReceived?.call(
payload,
);
return;
}
await sl<ReminderActionExecutor>().handleAction(
action: action,
payload: payload,
);
});
}
}