From c4ca1ab79c147e8388109dcc7cf7d8b6c1ac96a2 Mon Sep 17 00:00:00 2001 From: Zixuan James Li Date: Wed, 26 Jun 2024 16:42:19 -0400 Subject: [PATCH 1/2] message: Handle message deletion event. When handling a message deletion event, we just simply remove the messages from the per account store and wipe them from the message list view if they are present. Because message deletion can affect what items are to be seen (date separator, recipient header, etc.), if the view has any message deleted, a `_reprocessAll` call is triggered. It would be ideal to just removed the affected items for better performance. Fixes #120. Signed-off-by: Zixuan James Li --- lib/model/message.dart | 7 +++- lib/model/message_list.dart | 30 +++++++++++++++ test/example_data.dart | 15 ++++++++ test/model/message_list_test.dart | 61 +++++++++++++++++++++++++++++++ test/model/message_test.dart | 33 +++++++++++++++++ 5 files changed, 145 insertions(+), 1 deletion(-) diff --git a/lib/model/message.dart b/lib/model/message.dart index 4ccf45d2ca..75e22f6bde 100644 --- a/lib/model/message.dart +++ b/lib/model/message.dart @@ -175,7 +175,12 @@ class MessageStoreImpl with MessageStore { } void handleDeleteMessageEvent(DeleteMessageEvent event) { - // TODO handle DeleteMessageEvent, particularly in MessageListView + for (final messageId in event.messageIds) { + messages.remove(messageId); + } + for (final view in _messageListViews) { + view.handleDeleteMessageEvent(event); + } } void handleUpdateMessageFlagsEvent(UpdateMessageFlagsEvent event) { diff --git a/lib/model/message_list.dart b/lib/model/message_list.dart index 117b30e614..e0f59d99c7 100644 --- a/lib/model/message_list.dart +++ b/lib/model/message_list.dart @@ -155,6 +155,30 @@ mixin _MessageSequence { _processMessage(messages.length - 1); } + /// Removes the given messages, if present. + /// + /// Returns true if at least one message was present, false otherwise. + /// If none of [messageIds] are found, this is a no-op. + bool _removeMessagesById(Iterable messageIds) { + final messagesToRemoveById = {}; + final contentToRemove = Set.identity(); + for (final messageId in messageIds) { + final index = _findMessageWithId(messageId); + if (index == -1) continue; + messagesToRemoveById.add(messageId); + contentToRemove.add(contents[index]); + } + if (messagesToRemoveById.isEmpty) return false; + + assert(contents.length == messages.length); + messages.removeWhere((message) => messagesToRemoveById.contains(message.id)); + contents.removeWhere((content) => contentToRemove.contains(content)); + assert(contents.length == messages.length); + _reprocessAll(); + + return true; + } + void _insertAllMessages(int index, Iterable toInsert) { // TODO parse/process messages in smaller batches, to not drop frames. // On a Pixel 5, a batch of 100 messages takes ~15-20ms in _insertAllMessages. @@ -429,6 +453,12 @@ class MessageListView with ChangeNotifier, _MessageSequence { } } + void handleDeleteMessageEvent(DeleteMessageEvent event) { + if (_removeMessagesById(event.messageIds)) { + notifyListeners(); + } + } + /// Add [MessageEvent.message] to this view, if it belongs here. void handleMessageEvent(MessageEvent event) { final message = event.message; diff --git a/test/example_data.dart b/test/example_data.dart index 36fc60fec0..8de15b9159 100644 --- a/test/example_data.dart +++ b/test/example_data.dart @@ -388,6 +388,21 @@ const _unreadMsgs = unreadMsgs; // Events. // +DeleteMessageEvent deleteMessageEvent(List messages) { + assert(messages.isNotEmpty); + final streamId = messages.first.streamId; + final topic = messages.first.topic; + assert(messages.every((m) => m.streamId == streamId)); + assert(messages.every((m) => m.topic == topic)); + return DeleteMessageEvent( + id: 0, + messageIds: messages.map((message) => message.id).toList(), + messageType: MessageType.stream, + streamId: messages[0].streamId, + topic: messages[0].topic, + ); +} + UpdateMessageEvent updateMessageEditEvent( Message origMessage, { int? userId = -1, // null means null; default is [selfUser.userId] diff --git a/test/model/message_list_test.dart b/test/model/message_list_test.dart index 3d61ffa795..1ab502e380 100644 --- a/test/model/message_list_test.dart +++ b/test/model/message_list_test.dart @@ -269,6 +269,67 @@ void main() { check(model).fetched.isFalse(); }); + group('DeleteMessageEvent', () { + final stream = eg.stream(); + final messages = List.generate(30, (i) => eg.streamMessage(stream: stream)); + + test('in narrow', () async { + await prepare(narrow: StreamNarrow(stream.streamId)); + await prepareMessages(foundOldest: true, messages: messages); + + check(model).messages.length.equals(30); + await store.handleEvent(eg.deleteMessageEvent(messages.sublist(0, 10))); + checkNotifiedOnce(); + check(model).messages.length.equals(20); + }); + + test('not all in narrow', () async { + await prepare(narrow: StreamNarrow(stream.streamId)); + await prepareMessages(foundOldest: true, messages: messages.sublist(5)); + + check(model).messages.length.equals(25); + await store.handleEvent(eg.deleteMessageEvent(messages.sublist(0, 10))); + checkNotifiedOnce(); + check(model).messages.length.equals(20); + }); + + test('not in narrow', () async { + await prepare(narrow: StreamNarrow(stream.streamId)); + await prepareMessages(foundOldest: true, messages: messages.sublist(5)); + + check(model).messages.length.equals(25); + await store.handleEvent(eg.deleteMessageEvent(messages.sublist(0, 5))); + checkNotNotified(); + check(model).messages.length.equals(25); + }); + + test('complete message deletion', () async { + await prepare(narrow: StreamNarrow(stream.streamId)); + await prepareMessages(foundOldest: true, messages: messages.sublist(0, 25)); + + check(model).messages.length.equals(25); + await store.handleEvent(eg.deleteMessageEvent(messages)); + checkNotifiedOnce(); + check(model).messages.length.equals(0); + }); + + test('non-consecutive message deletion', () async { + await prepare(narrow: StreamNarrow(stream.streamId)); + await prepareMessages(foundOldest: true, messages: messages); + final messagesToDelete = messages.sublist(2, 5) + messages.sublist(10, 15); + + check(model).messages.length.equals(30); + await store.handleEvent(eg.deleteMessageEvent(messagesToDelete)); + checkNotifiedOnce(); + final expected = [ + ...messages.sublist(0, 2), + ...messages.sublist(5, 10), + ...messages.sublist(15)].map((message) => message.id); + check(model).messages.length.equals(expected.length); + check(model.messages.map((message) => message.id)).deepEquals(expected); + }); + }); + group('notifyListenersIfMessagePresent', () { test('message present', () async { await prepare(narrow: const CombinedFeedNarrow()); diff --git a/test/model/message_test.dart b/test/model/message_test.dart index 10e4334fcf..248d9fcfc3 100644 --- a/test/model/message_test.dart +++ b/test/model/message_test.dart @@ -375,6 +375,39 @@ void main() { }); }); + group('handleDeleteMessageEvent', () { + test('delete an unknown message', () async { + final message1 = eg.streamMessage(); + final message2 = eg.streamMessage(); + await prepare(); + await prepareMessages([message1]); + await store.handleEvent(eg.deleteMessageEvent([message2])); + checkNotNotified(); + check(store).messages.values.single.id.equals(message1.id); + }); + + test('delete messages', () async { + final message1 = eg.streamMessage(); + final message2 = eg.streamMessage(); + await prepare(); + await prepareMessages([message1, message2]); + await store.handleEvent(eg.deleteMessageEvent([message1, message2])); + checkNotifiedOnce(); + check(store).messages.isEmpty(); + }); + + test('delete an unknown message with a known message', () async { + final message1 = eg.streamMessage(); + final message2 = eg.streamMessage(); + final message3 = eg.streamMessage(); + await prepare(); + await prepareMessages([message1, message2]); + await store.handleEvent(eg.deleteMessageEvent([message2, message3])); + checkNotifiedOnce(); + check(store).messages.values.single.id.equals(message1.id); + }); + }); + group('handleReactionEvent', () { test('add reaction', () async { final originalMessage = eg.streamMessage(reactions: []); From 1f9756379d07a490576ea6c3748f89345295061b Mon Sep 17 00:00:00 2001 From: Greg Price Date: Mon, 1 Jul 2024 17:41:45 -0700 Subject: [PATCH 2/2] message test [nfc]: Tighten message-deletion test slightly The `length` check isn't needed because it's covered by the `deepEquals`. (It was needed in a previous draft of this test). Then we can also inline `expected`, and reformat slightly. --- test/model/message_list_test.dart | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/test/model/message_list_test.dart b/test/model/message_list_test.dart index 1ab502e380..b7c1086f96 100644 --- a/test/model/message_list_test.dart +++ b/test/model/message_list_test.dart @@ -321,12 +321,11 @@ void main() { check(model).messages.length.equals(30); await store.handleEvent(eg.deleteMessageEvent(messagesToDelete)); checkNotifiedOnce(); - final expected = [ + check(model.messages.map((message) => message.id)).deepEquals([ ...messages.sublist(0, 2), ...messages.sublist(5, 10), - ...messages.sublist(15)].map((message) => message.id); - check(model).messages.length.equals(expected.length); - check(model.messages.map((message) => message.id)).deepEquals(expected); + ...messages.sublist(15), + ].map((message) => message.id)); }); });