feat: 重构 Reminder Notification 系统并更新应用包名

This commit is contained in:
qzl
2026-03-30 18:36:57 +08:00
parent 9fb2a6857b
commit 91bf3c3f96
90 changed files with 5133 additions and 3017 deletions
@@ -1,3 +1,4 @@
import 'dart:async';
import 'dart:io';
import 'package:flutter/material.dart';
@@ -12,7 +13,7 @@ import '../../../../app/router/app_route_observer.dart';
import '../../../../app/router/app_routes.dart';
import '../../../../core/l10n/l10n.dart';
import '../../../../core/theme/design_tokens.dart';
import '../../../../features/messages/data/repositories/inbox_repository.dart';
import '../../../../core/inbox/inbox_sync_store.dart';
import '../../../chat/presentation/bloc/chat_bloc.dart';
import '../../data/voice_recorder.dart';
import '../controllers/home_keyboard_inset_calculator.dart';
@@ -73,7 +74,7 @@ class _HomeScreenState extends State<HomeScreen>
final ScrollController _scrollController = ScrollController();
late final ChatBloc _chatBloc;
late final VoiceRecorder _voiceRecorder;
late final InboxRepository _inboxRepository;
late final InboxSyncStore _inboxSyncStore;
late final Future<String> Function(String filePath) _transcribeAudio;
late final AnimationController _listeningAnimationController;
bool _isRecording = false;
@@ -110,7 +111,7 @@ class _HomeScreenState extends State<HomeScreen>
_chatBloc = context.read<ChatBloc>();
}
_voiceRecorder = widget.voiceRecorder ?? RecordVoiceRecorder();
_inboxRepository = sl<InboxRepository>();
_inboxSyncStore = sl<InboxSyncStore>();
_transcribeAudio =
widget.onTranscribeAudio ?? _chatBloc.transcribeAudioFile;
_listeningAnimationController = AnimationController(
@@ -118,24 +119,26 @@ class _HomeScreenState extends State<HomeScreen>
duration: const Duration(milliseconds: _rippleDurationMs),
);
_selectedImages.addAll(widget.initialSelectedImages);
if (widget.autoLoadHistory && _chatBloc.state.items.isEmpty) {
if (widget.autoLoadHistory &&
_chatBloc.state.items.isEmpty &&
!_chatBloc.state.isLoadingHistory) {
_chatBloc.loadHistory();
}
_scrollController.addListener(_handleScrollChanged);
_previousItemCount = _chatBloc.state.items.length;
_previousIsLoadingHistory = _chatBloc.state.isLoadingHistory;
_loadUnreadCount();
_inboxSyncStore.addListener(_handleInboxStateChanged);
unawaited(_inboxSyncStore.ensureStarted());
_handleInboxStateChanged();
}
Future<void> _loadUnreadCount() async {
try {
final messages = await _inboxRepository.getMessages(isRead: false);
if (mounted) {
setState(() => _unreadCount = messages.length);
}
} catch (_) {
// Ignore errors
void _handleInboxStateChanged() {
if (!mounted) {
return;
}
setState(() {
_unreadCount = _inboxSyncStore.unreadCount;
});
}
@override
@@ -145,6 +148,7 @@ class _HomeScreenState extends State<HomeScreen>
_scrollController.dispose();
_listeningAnimationController.dispose();
_voiceRecorder.dispose();
_inboxSyncStore.removeListener(_handleInboxStateChanged);
if (_routeAwareSubscribed) {
appRouteObserver.unsubscribe(this);
_routeAwareSubscribed = false;
@@ -164,6 +168,7 @@ class _HomeScreenState extends State<HomeScreen>
@override
void didPopNext() {
unawaited(_inboxSyncStore.refreshSnapshot());
_applyViewportDecision(
_dispatchViewportEvent(
type: ViewportEventType.screenResumedFromSubRoute,
@@ -49,8 +49,7 @@ class HomeDateDivider extends StatelessWidget {
Widget build(BuildContext context) {
final colorScheme = Theme.of(context).colorScheme;
final now = DateTime.now();
const weekdays = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'];
final weekday = weekdays[date.weekday - 1];
final weekday = _weekdayLabel(context, date.weekday);
final label = date.year == now.year
? context.l10n.homeDateLabelNoYear(date.month, date.day, weekday)
: context.l10n.homeDateLabelWithYear(
@@ -69,6 +68,19 @@ class HomeDateDivider extends StatelessWidget {
),
);
}
String _weekdayLabel(BuildContext context, int weekday) {
final labels = [
context.l10n.calendarWeekdaySun,
context.l10n.calendarWeekdayMon,
context.l10n.calendarWeekdayTue,
context.l10n.calendarWeekdayWed,
context.l10n.calendarWeekdayThu,
context.l10n.calendarWeekdayFri,
context.l10n.calendarWeekdaySat,
];
return labels[weekday % 7];
}
}
class HomeLoadMoreButton extends StatelessWidget {