Solian/lib/controllers/chat_events_controller.dart

147 lines
4.2 KiB
Dart
Raw Normal View History

2024-09-16 19:50:49 +08:00
import 'dart:math' as math;
import 'package:get/get.dart';
import 'package:solian/models/channel.dart';
import 'package:solian/models/event.dart';
2024-09-14 00:30:33 +08:00
import 'package:solian/providers/database/database.dart';
import 'package:solian/providers/database/services/messages.dart';
class ChatEventController {
2024-09-14 00:30:33 +08:00
late final MessagesFetchingProvider src;
2024-09-14 00:30:33 +08:00
final RxList<LocalMessageEventTableData> currentEvents =
RxList.empty(growable: true);
final RxInt totalEvents = 0.obs;
2024-09-15 10:55:27 +08:00
final RxBool isLoading = true.obs;
Channel? channel;
2024-06-28 02:49:28 +08:00
String? scope;
2024-08-02 00:54:19 +08:00
Future<void> initialize() async {
2024-09-14 00:30:33 +08:00
src = Get.find();
currentEvents.clear();
}
2024-09-14 00:30:33 +08:00
Future<LocalMessageEventTableData?> getEvent(int id) async {
2024-06-28 04:54:03 +08:00
if (channel == null || scope == null) return null;
2024-09-14 00:30:33 +08:00
return await src.getEvent(id, channel!, scope: scope!);
2024-06-28 02:49:28 +08:00
}
2024-09-15 10:55:27 +08:00
Future<void> getInitialEvents(Channel channel, String scope) async {
2024-06-28 02:49:28 +08:00
this.channel = channel;
this.scope = scope;
2024-09-16 19:50:49 +08:00
const firstTake = 20;
const furtherTake = 100;
isLoading.value = true;
2024-09-16 19:50:49 +08:00
await syncLocal(channel, take: firstTake);
isLoading.value = false;
// Take a small range of messages to check is local database up to date
var isUpToDate = true;
final result =
await src.pullRemoteEvents(channel, scope: scope, take: firstTake);
totalEvents.value = result?.$2 ?? 0;
if ((result?.$1.length ?? 0) > 0) {
final minId = result!.$1.map((x) => x.id).reduce(math.min);
isUpToDate = await src.getEventFromLocal(minId) != null;
}
syncLocal(channel, take: firstTake);
2024-09-15 18:25:04 +08:00
2024-09-16 19:50:49 +08:00
if (!isUpToDate) {
// Loading more content due to isn't up to date
final result =
await src.pullRemoteEvents(channel, scope: scope, take: furtherTake);
2024-06-28 04:54:03 +08:00
totalEvents.value = result?.$2 ?? 0;
2024-09-16 19:50:49 +08:00
syncLocal(channel, take: furtherTake);
}
}
Future<void> loadEvents(Channel channel, String scope) async {
2024-09-15 18:25:04 +08:00
const take = 20;
final offset = currentEvents.length;
isLoading.value = true;
2024-09-15 18:25:04 +08:00
await syncLocal(channel, take: take, offset: offset);
2024-09-16 19:50:49 +08:00
src
.pullRemoteEvents(channel, scope: scope, take: take, offset: offset)
.then((result) {
2024-06-28 04:54:03 +08:00
totalEvents.value = result?.$2 ?? 0;
2024-09-15 18:25:04 +08:00
syncLocal(channel, take: take, offset: offset);
});
isLoading.value = false;
}
2024-09-15 18:25:04 +08:00
Future<bool> syncLocal(Channel channel,
{required int take, int offset = 0}) async {
final data = await src.listEvents(channel, take: take, offset: offset);
2024-09-16 19:50:49 +08:00
if (currentEvents.length >= offset + take) {
currentEvents.replaceRange(offset, offset + take, data);
} else {
currentEvents.insertAll(currentEvents.length, data);
}
2024-06-28 14:56:57 +08:00
for (final x in data.reversed) {
applyEvent(x);
}
return true;
}
receiveEvent(Event remote) async {
2024-09-14 00:30:33 +08:00
LocalMessageEventTableData entry;
2024-09-15 18:25:04 +08:00
entry = await src.receiveEvent(remote);
2024-09-15 10:55:27 +08:00
totalEvents.value++;
2024-07-06 17:12:57 +08:00
insertEvent(entry);
2024-06-28 14:56:57 +08:00
applyEvent(entry);
}
2024-09-14 00:30:33 +08:00
void insertEvent(LocalMessageEventTableData entry) {
if (entry.channelId != channel?.id) return;
2024-09-14 00:30:33 +08:00
final idx = currentEvents.indexWhere(
(x) => x.data!.uuid == entry.data!.uuid,
);
if (idx != -1) {
currentEvents[idx] = entry;
} else {
currentEvents.insert(0, entry);
}
2024-07-06 17:12:57 +08:00
}
2024-09-14 00:30:33 +08:00
void applyEvent(LocalMessageEventTableData entry) {
2024-07-06 17:12:57 +08:00
if (entry.channelId != channel?.id) return;
2024-09-14 00:30:33 +08:00
switch (entry.data!.type) {
case 'messages.edit':
2024-09-14 00:30:33 +08:00
final body = EventMessageBody.fromJson(entry.data!.body);
if (body.relatedEvent != null) {
final idx =
2024-09-14 00:30:33 +08:00
currentEvents.indexWhere((x) => x.data!.id == body.relatedEvent);
if (idx != -1) {
2024-09-14 00:30:33 +08:00
currentEvents[idx].data!.body = entry.data!.body;
currentEvents[idx].data!.updatedAt = entry.data!.updatedAt;
}
}
case 'messages.delete':
2024-09-14 00:30:33 +08:00
final body = EventMessageBody.fromJson(entry.data!.body);
if (body.relatedEvent != null) {
currentEvents.removeWhere((x) => x.id == body.relatedEvent);
}
}
}
2024-09-14 00:30:33 +08:00
Future<void> addPendingEvent(Event info) async {
currentEvents.insert(
0,
2024-09-14 00:30:33 +08:00
LocalMessageEventTableData(
id: info.id,
channelId: info.channelId,
createdAt: DateTime.now(),
data: info,
),
);
}
}