From 2252a7418638763048989612f2eb6e987e8174e3 Mon Sep 17 00:00:00 2001 From: provokateurin Date: Wed, 30 Oct 2024 11:41:29 +0100 Subject: [PATCH] feat(talk_app): Implement swipe-to-reply Signed-off-by: provokateurin --- .../packages/talk_app/lib/src/pages/room.dart | 13 +++ .../talk_app/test/room_page_test.dart | 104 ++++++++++++++++++ 2 files changed, 117 insertions(+) diff --git a/packages/neon_framework/packages/talk_app/lib/src/pages/room.dart b/packages/neon_framework/packages/talk_app/lib/src/pages/room.dart index 85c8c25951f..e18e910892e 100644 --- a/packages/neon_framework/packages/talk_app/lib/src/pages/room.dart +++ b/packages/neon_framework/packages/talk_app/lib/src/pages/room.dart @@ -123,6 +123,19 @@ class _TalkRoomPageState extends State { previousChatMessage: previousMessage, ); + if (canReplyToMessage(room, message)) { + child = Dismissible( + key: Key(message.id.toString()), + confirmDismiss: (_) async { + bloc.setReplyChatMessage(message); + + // We don't use the real dismiss feature as we don't want the widget to be removed from the list + return false; + }, + child: child, + ); + } + if (previousMessage == null || (tz.local.translate(previousMessage.timestamp * 1000) ~/ _millisecondsPerDay) != (tz.local.translate(message.timestamp * 1000) ~/ _millisecondsPerDay)) { diff --git a/packages/neon_framework/packages/talk_app/test/room_page_test.dart b/packages/neon_framework/packages/talk_app/test/room_page_test.dart index 00d50a564bf..2e67ba1f6d9 100644 --- a/packages/neon_framework/packages/talk_app/test/room_page_test.dart +++ b/packages/neon_framework/packages/talk_app/test/room_page_test.dart @@ -219,4 +219,108 @@ void main() { expect(find.byIcon(Icons.emoji_emotions_outlined), findsNothing); await expectLater(find.byType(TestApp), matchesGoldenFile('goldens/room_page_read_only.png')); }); + + group('Swipe-to-reply', () { + late spreed.ChatMessageWithParent chatMessage; + + setUp(() { + chatMessage = MockChatMessageWithParent(); + when(() => chatMessage.timestamp).thenReturn(0); + when(() => chatMessage.actorId).thenReturn('test'); + when(() => chatMessage.actorType).thenReturn(spreed.ActorType.users); + when(() => chatMessage.actorDisplayName).thenReturn('test'); + when(() => chatMessage.messageType).thenReturn(spreed.MessageType.comment); + when(() => chatMessage.message).thenReturn('abc'); + when(() => chatMessage.reactions).thenReturn(BuiltMap()); + when(() => chatMessage.messageParameters).thenReturn(BuiltMap()); + when(() => chatMessage.id).thenReturn(0); + when(() => chatMessage.isReplyable).thenReturn(true); + + when(() => bloc.messages).thenAnswer( + (_) => BehaviorSubject.seeded( + Result.success( + BuiltList([ + chatMessage, + ]), + ), + ), + ); + + when(() => bloc.reactions).thenAnswer((_) => BehaviorSubject.seeded(BuiltMap())); + }); + + testWidgets('Allowed', (tester) async { + final account = MockAccount(); + + await tester.pumpWidgetWithAccessibility( + TestApp( + localizationsDelegates: TalkLocalizations.localizationsDelegates, + supportedLocales: TalkLocalizations.supportedLocales, + appThemes: const [ + TalkTheme(), + ], + providers: [ + Provider.value(value: account), + NeonProvider.value(value: bloc), + NeonProvider.value(value: referencesBloc), + ], + child: const TalkRoomPage(), + ), + ); + + expect(find.byType(Dismissible), findsOne); + + await tester.drag(find.byType(TalkCommentMessage), const Offset(1000, 0)); + await tester.pumpAndSettle(); + verify(() => bloc.setReplyChatMessage(chatMessage)).called(1); + }); + + testWidgets('Read-only', (tester) async { + when(() => room.readOnly).thenReturn(1); + + final account = MockAccount(); + + await tester.pumpWidgetWithAccessibility( + TestApp( + localizationsDelegates: TalkLocalizations.localizationsDelegates, + supportedLocales: TalkLocalizations.supportedLocales, + appThemes: const [ + TalkTheme(), + ], + providers: [ + Provider.value(value: account), + NeonProvider.value(value: bloc), + NeonProvider.value(value: referencesBloc), + ], + child: const TalkRoomPage(), + ), + ); + + expect(find.byType(Dismissible), findsNothing); + }); + + testWidgets('No permission', (tester) async { + when(() => room.permissions).thenReturn(0); + + final account = MockAccount(); + + await tester.pumpWidgetWithAccessibility( + TestApp( + localizationsDelegates: TalkLocalizations.localizationsDelegates, + supportedLocales: TalkLocalizations.supportedLocales, + appThemes: const [ + TalkTheme(), + ], + providers: [ + Provider.value(value: account), + NeonProvider.value(value: bloc), + NeonProvider.value(value: referencesBloc), + ], + child: const TalkRoomPage(), + ), + ); + + expect(find.byType(Dismissible), findsNothing); + }); + }); }