refactor(calendar): remove deprecated reminder components
This commit is contained in:
@@ -20,7 +20,6 @@ import '../../features/calendar/data/calendar_api.dart';
|
|||||||
import '../../features/calendar/data/services/calendar_repository.dart';
|
import '../../features/calendar/data/services/calendar_repository.dart';
|
||||||
import '../../features/calendar/data/services/calendar_service.dart';
|
import '../../features/calendar/data/services/calendar_service.dart';
|
||||||
import '../../features/calendar/reminders/reminder_action_executor.dart';
|
import '../../features/calendar/reminders/reminder_action_executor.dart';
|
||||||
import '../../features/calendar/reminders/reminder_outbox_store.dart';
|
|
||||||
import '../../features/calendar/ui/calendar_state_manager.dart';
|
import '../../features/calendar/ui/calendar_state_manager.dart';
|
||||||
import '../../features/friends/data/friends_api.dart';
|
import '../../features/friends/data/friends_api.dart';
|
||||||
import '../../features/messages/data/inbox_api.dart';
|
import '../../features/messages/data/inbox_api.dart';
|
||||||
@@ -98,15 +97,11 @@ Future<void> configureDependencies() async {
|
|||||||
);
|
);
|
||||||
sl.registerSingleton<CalendarRepository>(calendarRepository);
|
sl.registerSingleton<CalendarRepository>(calendarRepository);
|
||||||
|
|
||||||
final reminderOutboxStore = ReminderOutboxStore(sharedPreferences);
|
|
||||||
sl.registerSingleton<ReminderOutboxStore>(reminderOutboxStore);
|
|
||||||
|
|
||||||
sl.registerSingleton<LocalNotificationService>(LocalNotificationService());
|
sl.registerSingleton<LocalNotificationService>(LocalNotificationService());
|
||||||
|
|
||||||
final reminderActionExecutor = ReminderActionExecutor(
|
final reminderActionExecutor = ReminderActionExecutor(
|
||||||
calendarService: calendarService,
|
calendarService: calendarService,
|
||||||
notificationService: sl<LocalNotificationService>(),
|
notificationService: sl<LocalNotificationService>(),
|
||||||
outboxStore: reminderOutboxStore,
|
|
||||||
);
|
);
|
||||||
sl.registerSingleton<ReminderActionExecutor>(reminderActionExecutor);
|
sl.registerSingleton<ReminderActionExecutor>(reminderActionExecutor);
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import 'dart:convert';
|
|||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/widgets.dart';
|
import 'package:flutter/widgets.dart';
|
||||||
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
|
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
import 'package:timezone/data/latest.dart' as tz_data;
|
import 'package:timezone/data/latest.dart' as tz_data;
|
||||||
import 'package:timezone/timezone.dart' as tz;
|
import 'package:timezone/timezone.dart' as tz;
|
||||||
|
|
||||||
@@ -12,8 +11,6 @@ import 'reminder_notification_callbacks.dart';
|
|||||||
import '../../features/calendar/data/models/schedule_item_model.dart';
|
import '../../features/calendar/data/models/schedule_item_model.dart';
|
||||||
import '../../features/calendar/reminders/models/reminder_action.dart';
|
import '../../features/calendar/reminders/models/reminder_action.dart';
|
||||||
import '../../features/calendar/reminders/models/reminder_payload.dart';
|
import '../../features/calendar/reminders/models/reminder_payload.dart';
|
||||||
import '../../features/calendar/reminders/reminder_action_dedupe_store.dart';
|
|
||||||
import '../../features/calendar/reminders/reminder_overlap_policy.dart';
|
|
||||||
|
|
||||||
typedef ReminderNotificationActionHandler =
|
typedef ReminderNotificationActionHandler =
|
||||||
Future<void> Function({
|
Future<void> Function({
|
||||||
@@ -38,9 +35,7 @@ class LocalNotificationService {
|
|||||||
static const String _actionSnooze = 'snooze10m';
|
static const String _actionSnooze = 'snooze10m';
|
||||||
|
|
||||||
final FlutterLocalNotificationsPlugin _plugin;
|
final FlutterLocalNotificationsPlugin _plugin;
|
||||||
final ReminderOverlapPolicy _overlapPolicy;
|
|
||||||
final ReminderPermissionFallbackTracker? _permissionFallbackTracker;
|
final ReminderPermissionFallbackTracker? _permissionFallbackTracker;
|
||||||
ReminderActionDedupeStore? _dedupeStore;
|
|
||||||
bool _initialized = false;
|
bool _initialized = false;
|
||||||
bool _canDeliverSystemNotification = true;
|
bool _canDeliverSystemNotification = true;
|
||||||
ReminderNotificationActionHandler? _actionHandler;
|
ReminderNotificationActionHandler? _actionHandler;
|
||||||
@@ -50,12 +45,8 @@ class LocalNotificationService {
|
|||||||
|
|
||||||
LocalNotificationService({
|
LocalNotificationService({
|
||||||
FlutterLocalNotificationsPlugin? plugin,
|
FlutterLocalNotificationsPlugin? plugin,
|
||||||
ReminderOverlapPolicy? overlapPolicy,
|
|
||||||
ReminderActionDedupeStore? dedupeStore,
|
|
||||||
ReminderPermissionFallbackTracker? permissionFallbackTracker,
|
ReminderPermissionFallbackTracker? permissionFallbackTracker,
|
||||||
}) : _plugin = plugin ?? FlutterLocalNotificationsPlugin(),
|
}) : _plugin = plugin ?? FlutterLocalNotificationsPlugin(),
|
||||||
_overlapPolicy = overlapPolicy ?? const ReminderOverlapPolicy(),
|
|
||||||
_dedupeStore = dedupeStore,
|
|
||||||
_permissionFallbackTracker = permissionFallbackTracker;
|
_permissionFallbackTracker = permissionFallbackTracker;
|
||||||
|
|
||||||
void bindActionHandler(ReminderNotificationActionHandler handler) {
|
void bindActionHandler(ReminderNotificationActionHandler handler) {
|
||||||
@@ -128,19 +119,9 @@ class LocalNotificationService {
|
|||||||
>();
|
>();
|
||||||
await iosImpl?.requestPermissions(alert: true, badge: true, sound: true);
|
await iosImpl?.requestPermissions(alert: true, badge: true, sound: true);
|
||||||
|
|
||||||
await _ensureDedupeStore();
|
|
||||||
|
|
||||||
_initialized = true;
|
_initialized = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _ensureDedupeStore() async {
|
|
||||||
if (_dedupeStore != null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
_dedupeStore = ReminderActionDedupeStore(prefs);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _refreshAndroidNotificationAvailability() async {
|
Future<void> _refreshAndroidNotificationAvailability() async {
|
||||||
if (defaultTargetPlatform != TargetPlatform.android) {
|
if (defaultTargetPlatform != TargetPlatform.android) {
|
||||||
return;
|
return;
|
||||||
@@ -166,8 +147,9 @@ class LocalNotificationService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final now = DateTime.now();
|
final now = DateTime.now();
|
||||||
final fireAt = _overlapPolicy.resolveFirstFireAt(event, now: now);
|
final reminderMinutes = event.metadata?.reminderMinutes ?? 0;
|
||||||
if (fireAt == null) {
|
final fireAt = event.startAt.subtract(Duration(minutes: reminderMinutes));
|
||||||
|
if (fireAt.isBefore(now)) {
|
||||||
await cancelEventReminder(event.id);
|
await cancelEventReminder(event.id);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -225,31 +207,21 @@ class LocalNotificationService {
|
|||||||
) async {
|
) async {
|
||||||
await initialize();
|
await initialize();
|
||||||
await _refreshAndroidNotificationAvailability();
|
await _refreshAndroidNotificationAvailability();
|
||||||
if (!_canDeliverSystemNotification) {
|
for (final event in events) {
|
||||||
_clearAllInAppFallbackTimers();
|
if (!_canDeliverSystemNotification) {
|
||||||
final now = DateTime.now();
|
final reminderMinutes = event.metadata?.reminderMinutes ?? 0;
|
||||||
final groups = _overlapPolicy.groupByMinute(events, now: now);
|
final fireAt = event.startAt.subtract(
|
||||||
for (final group in groups) {
|
Duration(minutes: reminderMinutes),
|
||||||
if (group.isAggregate) {
|
|
||||||
await _scheduleInAppAggregateFallback(group.events, group.fireAt);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
await _scheduleInAppFallbackRemindersFrom(
|
|
||||||
event: group.events.first,
|
|
||||||
firstFireAt: group.fireAt,
|
|
||||||
);
|
);
|
||||||
}
|
if (fireAt.isAfter(DateTime.now())) {
|
||||||
return;
|
await _scheduleInAppFallbackRemindersFrom(
|
||||||
}
|
event: event,
|
||||||
|
firstFireAt: fireAt,
|
||||||
final now = DateTime.now();
|
);
|
||||||
final groups = _overlapPolicy.groupByMinute(events, now: now);
|
}
|
||||||
for (final group in groups) {
|
|
||||||
if (group.isAggregate) {
|
|
||||||
await _scheduleAggregateReminder(group.events, group.fireAt);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
await upsertEventReminder(group.events.first);
|
await upsertEventReminder(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -693,21 +665,6 @@ class LocalNotificationService {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final dedupeStore = _dedupeStore;
|
|
||||||
if (dedupeStore != null) {
|
|
||||||
final notificationId = response.id?.toString() ?? payload.eventId;
|
|
||||||
final fireTimeBucket =
|
|
||||||
payload.fireTimeBucket ??
|
|
||||||
(payload.startAt.millisecondsSinceEpoch ~/
|
|
||||||
const Duration(minutes: 1).inMilliseconds);
|
|
||||||
final actionExecutionId =
|
|
||||||
'$notificationId|${action.value}|$fireTimeBucket';
|
|
||||||
final isNew = await dedupeStore.markIfNew(actionExecutionId);
|
|
||||||
if (!isNew) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await handler(action: action, payload: payload);
|
await handler(action: action, payload: payload);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,8 +5,6 @@ import 'package:flutter/foundation.dart';
|
|||||||
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
|
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
import '../../features/calendar/reminders/reminder_cold_start_queue.dart';
|
|
||||||
|
|
||||||
typedef ReminderNotificationResponseHandler =
|
typedef ReminderNotificationResponseHandler =
|
||||||
Future<void> Function(NotificationResponse response);
|
Future<void> Function(NotificationResponse response);
|
||||||
|
|
||||||
@@ -15,8 +13,6 @@ class ReminderNotificationCallbacks {
|
|||||||
'calendar_reminder_pending_notification_responses_v1';
|
'calendar_reminder_pending_notification_responses_v1';
|
||||||
static ReminderNotificationResponseHandler? _responseHandler;
|
static ReminderNotificationResponseHandler? _responseHandler;
|
||||||
static Future<void> _pendingStorageLock = Future<void>.value();
|
static Future<void> _pendingStorageLock = Future<void>.value();
|
||||||
static final ReminderColdStartQueue _coldStartQueue =
|
|
||||||
ReminderColdStartQueue();
|
|
||||||
|
|
||||||
@visibleForTesting
|
@visibleForTesting
|
||||||
static Future<void> resetForTest() async {
|
static Future<void> resetForTest() async {
|
||||||
@@ -106,38 +102,35 @@ class ReminderNotificationCallbacks {
|
|||||||
|
|
||||||
final remaining = <String>[];
|
final remaining = <String>[];
|
||||||
for (final raw in pending) {
|
for (final raw in pending) {
|
||||||
_coldStartQueue.enqueue(() async {
|
Map<String, dynamic> parsed;
|
||||||
Map<String, dynamic> parsed;
|
try {
|
||||||
try {
|
parsed = Map<String, dynamic>.from(jsonDecode(raw) as Map);
|
||||||
parsed = Map<String, dynamic>.from(jsonDecode(raw) as Map);
|
} catch (_) {
|
||||||
} catch (_) {
|
continue;
|
||||||
return;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
final id = parsed['id'] as int?;
|
final id = parsed['id'] as int?;
|
||||||
final actionId = parsed['actionId'] as String?;
|
final actionId = parsed['actionId'] as String?;
|
||||||
final payload = parsed['payload'] as String?;
|
final payload = parsed['payload'] as String?;
|
||||||
final typeIndex = (parsed['type'] as int?) ?? 0;
|
final typeIndex = (parsed['type'] as int?) ?? 0;
|
||||||
final input = parsed['input'] as String?;
|
final input = parsed['input'] as String?;
|
||||||
final type = NotificationResponseType.values[typeIndex.clamp(0, 1)];
|
final type = NotificationResponseType.values[typeIndex.clamp(0, 1)];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await handler(
|
await handler(
|
||||||
NotificationResponse(
|
NotificationResponse(
|
||||||
id: id,
|
id: id,
|
||||||
actionId: actionId,
|
actionId: actionId,
|
||||||
payload: payload,
|
payload: payload,
|
||||||
input: input,
|
input: input,
|
||||||
notificationResponseType: type,
|
notificationResponseType: type,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
} catch (_) {
|
} catch (_) {
|
||||||
remaining.add(raw);
|
remaining.add(raw);
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await _coldStartQueue.replay();
|
|
||||||
if (remaining.isEmpty) {
|
if (remaining.isEmpty) {
|
||||||
await prefs.remove(_pendingKey);
|
await prefs.remove(_pendingKey);
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -29,8 +29,6 @@ class AuthSessionBootstrapper {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await _reminderActionExecutor.replayPendingActions();
|
|
||||||
|
|
||||||
final now = DateTime.now();
|
final now = DateTime.now();
|
||||||
final start = now.subtract(const Duration(days: 90));
|
final start = now.subtract(const Duration(days: 90));
|
||||||
final end = now.add(const Duration(days: 90));
|
final end = now.add(const Duration(days: 90));
|
||||||
|
|||||||
@@ -1,52 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
|
|
||||||
typedef SetStringListFn = Future<bool> Function(String key, List<String> value);
|
|
||||||
|
|
||||||
class ReminderActionDedupeStore {
|
|
||||||
static const String _key = 'calendar_reminder_action_dedupe_v1';
|
|
||||||
static const int _maxEntries = 512;
|
|
||||||
|
|
||||||
final SharedPreferences _prefs;
|
|
||||||
final SetStringListFn _setStringList;
|
|
||||||
Future<void> _queue = Future<void>.value();
|
|
||||||
|
|
||||||
ReminderActionDedupeStore(
|
|
||||||
SharedPreferences prefs, {
|
|
||||||
SetStringListFn? setStringList,
|
|
||||||
}) : _prefs = prefs,
|
|
||||||
_setStringList = setStringList ?? prefs.setStringList;
|
|
||||||
|
|
||||||
Future<bool> markIfNew(String actionExecutionId) async {
|
|
||||||
final completer = Completer<bool>();
|
|
||||||
_queue = _queue
|
|
||||||
.then((_) async {
|
|
||||||
completer.complete(await _markIfNewInternal(actionExecutionId));
|
|
||||||
})
|
|
||||||
.catchError((_) {
|
|
||||||
if (!completer.isCompleted) {
|
|
||||||
completer.complete(false);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return completer.future;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> _markIfNewInternal(String actionExecutionId) async {
|
|
||||||
final current = List<String>.from(
|
|
||||||
_prefs.getStringList(_key) ?? const <String>[],
|
|
||||||
);
|
|
||||||
if (current.contains(actionExecutionId)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
current.add(actionExecutionId);
|
|
||||||
if (current.length > _maxEntries) {
|
|
||||||
current.removeRange(0, current.length - _maxEntries);
|
|
||||||
}
|
|
||||||
|
|
||||||
final saved = await _setStringList(_key, current);
|
|
||||||
return saved;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,26 +1,17 @@
|
|||||||
import 'dart:math';
|
|
||||||
|
|
||||||
import '../data/services/calendar_service.dart';
|
import '../data/services/calendar_service.dart';
|
||||||
import '../../../core/notifications/local_notification_service.dart';
|
import '../../../core/notifications/local_notification_service.dart';
|
||||||
import 'models/reminder_action.dart';
|
import 'models/reminder_action.dart';
|
||||||
import 'models/reminder_payload.dart';
|
import 'models/reminder_payload.dart';
|
||||||
import 'reminder_outbox_store.dart';
|
|
||||||
|
|
||||||
class ReminderActionExecutor {
|
class ReminderActionExecutor {
|
||||||
final CalendarService _calendarService;
|
final CalendarService _calendarService;
|
||||||
final LocalNotificationService _notificationService;
|
final LocalNotificationService _notificationService;
|
||||||
final ReminderOutboxStore _outboxStore;
|
|
||||||
final Random _random;
|
|
||||||
|
|
||||||
ReminderActionExecutor({
|
ReminderActionExecutor({
|
||||||
required CalendarService calendarService,
|
required CalendarService calendarService,
|
||||||
required LocalNotificationService notificationService,
|
required LocalNotificationService notificationService,
|
||||||
required ReminderOutboxStore outboxStore,
|
|
||||||
Random? random,
|
|
||||||
}) : _calendarService = calendarService,
|
}) : _calendarService = calendarService,
|
||||||
_notificationService = notificationService,
|
_notificationService = notificationService;
|
||||||
_outboxStore = outboxStore,
|
|
||||||
_random = random ?? Random();
|
|
||||||
|
|
||||||
Future<void> handleAction({
|
Future<void> handleAction({
|
||||||
required ReminderAction action,
|
required ReminderAction action,
|
||||||
@@ -35,7 +26,7 @@ class ReminderActionExecutor {
|
|||||||
if (action == ReminderAction.archive) {
|
if (action == ReminderAction.archive) {
|
||||||
for (final id in ids) {
|
for (final id in ids) {
|
||||||
await _notificationService.cancelEventReminder(id);
|
await _notificationService.cancelEventReminder(id);
|
||||||
await _archiveEvent(id, ReminderAction.archive);
|
await _archiveEvent(id);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -47,21 +38,6 @@ class ReminderActionExecutor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> replayPendingActions() async {
|
|
||||||
final pending = await _outboxStore.listPending();
|
|
||||||
for (final item in pending) {
|
|
||||||
if (item.targetStatus != 'archived') {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
await _calendarService.archiveEvent(item.eventId);
|
|
||||||
await _outboxStore.markDone(item.opId);
|
|
||||||
} catch (error) {
|
|
||||||
await _outboxStore.markRetry(item.opId, error.toString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _snoozeEvent(String eventId) async {
|
Future<void> _snoozeEvent(String eventId) async {
|
||||||
final event = await _calendarService.getEventById(eventId);
|
final event = await _calendarService.getEventById(eventId);
|
||||||
if (event == null) {
|
if (event == null) {
|
||||||
@@ -71,37 +47,21 @@ class ReminderActionExecutor {
|
|||||||
final endAt = event.endAt;
|
final endAt = event.endAt;
|
||||||
if (endAt != null && !now.isBefore(endAt)) {
|
if (endAt != null && !now.isBefore(endAt)) {
|
||||||
await _notificationService.cancelEventReminder(eventId);
|
await _notificationService.cancelEventReminder(eventId);
|
||||||
await _archiveEvent(eventId, ReminderAction.archive);
|
await _archiveEvent(eventId);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final nextAt = now.add(const Duration(minutes: 10));
|
final nextAt = now.add(const Duration(minutes: 10));
|
||||||
if (endAt != null && !nextAt.isBefore(endAt)) {
|
if (endAt != null && !nextAt.isBefore(endAt)) {
|
||||||
await _notificationService.cancelEventReminder(eventId);
|
await _notificationService.cancelEventReminder(eventId);
|
||||||
await _archiveEvent(eventId, ReminderAction.archive);
|
await _archiveEvent(eventId);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
await _notificationService.scheduleReminderAt(event, nextAt);
|
await _notificationService.scheduleReminderAt(event, nextAt);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _archiveEvent(String eventId, ReminderAction action) async {
|
Future<void> _archiveEvent(String eventId) async {
|
||||||
try {
|
await _calendarService.archiveEvent(eventId);
|
||||||
await _calendarService.archiveEvent(eventId);
|
|
||||||
return;
|
|
||||||
} catch (_) {
|
|
||||||
// fall through to enqueue local outbox for retry
|
|
||||||
}
|
|
||||||
|
|
||||||
final opId =
|
|
||||||
'${DateTime.now().millisecondsSinceEpoch}-${_random.nextInt(1 << 32)}';
|
|
||||||
final outboxItem = ReminderOutboxItem(
|
|
||||||
opId: opId,
|
|
||||||
eventId: eventId,
|
|
||||||
action: action,
|
|
||||||
targetStatus: 'archived',
|
|
||||||
occurredAt: DateTime.now(),
|
|
||||||
);
|
|
||||||
await _outboxStore.enqueue(outboxItem);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,57 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
import 'dart:collection';
|
|
||||||
|
|
||||||
typedef ReminderColdStartReplayTask = Future<void> Function();
|
|
||||||
typedef ReminderColdStartTaskErrorHandler =
|
|
||||||
void Function(Object error, StackTrace stackTrace);
|
|
||||||
|
|
||||||
class ReminderColdStartQueue {
|
|
||||||
final Queue<ReminderColdStartReplayTask> _tasks =
|
|
||||||
Queue<ReminderColdStartReplayTask>();
|
|
||||||
final ReminderColdStartTaskErrorHandler? _onTaskError;
|
|
||||||
Future<void>? _inFlightReplay;
|
|
||||||
|
|
||||||
ReminderColdStartQueue({ReminderColdStartTaskErrorHandler? onTaskError})
|
|
||||||
: _onTaskError = onTaskError;
|
|
||||||
|
|
||||||
void enqueue(ReminderColdStartReplayTask task) {
|
|
||||||
_tasks.add(task);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> replay() {
|
|
||||||
final inFlightReplay = _inFlightReplay;
|
|
||||||
if (inFlightReplay != null) {
|
|
||||||
return inFlightReplay;
|
|
||||||
}
|
|
||||||
|
|
||||||
final replayCompleter = Completer<void>();
|
|
||||||
final replayFuture = replayCompleter.future;
|
|
||||||
_inFlightReplay = replayFuture;
|
|
||||||
|
|
||||||
scheduleMicrotask(() async {
|
|
||||||
try {
|
|
||||||
await _replayInternal();
|
|
||||||
replayCompleter.complete();
|
|
||||||
} catch (error, stackTrace) {
|
|
||||||
replayCompleter.completeError(error, stackTrace);
|
|
||||||
} finally {
|
|
||||||
if (identical(_inFlightReplay, replayFuture)) {
|
|
||||||
_inFlightReplay = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return replayFuture;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _replayInternal() async {
|
|
||||||
while (_tasks.isNotEmpty) {
|
|
||||||
final task = _tasks.removeFirst();
|
|
||||||
try {
|
|
||||||
await task();
|
|
||||||
} catch (error, stackTrace) {
|
|
||||||
_onTaskError?.call(error, stackTrace);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,202 +0,0 @@
|
|||||||
import 'dart:convert';
|
|
||||||
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
|
|
||||||
import 'models/reminder_action.dart';
|
|
||||||
|
|
||||||
class ReminderOutboxItem {
|
|
||||||
final String opId;
|
|
||||||
final String eventId;
|
|
||||||
final ReminderAction action;
|
|
||||||
final String? targetStatus;
|
|
||||||
final DateTime occurredAt;
|
|
||||||
final int retryCount;
|
|
||||||
final DateTime? nextRetryAt;
|
|
||||||
final ReminderOutboxState state;
|
|
||||||
final String? lastError;
|
|
||||||
|
|
||||||
const ReminderOutboxItem({
|
|
||||||
required this.opId,
|
|
||||||
required this.eventId,
|
|
||||||
required this.action,
|
|
||||||
required this.occurredAt,
|
|
||||||
this.targetStatus,
|
|
||||||
this.retryCount = 0,
|
|
||||||
this.nextRetryAt,
|
|
||||||
this.state = ReminderOutboxState.pending,
|
|
||||||
this.lastError,
|
|
||||||
});
|
|
||||||
|
|
||||||
String get idempotencyBucket {
|
|
||||||
final bucket =
|
|
||||||
occurredAt.millisecondsSinceEpoch ~/
|
|
||||||
const Duration(minutes: 1).inMilliseconds;
|
|
||||||
return '$eventId|${action.value}|$bucket';
|
|
||||||
}
|
|
||||||
|
|
||||||
ReminderOutboxItem copyWith({
|
|
||||||
int? retryCount,
|
|
||||||
DateTime? nextRetryAt,
|
|
||||||
ReminderOutboxState? state,
|
|
||||||
String? lastError,
|
|
||||||
}) {
|
|
||||||
return ReminderOutboxItem(
|
|
||||||
opId: opId,
|
|
||||||
eventId: eventId,
|
|
||||||
action: action,
|
|
||||||
targetStatus: targetStatus,
|
|
||||||
occurredAt: occurredAt,
|
|
||||||
retryCount: retryCount ?? this.retryCount,
|
|
||||||
nextRetryAt: nextRetryAt ?? this.nextRetryAt,
|
|
||||||
state: state ?? this.state,
|
|
||||||
lastError: lastError ?? this.lastError,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
|
||||||
return {
|
|
||||||
'opId': opId,
|
|
||||||
'eventId': eventId,
|
|
||||||
'action': action.value,
|
|
||||||
'targetStatus': targetStatus,
|
|
||||||
'occurredAt': occurredAt.toIso8601String(),
|
|
||||||
'retryCount': retryCount,
|
|
||||||
'nextRetryAt': nextRetryAt?.toIso8601String(),
|
|
||||||
'state': state.value,
|
|
||||||
'lastError': lastError,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
factory ReminderOutboxItem.fromJson(Map<String, dynamic> json) {
|
|
||||||
return ReminderOutboxItem(
|
|
||||||
opId: (json['opId'] as String?) ?? '',
|
|
||||||
eventId: (json['eventId'] as String?) ?? '',
|
|
||||||
action: ReminderAction.fromValue(
|
|
||||||
(json['action'] as String?) ?? 'timeout_30s',
|
|
||||||
),
|
|
||||||
targetStatus: json['targetStatus'] as String?,
|
|
||||||
occurredAt: DateTime.parse(json['occurredAt'] as String),
|
|
||||||
retryCount: (json['retryCount'] as int?) ?? 0,
|
|
||||||
nextRetryAt: json['nextRetryAt'] != null
|
|
||||||
? DateTime.parse(json['nextRetryAt'] as String)
|
|
||||||
: null,
|
|
||||||
state: ReminderOutboxState.fromValue(
|
|
||||||
(json['state'] as String?) ?? 'pending',
|
|
||||||
),
|
|
||||||
lastError: json['lastError'] as String?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
enum ReminderOutboxState {
|
|
||||||
pending('pending'),
|
|
||||||
done('done'),
|
|
||||||
dead('dead');
|
|
||||||
|
|
||||||
const ReminderOutboxState(this.value);
|
|
||||||
final String value;
|
|
||||||
|
|
||||||
static ReminderOutboxState fromValue(String raw) {
|
|
||||||
return ReminderOutboxState.values.firstWhere(
|
|
||||||
(item) => item.value == raw,
|
|
||||||
orElse: () => ReminderOutboxState.pending,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class ReminderOutboxStore {
|
|
||||||
static const String _key = 'calendar_reminder_outbox_v1';
|
|
||||||
final SharedPreferences _prefs;
|
|
||||||
|
|
||||||
ReminderOutboxStore(this._prefs);
|
|
||||||
|
|
||||||
Future<void> enqueue(ReminderOutboxItem item) async {
|
|
||||||
final current = await _readAll();
|
|
||||||
final duplicated = current.any(
|
|
||||||
(existing) =>
|
|
||||||
existing.state == ReminderOutboxState.pending &&
|
|
||||||
existing.idempotencyBucket == item.idempotencyBucket,
|
|
||||||
);
|
|
||||||
if (duplicated) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
current.add(item);
|
|
||||||
await _writeAll(current);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<List<ReminderOutboxItem>> listPending() async {
|
|
||||||
final all = await _readAll();
|
|
||||||
final now = DateTime.now();
|
|
||||||
return all
|
|
||||||
.where((item) => item.state == ReminderOutboxState.pending)
|
|
||||||
.where(
|
|
||||||
(item) => item.nextRetryAt == null || !item.nextRetryAt!.isAfter(now),
|
|
||||||
)
|
|
||||||
.toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> markDone(String opId) async {
|
|
||||||
final all = await _readAll();
|
|
||||||
final updated = all
|
|
||||||
.map(
|
|
||||||
(item) => item.opId == opId
|
|
||||||
? item.copyWith(
|
|
||||||
state: ReminderOutboxState.done,
|
|
||||||
nextRetryAt: null,
|
|
||||||
)
|
|
||||||
: item,
|
|
||||||
)
|
|
||||||
.toList();
|
|
||||||
await _writeAll(updated);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> markRetry(String opId, String error) async {
|
|
||||||
final all = await _readAll();
|
|
||||||
final updated = all.map((item) {
|
|
||||||
if (item.opId != opId) {
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
final nextRetryCount = item.retryCount + 1;
|
|
||||||
if (nextRetryCount >= 8) {
|
|
||||||
return item.copyWith(
|
|
||||||
retryCount: nextRetryCount,
|
|
||||||
state: ReminderOutboxState.dead,
|
|
||||||
lastError: error,
|
|
||||||
nextRetryAt: null,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
final delayMinutes = nextRetryCount == 1 ? 0 : 1 << (nextRetryCount - 1);
|
|
||||||
return item.copyWith(
|
|
||||||
retryCount: nextRetryCount,
|
|
||||||
lastError: error,
|
|
||||||
nextRetryAt: DateTime.now().add(Duration(minutes: delayMinutes)),
|
|
||||||
);
|
|
||||||
}).toList();
|
|
||||||
await _writeAll(updated);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<List<ReminderOutboxItem>> _readAll() async {
|
|
||||||
try {
|
|
||||||
final raw = _prefs.getString(_key);
|
|
||||||
if (raw == null || raw.isEmpty) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
final list = jsonDecode(raw) as List<dynamic>;
|
|
||||||
return list
|
|
||||||
.whereType<Map>()
|
|
||||||
.map(
|
|
||||||
(item) =>
|
|
||||||
ReminderOutboxItem.fromJson(Map<String, dynamic>.from(item)),
|
|
||||||
)
|
|
||||||
.toList();
|
|
||||||
} catch (_) {
|
|
||||||
await _prefs.remove(_key);
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _writeAll(List<ReminderOutboxItem> items) async {
|
|
||||||
final raw = jsonEncode(items.map((item) => item.toJson()).toList());
|
|
||||||
await _prefs.setString(_key, raw);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
import '../data/models/schedule_item_model.dart';
|
|
||||||
|
|
||||||
class ReminderOverlapGroup {
|
|
||||||
final DateTime fireAt;
|
|
||||||
final List<ScheduleItemModel> events;
|
|
||||||
|
|
||||||
const ReminderOverlapGroup({required this.fireAt, required this.events});
|
|
||||||
|
|
||||||
bool get isAggregate => events.length > 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
class ReminderOverlapPolicy {
|
|
||||||
const ReminderOverlapPolicy();
|
|
||||||
|
|
||||||
List<ReminderOverlapGroup> groupByMinute(
|
|
||||||
Iterable<ScheduleItemModel> events, {
|
|
||||||
required DateTime now,
|
|
||||||
}) {
|
|
||||||
final buckets = <String, List<ScheduleItemModel>>{};
|
|
||||||
final minuteToFireAt = <String, DateTime>{};
|
|
||||||
|
|
||||||
for (final event in events) {
|
|
||||||
final fireAt = resolveFirstFireAt(event, now: now);
|
|
||||||
if (fireAt == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
final minute = DateTime(
|
|
||||||
fireAt.year,
|
|
||||||
fireAt.month,
|
|
||||||
fireAt.day,
|
|
||||||
fireAt.hour,
|
|
||||||
fireAt.minute,
|
|
||||||
);
|
|
||||||
final key = minute.toIso8601String();
|
|
||||||
buckets.putIfAbsent(key, () => <ScheduleItemModel>[]).add(event);
|
|
||||||
minuteToFireAt[key] = minuteToFireAt[key] ?? fireAt;
|
|
||||||
}
|
|
||||||
|
|
||||||
final groups = buckets.entries
|
|
||||||
.map(
|
|
||||||
(entry) => ReminderOverlapGroup(
|
|
||||||
fireAt: minuteToFireAt[entry.key]!,
|
|
||||||
events: entry.value,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
groups.sort((left, right) => left.fireAt.compareTo(right.fireAt));
|
|
||||||
return groups;
|
|
||||||
}
|
|
||||||
|
|
||||||
DateTime? resolveFirstFireAt(
|
|
||||||
ScheduleItemModel event, {
|
|
||||||
required DateTime now,
|
|
||||||
}) {
|
|
||||||
if (event.status != ScheduleStatus.active) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
final reminderMinutes = event.metadata?.reminderMinutes;
|
|
||||||
if (reminderMinutes == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
final remindAt = event.startAt.subtract(Duration(minutes: reminderMinutes));
|
|
||||||
final endAt = event.endAt;
|
|
||||||
|
|
||||||
if (endAt != null && !now.isBefore(endAt)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (now.isBefore(remindAt)) {
|
|
||||||
return remindAt;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (endAt != null && now.isBefore(endAt)) {
|
|
||||||
return now.add(const Duration(seconds: 5));
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,73 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import '../../../../core/theme/design_tokens.dart';
|
|
||||||
import '../models/reminder_action.dart';
|
|
||||||
import '../models/reminder_payload.dart';
|
|
||||||
import '../reminder_action_executor.dart';
|
|
||||||
import 'reminder_presentation_coordinator.dart';
|
|
||||||
import 'widgets/reminder_action_sheet.dart';
|
|
||||||
|
|
||||||
class ReminderForegroundPresenter {
|
|
||||||
final GlobalKey<NavigatorState> _navigatorKey;
|
|
||||||
final ReminderActionExecutor _executor;
|
|
||||||
final ReminderPresentationCoordinator _coordinator;
|
|
||||||
bool _isPresenting = false;
|
|
||||||
|
|
||||||
ReminderForegroundPresenter({
|
|
||||||
required GlobalKey<NavigatorState> navigatorKey,
|
|
||||||
required ReminderActionExecutor executor,
|
|
||||||
ReminderPresentationCoordinator? coordinator,
|
|
||||||
}) : _navigatorKey = navigatorKey,
|
|
||||||
_executor = executor,
|
|
||||||
_coordinator = coordinator ?? ReminderPresentationCoordinator();
|
|
||||||
|
|
||||||
Future<void> present(ReminderPayload payload) async {
|
|
||||||
final context = _navigatorKey.currentContext;
|
|
||||||
if (context == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final lifecycleState = WidgetsBinding.instance.lifecycleState;
|
|
||||||
final isAppActive = lifecycleState == AppLifecycleState.resumed;
|
|
||||||
final shouldPresent = _coordinator.shouldPresent(
|
|
||||||
eventId: payload.eventId,
|
|
||||||
isAppActive: isAppActive,
|
|
||||||
);
|
|
||||||
if (!shouldPresent || _isPresenting) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_isPresenting = true;
|
|
||||||
try {
|
|
||||||
final action = await showModalBottomSheet<ReminderAction>(
|
|
||||||
context: context,
|
|
||||||
useRootNavigator: true,
|
|
||||||
isScrollControlled: true,
|
|
||||||
backgroundColor: Colors.transparent,
|
|
||||||
builder: (sheetContext) {
|
|
||||||
return SafeArea(
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.all(AppSpacing.md),
|
|
||||||
child: ReminderActionSheet(
|
|
||||||
onSnooze: () {
|
|
||||||
Navigator.of(sheetContext).pop(ReminderAction.snooze10m);
|
|
||||||
},
|
|
||||||
onArchive: () {
|
|
||||||
Navigator.of(sheetContext).pop(ReminderAction.archive);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
if (action == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await _executor.handleAction(action: action, payload: payload);
|
|
||||||
} finally {
|
|
||||||
_isPresenting = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
typedef ReminderPresentationNow = DateTime Function();
|
|
||||||
|
|
||||||
class ReminderPresentationCoordinator {
|
|
||||||
final Duration _dedupeWindow;
|
|
||||||
final ReminderPresentationNow _now;
|
|
||||||
final Map<String, DateTime> _lastPresentedAtByEventId = <String, DateTime>{};
|
|
||||||
|
|
||||||
ReminderPresentationCoordinator({
|
|
||||||
Duration dedupeWindow = const Duration(seconds: 30),
|
|
||||||
ReminderPresentationNow? now,
|
|
||||||
}) : _dedupeWindow = dedupeWindow,
|
|
||||||
_now = now ?? DateTime.now;
|
|
||||||
|
|
||||||
bool shouldPresent({required String eventId, required bool isAppActive}) {
|
|
||||||
if (!isAppActive) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
final currentTime = _now();
|
|
||||||
final lastPresentedAt = _lastPresentedAtByEventId[eventId];
|
|
||||||
if (lastPresentedAt != null &&
|
|
||||||
currentTime.difference(lastPresentedAt) < _dedupeWindow) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
_lastPresentedAtByEventId[eventId] = currentTime;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import '../../../../../core/theme/design_tokens.dart';
|
|
||||||
import '../../../../../shared/widgets/app_button.dart';
|
|
||||||
|
|
||||||
class ReminderActionSheet extends StatelessWidget {
|
|
||||||
const ReminderActionSheet({
|
|
||||||
super.key,
|
|
||||||
required this.onSnooze,
|
|
||||||
required this.onArchive,
|
|
||||||
});
|
|
||||||
|
|
||||||
final VoidCallback onSnooze;
|
|
||||||
final VoidCallback onArchive;
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Container(
|
|
||||||
width: double.infinity,
|
|
||||||
padding: const EdgeInsets.all(AppSpacing.lg),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: AppColors.white,
|
|
||||||
borderRadius: BorderRadius.circular(AppRadius.xl),
|
|
||||||
border: Border.all(color: AppColors.borderSecondary),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
'提醒操作',
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: Theme.of(
|
|
||||||
context,
|
|
||||||
).textTheme.titleMedium?.copyWith(color: AppColors.slate900),
|
|
||||||
),
|
|
||||||
const SizedBox(height: AppSpacing.lg),
|
|
||||||
Row(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Expanded(
|
|
||||||
child: AppButton(
|
|
||||||
text: '稍后提醒',
|
|
||||||
isOutlined: true,
|
|
||||||
onPressed: onSnooze,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(width: AppSpacing.md),
|
|
||||||
Expanded(
|
|
||||||
child: AppButton(text: '归档', onPressed: onArchive),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -17,7 +17,6 @@ import 'features/auth/presentation/bloc/auth_state.dart';
|
|||||||
import 'features/calendar/data/services/calendar_service.dart';
|
import 'features/calendar/data/services/calendar_service.dart';
|
||||||
import 'features/calendar/data/services/calendar_repository.dart';
|
import 'features/calendar/data/services/calendar_repository.dart';
|
||||||
import 'features/calendar/reminders/reminder_action_executor.dart';
|
import 'features/calendar/reminders/reminder_action_executor.dart';
|
||||||
import 'features/calendar/reminders/ui/reminder_foreground_presenter.dart';
|
|
||||||
import 'features/calendar/ui/calendar_state_manager.dart';
|
import 'features/calendar/ui/calendar_state_manager.dart';
|
||||||
import 'features/chat/presentation/bloc/chat_bloc.dart';
|
import 'features/chat/presentation/bloc/chat_bloc.dart';
|
||||||
import 'features/settings/data/services/settings_user_cache.dart';
|
import 'features/settings/data/services/settings_user_cache.dart';
|
||||||
@@ -28,10 +27,6 @@ void main() async {
|
|||||||
await configureDependencies();
|
await configureDependencies();
|
||||||
await AppConstants.init();
|
await AppConstants.init();
|
||||||
final rootNavigatorKey = GlobalKey<NavigatorState>();
|
final rootNavigatorKey = GlobalKey<NavigatorState>();
|
||||||
final reminderForegroundPresenter = ReminderForegroundPresenter(
|
|
||||||
navigatorKey: rootNavigatorKey,
|
|
||||||
executor: sl<ReminderActionExecutor>(),
|
|
||||||
);
|
|
||||||
sl<LocalNotificationService>().bindActionHandler(({
|
sl<LocalNotificationService>().bindActionHandler(({
|
||||||
required action,
|
required action,
|
||||||
required payload,
|
required payload,
|
||||||
@@ -41,9 +36,6 @@ void main() async {
|
|||||||
payload: payload,
|
payload: payload,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
sl<LocalNotificationService>().bindInAppReminderHandler(
|
|
||||||
reminderForegroundPresenter.present,
|
|
||||||
);
|
|
||||||
await sl<LocalNotificationService>().initialize();
|
await sl<LocalNotificationService>().initialize();
|
||||||
|
|
||||||
final authBloc = sl<AuthBloc>();
|
final authBloc = sl<AuthBloc>();
|
||||||
|
|||||||
@@ -36,7 +36,6 @@ void main() {
|
|||||||
|
|
||||||
verifyNever(() => calendarService.getEventsForRange(any(), any()));
|
verifyNever(() => calendarService.getEventsForRange(any(), any()));
|
||||||
verifyNever(() => notificationService.rebuildUpcomingReminders(any()));
|
verifyNever(() => notificationService.rebuildUpcomingReminders(any()));
|
||||||
verifyNever(() => reminderActionExecutor.replayPendingActions());
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('fetches upcoming events after authenticated state', () async {
|
test('fetches upcoming events after authenticated state', () async {
|
||||||
@@ -46,9 +45,6 @@ void main() {
|
|||||||
when(
|
when(
|
||||||
() => notificationService.rebuildUpcomingReminders(any()),
|
() => notificationService.rebuildUpcomingReminders(any()),
|
||||||
).thenAnswer((_) async {});
|
).thenAnswer((_) async {});
|
||||||
when(
|
|
||||||
() => reminderActionExecutor.replayPendingActions(),
|
|
||||||
).thenAnswer((_) async {});
|
|
||||||
|
|
||||||
await bootstrapper.syncForAuthState(
|
await bootstrapper.syncForAuthState(
|
||||||
const AuthAuthenticated(
|
const AuthAuthenticated(
|
||||||
@@ -58,37 +54,5 @@ void main() {
|
|||||||
|
|
||||||
verify(() => calendarService.getEventsForRange(any(), any())).called(1);
|
verify(() => calendarService.getEventsForRange(any(), any())).called(1);
|
||||||
verify(() => notificationService.rebuildUpcomingReminders(any())).called(1);
|
verify(() => notificationService.rebuildUpcomingReminders(any())).called(1);
|
||||||
verify(() => reminderActionExecutor.replayPendingActions()).called(1);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('retries sync when previous bootstrap failed', () async {
|
|
||||||
when(
|
|
||||||
() => reminderActionExecutor.replayPendingActions(),
|
|
||||||
).thenThrow(Exception('offline'));
|
|
||||||
|
|
||||||
await bootstrapper.syncForAuthState(
|
|
||||||
const AuthAuthenticated(
|
|
||||||
user: AuthUser(id: 'u1', phone: 'a@test.com'),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
when(
|
|
||||||
() => reminderActionExecutor.replayPendingActions(),
|
|
||||||
).thenAnswer((_) async {});
|
|
||||||
when(
|
|
||||||
() => calendarService.getEventsForRange(any(), any()),
|
|
||||||
).thenAnswer((_) async => []);
|
|
||||||
when(
|
|
||||||
() => notificationService.rebuildUpcomingReminders(any()),
|
|
||||||
).thenAnswer((_) async {});
|
|
||||||
|
|
||||||
await bootstrapper.syncForAuthState(
|
|
||||||
const AuthAuthenticated(
|
|
||||||
user: AuthUser(id: 'u1', phone: 'a@test.com'),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
verify(() => reminderActionExecutor.replayPendingActions()).called(2);
|
|
||||||
verify(() => calendarService.getEventsForRange(any(), any())).called(1);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,78 +0,0 @@
|
|||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
import 'package:social_app/features/calendar/reminders/reminder_action_dedupe_store.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
const dedupeKey = 'calendar_reminder_action_dedupe_v1';
|
|
||||||
|
|
||||||
setUp(() {
|
|
||||||
SharedPreferences.setMockInitialValues({});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('markIfNew returns true first and false for duplicate id', () async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final store = ReminderActionDedupeStore(prefs);
|
|
||||||
|
|
||||||
expect(await store.markIfNew('action_1'), isTrue);
|
|
||||||
expect(await store.markIfNew('action_1'), isFalse);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('markIfNew dedupes after store re-initialization', () async {
|
|
||||||
final firstPrefs = await SharedPreferences.getInstance();
|
|
||||||
final firstStore = ReminderActionDedupeStore(firstPrefs);
|
|
||||||
|
|
||||||
expect(await firstStore.markIfNew('action_restart'), isTrue);
|
|
||||||
|
|
||||||
final secondPrefs = await SharedPreferences.getInstance();
|
|
||||||
final secondStore = ReminderActionDedupeStore(secondPrefs);
|
|
||||||
|
|
||||||
expect(await secondStore.markIfNew('action_restart'), isFalse);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('markIfNew trims history to max capacity', () async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final store = ReminderActionDedupeStore(prefs);
|
|
||||||
|
|
||||||
for (var i = 0; i < 513; i++) {
|
|
||||||
expect(await store.markIfNew('action_$i'), isTrue);
|
|
||||||
}
|
|
||||||
|
|
||||||
final stored = prefs.getStringList(dedupeKey)!;
|
|
||||||
expect(stored.length, 512);
|
|
||||||
expect(stored.first, 'action_1');
|
|
||||||
expect(stored.last, 'action_512');
|
|
||||||
});
|
|
||||||
|
|
||||||
test(
|
|
||||||
'markIfNew is serialized and does not return true twice in parallel',
|
|
||||||
() async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final store = ReminderActionDedupeStore(
|
|
||||||
prefs,
|
|
||||||
setStringList: (key, value) async {
|
|
||||||
await Future<void>.delayed(const Duration(milliseconds: 20));
|
|
||||||
return prefs.setStringList(key, value);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
final results = await Future.wait<bool>([
|
|
||||||
store.markIfNew('parallel_action'),
|
|
||||||
store.markIfNew('parallel_action'),
|
|
||||||
]);
|
|
||||||
|
|
||||||
expect(results.where((item) => item).length, 1);
|
|
||||||
expect(results.where((item) => !item).length, 1);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
test('markIfNew returns false when persistence write fails', () async {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final store = ReminderActionDedupeStore(
|
|
||||||
prefs,
|
|
||||||
setStringList: (key, value) async => false,
|
|
||||||
);
|
|
||||||
|
|
||||||
expect(await store.markIfNew('action_write_fail'), isFalse);
|
|
||||||
expect(prefs.getStringList(dedupeKey), isNull);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,13 +1,11 @@
|
|||||||
import 'package:flutter_test/flutter_test.dart';
|
import 'package:flutter_test/flutter_test.dart';
|
||||||
import 'package:mocktail/mocktail.dart';
|
import 'package:mocktail/mocktail.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
import 'package:social_app/core/notifications/local_notification_service.dart';
|
import 'package:social_app/core/notifications/local_notification_service.dart';
|
||||||
import 'package:social_app/features/calendar/data/models/schedule_item_model.dart';
|
import 'package:social_app/features/calendar/data/models/schedule_item_model.dart';
|
||||||
import 'package:social_app/features/calendar/data/services/calendar_service.dart';
|
import 'package:social_app/features/calendar/data/services/calendar_service.dart';
|
||||||
import 'package:social_app/features/calendar/reminders/models/reminder_action.dart';
|
import 'package:social_app/features/calendar/reminders/models/reminder_action.dart';
|
||||||
import 'package:social_app/features/calendar/reminders/models/reminder_payload.dart';
|
import 'package:social_app/features/calendar/reminders/models/reminder_payload.dart';
|
||||||
import 'package:social_app/features/calendar/reminders/reminder_action_executor.dart';
|
import 'package:social_app/features/calendar/reminders/reminder_action_executor.dart';
|
||||||
import 'package:social_app/features/calendar/reminders/reminder_outbox_store.dart';
|
|
||||||
|
|
||||||
class MockCalendarService extends Mock implements CalendarService {}
|
class MockCalendarService extends Mock implements CalendarService {}
|
||||||
|
|
||||||
@@ -17,19 +15,14 @@ class MockLocalNotificationService extends Mock
|
|||||||
void main() {
|
void main() {
|
||||||
late MockCalendarService calendarService;
|
late MockCalendarService calendarService;
|
||||||
late MockLocalNotificationService notificationService;
|
late MockLocalNotificationService notificationService;
|
||||||
late ReminderOutboxStore outboxStore;
|
|
||||||
late ReminderActionExecutor executor;
|
late ReminderActionExecutor executor;
|
||||||
|
|
||||||
setUp(() async {
|
setUp(() {
|
||||||
SharedPreferences.setMockInitialValues({});
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
calendarService = MockCalendarService();
|
calendarService = MockCalendarService();
|
||||||
notificationService = MockLocalNotificationService();
|
notificationService = MockLocalNotificationService();
|
||||||
outboxStore = ReminderOutboxStore(prefs);
|
|
||||||
executor = ReminderActionExecutor(
|
executor = ReminderActionExecutor(
|
||||||
calendarService: calendarService,
|
calendarService: calendarService,
|
||||||
notificationService: notificationService,
|
notificationService: notificationService,
|
||||||
outboxStore: outboxStore,
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -53,33 +46,6 @@ void main() {
|
|||||||
|
|
||||||
verify(() => notificationService.cancelEventReminder('evt_1')).called(1);
|
verify(() => notificationService.cancelEventReminder('evt_1')).called(1);
|
||||||
verify(() => calendarService.archiveEvent('evt_1')).called(1);
|
verify(() => calendarService.archiveEvent('evt_1')).called(1);
|
||||||
final pending = await outboxStore.listPending();
|
|
||||||
expect(pending, isEmpty);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('archive failure writes pending outbox item', () async {
|
|
||||||
when(
|
|
||||||
() => notificationService.cancelEventReminder('evt_1'),
|
|
||||||
).thenAnswer((_) async {});
|
|
||||||
when(
|
|
||||||
() => calendarService.archiveEvent('evt_1'),
|
|
||||||
).thenThrow(Exception('offline'));
|
|
||||||
|
|
||||||
await executor.handleAction(
|
|
||||||
action: ReminderAction.archive,
|
|
||||||
payload: ReminderPayload(
|
|
||||||
eventId: 'evt_1',
|
|
||||||
title: 'sync',
|
|
||||||
startAt: DateTime.parse('2026-03-18T16:00:00+08:00'),
|
|
||||||
timezone: 'Asia/Shanghai',
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
final pending = await outboxStore.listPending();
|
|
||||||
expect(pending.length, 1);
|
|
||||||
expect(pending.first.eventId, 'evt_1');
|
|
||||||
expect(pending.first.state, ReminderOutboxState.pending);
|
|
||||||
verify(() => calendarService.archiveEvent('evt_1')).called(1);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('snooze reschedules +10m when event not expired', () async {
|
test('snooze reschedules +10m when event not expired', () async {
|
||||||
@@ -151,24 +117,4 @@ void main() {
|
|||||||
verify(() => calendarService.archiveEvent('evt_fallback')).called(1);
|
verify(() => calendarService.archiveEvent('evt_fallback')).called(1);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
test('replay keeps pending item when targetStatus is not archived', () async {
|
|
||||||
const opId = 'op_non_archived';
|
|
||||||
await outboxStore.enqueue(
|
|
||||||
ReminderOutboxItem(
|
|
||||||
opId: opId,
|
|
||||||
eventId: 'evt_1',
|
|
||||||
action: ReminderAction.archive,
|
|
||||||
targetStatus: 'ignored',
|
|
||||||
occurredAt: DateTime.parse('2026-03-18T16:00:00+08:00'),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
await executor.replayPendingActions();
|
|
||||||
|
|
||||||
final pending = await outboxStore.listPending();
|
|
||||||
expect(pending.length, 1);
|
|
||||||
expect(pending.first.opId, opId);
|
|
||||||
verifyNever(() => calendarService.archiveEvent(any()));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,41 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
import 'package:social_app/features/calendar/reminders/ui/widgets/reminder_action_sheet.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
Future<void> pumpSheet(
|
|
||||||
WidgetTester tester, {
|
|
||||||
required VoidCallback onSnooze,
|
|
||||||
required VoidCallback onArchive,
|
|
||||||
}) {
|
|
||||||
return tester.pumpWidget(
|
|
||||||
MaterialApp(
|
|
||||||
home: Scaffold(
|
|
||||||
body: ReminderActionSheet(onSnooze: onSnooze, onArchive: onArchive),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
testWidgets('tap snooze button triggers onSnooze', (tester) async {
|
|
||||||
var snoozed = false;
|
|
||||||
|
|
||||||
await pumpSheet(tester, onSnooze: () => snoozed = true, onArchive: () {});
|
|
||||||
|
|
||||||
await tester.tap(find.text('稍后提醒'));
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(snoozed, isTrue);
|
|
||||||
});
|
|
||||||
|
|
||||||
testWidgets('tap archive button triggers onArchive', (tester) async {
|
|
||||||
var archived = false;
|
|
||||||
|
|
||||||
await pumpSheet(tester, onSnooze: () {}, onArchive: () => archived = true);
|
|
||||||
|
|
||||||
await tester.tap(find.text('归档'));
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(archived, isTrue);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,123 +0,0 @@
|
|||||||
import 'dart:async';
|
|
||||||
|
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
import 'package:social_app/features/calendar/reminders/reminder_cold_start_queue.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
test('replays queued actions in enqueue order', () async {
|
|
||||||
final queue = ReminderColdStartQueue();
|
|
||||||
final events = <String>[];
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
await Future<void>.delayed(const Duration(milliseconds: 20));
|
|
||||||
events.add('first');
|
|
||||||
});
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('second');
|
|
||||||
});
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('third');
|
|
||||||
});
|
|
||||||
|
|
||||||
await queue.replay();
|
|
||||||
|
|
||||||
expect(events, <String>['first', 'second', 'third']);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('single failure does not block following queued actions', () async {
|
|
||||||
final queue = ReminderColdStartQueue();
|
|
||||||
final events = <String>[];
|
|
||||||
final errors = <Object>[];
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('before');
|
|
||||||
});
|
|
||||||
queue.enqueue(() async {
|
|
||||||
throw StateError('boom');
|
|
||||||
});
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('after');
|
|
||||||
});
|
|
||||||
|
|
||||||
final observableQueue = ReminderColdStartQueue(
|
|
||||||
onTaskError: (Object error, StackTrace _) {
|
|
||||||
errors.add(error);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
observableQueue.enqueue(() async {
|
|
||||||
throw StateError('boom_observable');
|
|
||||||
});
|
|
||||||
|
|
||||||
await queue.replay();
|
|
||||||
await observableQueue.replay();
|
|
||||||
|
|
||||||
expect(events, <String>['before', 'after']);
|
|
||||||
expect(errors.length, 1);
|
|
||||||
expect(errors.first, isA<StateError>());
|
|
||||||
});
|
|
||||||
|
|
||||||
test('concurrent replay calls join the same in-flight replay', () async {
|
|
||||||
final queue = ReminderColdStartQueue();
|
|
||||||
final taskGate = Completer<void>();
|
|
||||||
var runCount = 0;
|
|
||||||
var secondReplayCompleted = false;
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
runCount += 1;
|
|
||||||
await taskGate.future;
|
|
||||||
});
|
|
||||||
|
|
||||||
final firstReplay = queue.replay();
|
|
||||||
final secondReplay = queue.replay().then((_) {
|
|
||||||
secondReplayCompleted = true;
|
|
||||||
});
|
|
||||||
|
|
||||||
await Future<void>.delayed(const Duration(milliseconds: 10));
|
|
||||||
expect(secondReplayCompleted, isFalse);
|
|
||||||
|
|
||||||
taskGate.complete();
|
|
||||||
await Future.wait(<Future<void>>[firstReplay, secondReplay]);
|
|
||||||
|
|
||||||
expect(runCount, 1);
|
|
||||||
expect(secondReplayCompleted, isTrue);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('replay on empty queue does not block future enqueued tasks', () async {
|
|
||||||
final queue = ReminderColdStartQueue();
|
|
||||||
final events = <String>[];
|
|
||||||
|
|
||||||
await queue.replay();
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('after_empty_replay');
|
|
||||||
});
|
|
||||||
|
|
||||||
await queue.replay();
|
|
||||||
|
|
||||||
expect(events, <String>['after_empty_replay']);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('task-triggered replay reuses in-flight replay and completes', () async {
|
|
||||||
final queue = ReminderColdStartQueue();
|
|
||||||
final events = <String>[];
|
|
||||||
var nestedReplayCompleted = false;
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('first');
|
|
||||||
queue.replay().then((_) {
|
|
||||||
nestedReplayCompleted = true;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
queue.enqueue(() async {
|
|
||||||
events.add('second');
|
|
||||||
});
|
|
||||||
|
|
||||||
await queue.replay();
|
|
||||||
await Future<void>.delayed(Duration.zero);
|
|
||||||
|
|
||||||
expect(events, <String>['first', 'second']);
|
|
||||||
expect(nestedReplayCompleted, isTrue);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
import 'package:social_app/features/calendar/data/models/schedule_item_model.dart';
|
|
||||||
import 'package:social_app/features/calendar/reminders/reminder_overlap_policy.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
final policy = ReminderOverlapPolicy();
|
|
||||||
|
|
||||||
test('groups reminders in same minute bucket', () {
|
|
||||||
final now = DateTime(2026, 3, 18, 15, 40, 0);
|
|
||||||
final eventA = ScheduleItemModel(
|
|
||||||
id: 'a',
|
|
||||||
ownerId: 'u1',
|
|
||||||
title: 'A',
|
|
||||||
startAt: DateTime(2026, 3, 18, 16, 0, 0),
|
|
||||||
endAt: DateTime(2026, 3, 18, 17, 0, 0),
|
|
||||||
metadata: ScheduleMetadata(reminderMinutes: 15),
|
|
||||||
);
|
|
||||||
final eventB = ScheduleItemModel(
|
|
||||||
id: 'b',
|
|
||||||
ownerId: 'u1',
|
|
||||||
title: 'B',
|
|
||||||
startAt: DateTime(2026, 3, 18, 16, 0, 20),
|
|
||||||
endAt: DateTime(2026, 3, 18, 17, 0, 0),
|
|
||||||
metadata: ScheduleMetadata(reminderMinutes: 15),
|
|
||||||
);
|
|
||||||
|
|
||||||
final groups = policy.groupByMinute([eventA, eventB], now: now);
|
|
||||||
expect(groups.length, 1);
|
|
||||||
expect(groups.first.events.length, 2);
|
|
||||||
expect(groups.first.isAggregate, isTrue);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('returns compensation fire time when remindAt already passed', () {
|
|
||||||
final now = DateTime(2026, 3, 18, 15, 50, 0);
|
|
||||||
final event = ScheduleItemModel(
|
|
||||||
id: 'a',
|
|
||||||
ownerId: 'u1',
|
|
||||||
title: 'A',
|
|
||||||
startAt: DateTime(2026, 3, 18, 16, 0, 0),
|
|
||||||
endAt: DateTime(2026, 3, 18, 16, 30, 0),
|
|
||||||
metadata: ScheduleMetadata(reminderMinutes: 15),
|
|
||||||
);
|
|
||||||
|
|
||||||
final fireAt = policy.resolveFirstFireAt(event, now: now);
|
|
||||||
expect(fireAt, isNotNull);
|
|
||||||
expect(fireAt!.isAfter(now), isTrue);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
import 'package:flutter_test/flutter_test.dart';
|
|
||||||
import 'package:social_app/features/calendar/reminders/ui/reminder_presentation_coordinator.dart';
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
test('blocks foreground presentation when app is not active', () {
|
|
||||||
final coordinator = ReminderPresentationCoordinator();
|
|
||||||
|
|
||||||
final hiddenDecision = coordinator.shouldPresent(
|
|
||||||
eventId: 'event_1',
|
|
||||||
isAppActive: false,
|
|
||||||
);
|
|
||||||
final activeDecision = coordinator.shouldPresent(
|
|
||||||
eventId: 'event_1',
|
|
||||||
isAppActive: true,
|
|
||||||
);
|
|
||||||
|
|
||||||
expect(hiddenDecision, isFalse);
|
|
||||||
expect(activeDecision, isTrue);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('suppresses duplicate foreground presentation inside dedupe window', () {
|
|
||||||
var fakeNow = DateTime(2026, 3, 19, 10, 0, 0);
|
|
||||||
final coordinator = ReminderPresentationCoordinator(
|
|
||||||
dedupeWindow: const Duration(seconds: 30),
|
|
||||||
now: () => fakeNow,
|
|
||||||
);
|
|
||||||
|
|
||||||
final first = coordinator.shouldPresent(
|
|
||||||
eventId: 'event_42',
|
|
||||||
isAppActive: true,
|
|
||||||
);
|
|
||||||
fakeNow = fakeNow.add(const Duration(seconds: 10));
|
|
||||||
final second = coordinator.shouldPresent(
|
|
||||||
eventId: 'event_42',
|
|
||||||
isAppActive: true,
|
|
||||||
);
|
|
||||||
|
|
||||||
expect(first, isTrue);
|
|
||||||
expect(second, isFalse);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('allows same event again after dedupe window expires', () {
|
|
||||||
var fakeNow = DateTime(2026, 3, 19, 10, 0, 0);
|
|
||||||
final coordinator = ReminderPresentationCoordinator(
|
|
||||||
dedupeWindow: const Duration(seconds: 30),
|
|
||||||
now: () => fakeNow,
|
|
||||||
);
|
|
||||||
|
|
||||||
expect(
|
|
||||||
coordinator.shouldPresent(eventId: 'event_42', isAppActive: true),
|
|
||||||
isTrue,
|
|
||||||
);
|
|
||||||
fakeNow = fakeNow.add(const Duration(seconds: 31));
|
|
||||||
|
|
||||||
expect(
|
|
||||||
coordinator.shouldPresent(eventId: 'event_42', isAppActive: true),
|
|
||||||
isTrue,
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user