diff --git a/.changes/2498-chat-ng-reactions.md b/.changes/2498-chat-ng-reactions.md new file mode 100644 index 000000000000..9710766bc9c7 --- /dev/null +++ b/.changes/2498-chat-ng-reactions.md @@ -0,0 +1,5 @@ +[Labs] Chat-NG : + +- Added the ability to do reaction on message (by long press). +- You can also see reactions details (By long press on reaction) of messages. +- Moved `Edited` indicator inside chat bubble and UX improvements. diff --git a/app/ios/Podfile.lock b/app/ios/Podfile.lock index dfcad4ba3386..c4dc047d1dca 100644 --- a/app/ios/Podfile.lock +++ b/app/ios/Podfile.lock @@ -324,4 +324,4 @@ SPEC CHECKSUMS: PODFILE CHECKSUM: d2243213672c3c48aae53c36642ba411a6be7309 -COCOAPODS: 1.15.2 +COCOAPODS: 1.16.2 diff --git a/app/lib/features/chat_ng/actions/reaction_selection_action.dart b/app/lib/features/chat_ng/actions/reaction_selection_action.dart new file mode 100644 index 000000000000..1670e52d75a0 --- /dev/null +++ b/app/lib/features/chat_ng/actions/reaction_selection_action.dart @@ -0,0 +1,119 @@ +import 'dart:ui'; + +import 'package:acter/features/chat_ng/widgets/reactions/reaction_selector.dart'; +import 'package:flutter/material.dart'; + +// reaction selector action on chat message +void reactionSelectionAction({ + required BuildContext context, + required Widget messageWidget, + required bool isMe, + required String roomId, + required String messageId, +}) { + final RenderBox box = context.findRenderObject() as RenderBox; + final Offset position = box.localToGlobal(Offset.zero); + final messageSize = box.size; + + showGeneralDialog( + context: context, + barrierDismissible: true, + barrierLabel: '', + barrierColor: Colors.transparent, + transitionDuration: const Duration(milliseconds: 200), + pageBuilder: (context, animation, secondaryAnimation) { + return Stack( + children: [ + _ReactionOverlay( + animation: animation, + child: const SizedBox.expand(), + ), + Positioned( + left: position.dx, + top: position.dy, + width: messageSize.width, + child: messageWidget, + ), + Positioned( + left: position.dx, + top: position.dy - 60, + child: _AnimatedReactionSelector( + animation: animation, + messageId: messageId, + child: ReactionSelector( + isMe: isMe, + messageId: messageId, + roomId: roomId, + ), + ), + ), + ], + ); + }, + ); +} + +class _ReactionOverlay extends StatelessWidget { + final Animation animation; + final Widget child; + + const _ReactionOverlay({ + required this.animation, + required this.child, + }); + + @override + Widget build(BuildContext context) { + return GestureDetector( + onTap: () => Navigator.pop(context), + child: AnimatedBuilder( + animation: animation, + builder: (context, child) { + return BackdropFilter( + filter: ImageFilter.blur( + sigmaX: 8 * animation.value, + sigmaY: 8 * animation.value, + ), + child: Container( + color: Colors.black.withOpacity(0.1 * animation.value), + child: child, + ), + ); + }, + child: child, + ), + ); + } +} + +class _AnimatedReactionSelector extends StatelessWidget { + final Animation animation; + final Widget child; + final String messageId; + + const _AnimatedReactionSelector({ + required this.animation, + required this.child, + required this.messageId, + }); + + @override + Widget build(BuildContext context) { + return Hero( + tag: messageId, + child: Material( + color: Colors.transparent, + child: AnimatedBuilder( + animation: animation, + builder: (context, child) { + return Transform.scale( + scale: Curves.easeOut.transform(animation.value), + child: child, + ); + }, + child: child, + ), + ), + ); + } +} diff --git a/app/lib/features/chat_ng/actions/toggle_reaction_action.dart b/app/lib/features/chat_ng/actions/toggle_reaction_action.dart new file mode 100644 index 000000000000..5d297049e497 --- /dev/null +++ b/app/lib/features/chat_ng/actions/toggle_reaction_action.dart @@ -0,0 +1,19 @@ +import 'package:acter/features/chat/providers/chat_providers.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:logging/logging.dart'; + +final _log = Logger('a3::chat::toggle_reaction'); + +Future toggleReactionAction( + WidgetRef ref, + String roomId, + String uniqueId, + String emoji, +) async { + try { + final stream = await ref.read(timelineStreamProvider(roomId).future); + await stream.toggleReaction(uniqueId, emoji); + } catch (e, s) { + _log.severe('Reaction toggle failed', e, s); + } +} diff --git a/app/lib/features/chat_ng/pages/chat_room.dart b/app/lib/features/chat_ng/pages/chat_room.dart index 74b0a782490d..e57b735a97c0 100644 --- a/app/lib/features/chat_ng/pages/chat_room.dart +++ b/app/lib/features/chat_ng/pages/chat_room.dart @@ -63,7 +63,7 @@ class ChatRoomNgPage extends ConsumerWidget { lang.membersCount(members.length), style: Theme.of(context).textTheme.bodySmall, ), - skipLoadingOnReload: false, + skipLoadingOnReload: true, error: (e, s) { _log.severe('Failed to load active members', e, s); return Text(lang.errorLoadingMembersCount(e)); diff --git a/app/lib/features/chat_ng/providers/chat_room_messages_provider.dart b/app/lib/features/chat_ng/providers/chat_room_messages_provider.dart index 9aa129ef65cb..1ba2b37396dd 100644 --- a/app/lib/features/chat_ng/providers/chat_room_messages_provider.dart +++ b/app/lib/features/chat_ng/providers/chat_room_messages_provider.dart @@ -1,6 +1,7 @@ import 'package:acter/common/providers/chat_providers.dart'; import 'package:acter/common/providers/common_providers.dart'; import 'package:acter/common/providers/room_providers.dart'; +import 'package:acter/common/utils/utils.dart'; import 'package:acter/common/widgets/html_editor/models/mention_type.dart'; import 'package:acter/features/chat_ng/models/chat_room_state/chat_room_state.dart'; import 'package:acter/features/chat_ng/models/replied_to_msg_state.dart'; @@ -21,6 +22,7 @@ const _supportedTypes = [ typedef RoomMsgId = ({String roomId, String uniqueId}); typedef MentionQuery = (String, MentionType); +typedef ReactionItem = (String, List); final chatStateProvider = StateNotifierProvider.family( @@ -139,3 +141,18 @@ final repliedToMsgProvider = AsyncNotifierProvider.autoDispose .family(() { return RepliedToMessageNotifier(); }); + +final messageReactionsProvider = StateProvider.autoDispose + .family, RoomEventItem>((ref, item) { + List reactions = []; + + final reactionKeys = asDartStringList(item.reactionKeys()); + for (final key in reactionKeys) { + final records = item.reactionRecords(key); + if (records != null) { + reactions.add((key, records.toList())); + } + } + + return reactions; +}); diff --git a/app/lib/features/chat_ng/widgets/chat_bubble.dart b/app/lib/features/chat_ng/widgets/chat_bubble.dart index 96a3ba594239..9c5b5a2acb01 100644 --- a/app/lib/features/chat_ng/widgets/chat_bubble.dart +++ b/app/lib/features/chat_ng/widgets/chat_bubble.dart @@ -1,3 +1,4 @@ +import 'package:acter/common/extensions/acter_build_context.dart'; import 'package:acter/common/themes/acter_theme.dart'; import 'package:flutter/material.dart'; import 'package:acter/common/extensions/options.dart'; @@ -7,7 +8,7 @@ class ChatBubble extends StatelessWidget { final Widget child; final int? messageWidth; final BoxDecoration decoration; - final CrossAxisAlignment bubbleAlignment; + final MainAxisAlignment bubbleAlignment; final bool isEdited; final Widget? repliedToBuilder; @@ -43,7 +44,7 @@ class ChatBubble extends StatelessWidget { bottomRight: Radius.circular(16), ), ), - bubbleAlignment: CrossAxisAlignment.start, + bubbleAlignment: MainAxisAlignment.start, isEdited: isEdited, repliedToBuilder: repliedToBuilder, child: child, @@ -51,7 +52,7 @@ class ChatBubble extends StatelessWidget { } // for user's own messages - factory ChatBubble.user({ + factory ChatBubble.me({ Key? key, required BuildContext context, required Widget child, @@ -73,8 +74,9 @@ class ChatBubble extends StatelessWidget { bottomRight: Radius.circular(isNextMessageInGroup ? 16 : 4), ), ), - bubbleAlignment: CrossAxisAlignment.end, + bubbleAlignment: MainAxisAlignment.end, repliedToBuilder: repliedToBuilder, + isEdited: isEdited, child: DefaultTextStyle.merge( style: theme.textTheme.bodySmall ?.copyWith(color: theme.colorScheme.onPrimary), @@ -88,22 +90,26 @@ class ChatBubble extends StatelessWidget { final chatTheme = Theme.of(context).chatTheme; final size = MediaQuery.sizeOf(context); final msgWidth = messageWidth.map((w) => w.toDouble()); + final defaultWidth = + context.isLargeScreen ? size.width * 0.5 : size.width * 0.75; return Container( margin: const EdgeInsets.symmetric(horizontal: 8), - child: Column( - crossAxisAlignment: bubbleAlignment, + child: Row( + mainAxisAlignment: bubbleAlignment, + crossAxisAlignment: CrossAxisAlignment.end, children: [ + const SizedBox(width: 5), Container( constraints: BoxConstraints( - maxWidth: msgWidth ?? size.width, + maxWidth: msgWidth ?? defaultWidth, ), width: msgWidth, decoration: decoration, child: Padding( padding: const EdgeInsets.symmetric( - horizontal: 12, - vertical: 8, + horizontal: 16, + vertical: 16, ), child: Column( crossAxisAlignment: CrossAxisAlignment.start, @@ -113,19 +119,21 @@ class ChatBubble extends StatelessWidget { const SizedBox(height: 10), ], child, + if (isEdited) ...[ + const SizedBox(width: 5), + Align( + alignment: Alignment.bottomRight, + child: Text( + L10n.of(context).edited, + style: chatTheme.emptyChatPlaceholderTextStyle + .copyWith(fontSize: 12), + ), + ), + ], ], ), ), ), - if (isEdited) - Align( - alignment: Alignment(0.9, 0.0), - child: Text( - L10n.of(context).edited, - style: chatTheme.emptyChatPlaceholderTextStyle - .copyWith(fontSize: 12), - ), - ), ], ), ); diff --git a/app/lib/features/chat_ng/widgets/events/chat_event.dart b/app/lib/features/chat_ng/widgets/events/chat_event.dart index 678a87481076..91d3cb0406e3 100644 --- a/app/lib/features/chat_ng/widgets/events/chat_event.dart +++ b/app/lib/features/chat_ng/widgets/events/chat_event.dart @@ -70,15 +70,14 @@ class ChatEvent extends ConsumerWidget { final options = AvatarOptions.DM(avatarInfo, size: 14); final myId = ref.watch(myUserIdStrProvider); final messageId = msg.uniqueId(); - final isUser = myId == item.sender(); + final isMe = myId == item.sender(); // TODO: render a regular timeline event return Row( mainAxisAlignment: - !isUser ? MainAxisAlignment.start : MainAxisAlignment.end, + !isMe ? MainAxisAlignment.start : MainAxisAlignment.end, crossAxisAlignment: CrossAxisAlignment.end, - mainAxisSize: MainAxisSize.min, children: [ - (!isNextMessageInGroup && !isUser) + (!isNextMessageInGroup && !isMe) ? Padding( padding: const EdgeInsets.only(left: 8), child: ActerAvatar(options: options), @@ -89,7 +88,7 @@ class ChatEvent extends ConsumerWidget { roomId: roomId, messageId: messageId, item: item, - isUser: isUser, + isMe: isMe, isNextMessageInGroup: isNextMessageInGroup, ), ), diff --git a/app/lib/features/chat_ng/widgets/events/chat_event_item.dart b/app/lib/features/chat_ng/widgets/events/chat_event_item.dart index 996f6217c015..33e44d27ec58 100644 --- a/app/lib/features/chat_ng/widgets/events/chat_event_item.dart +++ b/app/lib/features/chat_ng/widgets/events/chat_event_item.dart @@ -1,15 +1,10 @@ -import 'package:acter/features/chat/utils.dart'; import 'package:acter/features/chat/widgets/messages/encrypted_message.dart'; import 'package:acter/features/chat/widgets/messages/redacted_message.dart'; import 'package:acter/features/chat_ng/widgets/chat_bubble.dart'; -import 'package:acter/features/chat_ng/widgets/events/file_message_event.dart'; -import 'package:acter/features/chat_ng/widgets/events/image_message_event.dart'; import 'package:acter/features/chat_ng/widgets/events/member_update_event.dart'; +import 'package:acter/features/chat_ng/widgets/events/message_event_item.dart'; import 'package:acter/features/chat_ng/widgets/events/state_update_event.dart'; -import 'package:acter/features/chat_ng/widgets/events/text_message_event.dart'; -import 'package:acter/features/chat_ng/widgets/events/video_message_event.dart'; -import 'package:acter/common/extensions/options.dart'; -import 'package:acter/features/chat_ng/widgets/replied_to_preview.dart'; + import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart' show RoomEventItem; import 'package:flutter/material.dart'; @@ -18,14 +13,14 @@ class ChatEventItem extends StatelessWidget { final String roomId; final String messageId; final RoomEventItem item; - final bool isUser; + final bool isMe; final bool isNextMessageInGroup; const ChatEventItem({ super.key, required this.roomId, required this.messageId, required this.item, - required this.isUser, + required this.isMe, required this.isNextMessageInGroup, }); @@ -35,14 +30,15 @@ class ChatEventItem extends StatelessWidget { return switch (eventType) { // handle message inner types separately - 'm.room.message' => buildMsgEventItem( - context, - roomId, - messageId, - item, + 'm.room.message' => MessageEventItem( + roomId: roomId, + messageId: messageId, + item: item, + isMe: isMe, + isNextMessageInGroup: isNextMessageInGroup, ), - 'm.room.redaction' => isUser - ? ChatBubble.user( + 'm.room.redaction' => isMe + ? ChatBubble.me( context: context, isNextMessageInGroup: isNextMessageInGroup, child: RedactedMessageWidget(), @@ -52,8 +48,8 @@ class ChatEventItem extends StatelessWidget { isNextMessageInGroup: isNextMessageInGroup, child: RedactedMessageWidget(), ), - 'm.room.encrypted' => isUser - ? ChatBubble.user( + 'm.room.encrypted' => isMe + ? ChatBubble.me( context: context, isNextMessageInGroup: isNextMessageInGroup, child: EncryptedMessageWidget(), @@ -64,7 +60,7 @@ class ChatEventItem extends StatelessWidget { child: EncryptedMessageWidget(), ), 'm.room.member' => MemberUpdateEvent( - isUser: isUser, + isMe: isMe, item: item, ), 'm.policy.rule.room' || @@ -92,88 +88,6 @@ class ChatEventItem extends StatelessWidget { }; } - Widget buildMsgEventItem( - BuildContext context, - String roomId, - String messageId, - RoomEventItem item, - ) { - final msgType = item.msgType(); - final content = item.msgContent(); - // shouldn't happen but in case return empty - if (msgType == null || content == null) return const SizedBox.shrink(); - - return switch (msgType) { - 'm.emote' || - 'm.notice' || - 'm.server_notice' || - 'm.text' => - buildTextMsgEvent(context, item), - 'm.image' => ImageMessageEvent( - messageId: messageId, - roomId: roomId, - content: content, - ), - 'm.video' => VideoMessageEvent( - roomId: roomId, - messageId: messageId, - content: content, - ), - 'm.file' => FileMessageEvent( - roomId: roomId, - messageId: messageId, - content: content, - ), - _ => _buildUnsupportedMessage(msgType), - }; - } - - Widget buildTextMsgEvent(BuildContext context, RoomEventItem item) { - final msgType = item.msgType(); - final repliedTo = item.inReplyTo(); - final wasEdited = item.wasEdited(); - final content = item.msgContent().expect('cannot be null'); - final isNotice = (msgType == 'm.notice' || msgType == 'm.server_notice'); - Widget? repliedToBuilder; - - // whether it contains `replied to` event. - if (repliedTo != null) { - repliedToBuilder = - RepliedToPreview(roomId: roomId, originalId: repliedTo); - } - - // if only consists of emojis - if (isOnlyEmojis(content.body())) { - return TextMessageEvent.emoji( - content: content, - roomId: roomId, - isUser: isUser, - ); - } - - late Widget child; - isNotice - ? child = TextMessageEvent.notice(content: content, roomId: roomId) - : child = TextMessageEvent(content: content, roomId: roomId); - - if (isUser) { - return ChatBubble.user( - context: context, - repliedToBuilder: repliedToBuilder, - isNextMessageInGroup: isNextMessageInGroup, - isEdited: wasEdited, - child: child, - ); - } - return ChatBubble( - context: context, - repliedToBuilder: repliedToBuilder, - isNextMessageInGroup: isNextMessageInGroup, - isEdited: wasEdited, - child: child, - ); - } - Widget _buildUnsupportedMessage(String? msgtype) { return Text( 'Unsupported event type: $msgtype', diff --git a/app/lib/features/chat_ng/widgets/events/member_update_event.dart b/app/lib/features/chat_ng/widgets/events/member_update_event.dart index 8c4474df4321..b318999a7d55 100644 --- a/app/lib/features/chat_ng/widgets/events/member_update_event.dart +++ b/app/lib/features/chat_ng/widgets/events/member_update_event.dart @@ -5,11 +5,11 @@ import 'package:flutter/material.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart'; class MemberUpdateEvent extends StatelessWidget { - final bool isUser; + final bool isMe; final RoomEventItem item; const MemberUpdateEvent({ super.key, - required this.isUser, + required this.isMe, required this.item, }); @@ -22,7 +22,7 @@ class MemberUpdateEvent extends StatelessWidget { final firstName = simplifyUserId(senderId); if (msgType == 'Joined') { - if (isUser) { + if (isMe) { textMsg = lang.chatYouJoined; } else if (firstName != null) { textMsg = lang.chatJoinedDisplayName(firstName); @@ -30,7 +30,7 @@ class MemberUpdateEvent extends StatelessWidget { textMsg = lang.chatJoinedUserId(senderId); } } else if (msgType == 'InvitationAccepted') { - if (isUser) { + if (isMe) { textMsg = lang.chatYouAcceptedInvite; } else if (firstName != null) { textMsg = lang.chatInvitationAcceptedDisplayName(firstName); @@ -38,7 +38,7 @@ class MemberUpdateEvent extends StatelessWidget { textMsg = lang.chatInvitationAcceptedUserId(senderId); } } else if (msgType == 'Invited') { - if (isUser) { + if (isMe) { textMsg = lang.chatYouInvited; } else if (firstName != null) { textMsg = lang.chatInvitedDisplayName(firstName); diff --git a/app/lib/features/chat_ng/widgets/events/message_event_item.dart b/app/lib/features/chat_ng/widgets/events/message_event_item.dart new file mode 100644 index 000000000000..0dd65a99244d --- /dev/null +++ b/app/lib/features/chat_ng/widgets/events/message_event_item.dart @@ -0,0 +1,182 @@ +import 'package:acter/features/chat/utils.dart'; +import 'package:acter/features/chat_ng/actions/reaction_selection_action.dart'; +import 'package:acter/features/chat_ng/widgets/chat_bubble.dart'; +import 'package:acter/features/chat_ng/widgets/events/file_message_event.dart'; +import 'package:acter/features/chat_ng/widgets/events/image_message_event.dart'; +import 'package:acter/features/chat_ng/widgets/events/text_message_event.dart'; +import 'package:acter/features/chat_ng/widgets/events/video_message_event.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reactions_list.dart'; +import 'package:acter/common/extensions/options.dart'; +import 'package:acter/features/chat_ng/widgets/replied_to_preview.dart'; +import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart' + show RoomEventItem; +import 'package:flutter/widgets.dart'; + +class MessageEventItem extends StatelessWidget { + final String roomId; + final String messageId; + final RoomEventItem item; + final bool isMe; + final bool isNextMessageInGroup; + + const MessageEventItem({ + super.key, + required this.roomId, + required this.messageId, + required this.item, + required this.isMe, + required this.isNextMessageInGroup, + }); + + @override + Widget build(BuildContext context) { + return Column( + crossAxisAlignment: + isMe ? CrossAxisAlignment.end : CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + _buildMessageUI(context, roomId, messageId, item, isMe), + _buildReactionsList(roomId, messageId, item, isMe), + ], + ); + } + + Widget _buildMessageUI( + BuildContext context, + String roomId, + String messageId, + RoomEventItem item, + bool isMe, + ) { + final messageWidget = buildMsgEventItem( + context, + roomId, + messageId, + item, + ); + + return GestureDetector( + onLongPressStart: (_) => reactionSelectionAction( + context: context, + messageWidget: messageWidget, + isMe: isMe, + roomId: roomId, + messageId: messageId, + ), + child: Hero( + tag: messageId, + child: messageWidget, + ), + ); + } + + Widget _buildReactionsList( + String roomId, + String messageId, + RoomEventItem item, + bool isMe, + ) { + return Padding( + padding: EdgeInsets.only( + right: isMe ? 12 : 0, + left: isMe ? 0 : 12, + ), + child: FractionalTranslation( + translation: Offset(0, -0.1), + child: ReactionsList( + roomId: roomId, + messageId: messageId, + item: item, + ), + ), + ); + } + + Widget buildMsgEventItem( + BuildContext context, + String roomId, + String messageId, + RoomEventItem item, + ) { + final msgType = item.msgType(); + final content = item.msgContent(); + // shouldn't happen but in case return empty + if (msgType == null || content == null) return const SizedBox.shrink(); + + return switch (msgType) { + 'm.emote' || + 'm.notice' || + 'm.server_notice' || + 'm.text' => + buildTextMsgEvent(context, item), + 'm.image' => ImageMessageEvent( + messageId: messageId, + roomId: roomId, + content: content, + ), + 'm.video' => VideoMessageEvent( + roomId: roomId, + messageId: messageId, + content: content, + ), + 'm.file' => FileMessageEvent( + roomId: roomId, + messageId: messageId, + content: content, + ), + _ => _buildUnsupportedMessage(msgType), + }; + } + + Widget buildTextMsgEvent(BuildContext context, RoomEventItem item) { + final msgType = item.msgType(); + final repliedTo = item.inReplyTo(); + final wasEdited = item.wasEdited(); + final content = item.msgContent().expect('cannot be null'); + final isNotice = (msgType == 'm.notice' || msgType == 'm.server_notice'); + Widget? repliedToBuilder; + + // whether it contains `replied to` event. + if (repliedTo != null) { + repliedToBuilder = + RepliedToPreview(roomId: roomId, originalId: repliedTo); + } + + // if only consists of emojis + if (isOnlyEmojis(content.body())) { + return TextMessageEvent.emoji( + content: content, + roomId: roomId, + isMe: isMe, + ); + } + + late Widget child; + isNotice + ? child = TextMessageEvent.notice(content: content, roomId: roomId) + : child = TextMessageEvent(content: content, roomId: roomId); + + if (isMe) { + return ChatBubble.me( + context: context, + repliedToBuilder: repliedToBuilder, + isNextMessageInGroup: isNextMessageInGroup, + isEdited: wasEdited, + child: child, + ); + } + return ChatBubble( + context: context, + repliedToBuilder: repliedToBuilder, + isNextMessageInGroup: isNextMessageInGroup, + isEdited: wasEdited, + child: child, + ); + } + + Widget _buildUnsupportedMessage(String? msgtype) { + return Text( + 'Unsupported event type: $msgtype', + ); + } +} diff --git a/app/lib/features/chat_ng/widgets/events/text_message_event.dart b/app/lib/features/chat_ng/widgets/events/text_message_event.dart index 68c5188ab997..795deeb72258 100644 --- a/app/lib/features/chat_ng/widgets/events/text_message_event.dart +++ b/app/lib/features/chat_ng/widgets/events/text_message_event.dart @@ -24,22 +24,22 @@ class TextMessageEvent extends StatelessWidget { required this.content, required this.roomId, required TextMessageType type, - bool isUser = false, + bool isMe = false, }) : _type = type, - _isUser = isUser; + _isUser = isMe; factory TextMessageEvent.emoji({ Key? key, required MsgContent content, required String roomId, - required bool isUser, + required bool isMe, }) { return TextMessageEvent.inner( key: key, content: content, roomId: roomId, type: TextMessageType.emoji, - isUser: isUser, + isMe: isMe, ); } diff --git a/app/lib/features/chat_ng/widgets/reactions/reaction_chips_widget.dart b/app/lib/features/chat_ng/widgets/reactions/reaction_chips_widget.dart new file mode 100644 index 000000000000..b3c11449d5b9 --- /dev/null +++ b/app/lib/features/chat_ng/widgets/reactions/reaction_chips_widget.dart @@ -0,0 +1,106 @@ +import 'package:acter/common/extensions/acter_build_context.dart'; +import 'package:acter/common/themes/app_theme.dart'; +import 'package:acter/features/chat_ng/providers/chat_room_messages_provider.dart'; +import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart' + show ReactionRecord; +import 'package:flutter/material.dart'; + +class ReactionChipsWidget extends StatelessWidget { + final List reactions; + final Function(String emoji) onReactionTap; + final VoidCallback onReactionLongPress; + + const ReactionChipsWidget({ + super.key, + required this.reactions, + required this.onReactionTap, + required this.onReactionLongPress, + }); + + @override + Widget build(BuildContext context) { + final colorScheme = Theme.of(context).colorScheme; + final isLargeScreen = context.isLargeScreen; + return Card( + elevation: 8, + margin: const EdgeInsets.symmetric(horizontal: 5, vertical: 2), + color: colorScheme.surface, + shape: RoundedRectangleBorder( + side: BorderSide(color: colorScheme.secondaryContainer), + borderRadius: BorderRadius.circular(12), + ), + child: Padding( + padding: isLargeScreen + ? EdgeInsets.all(3) + : EdgeInsets.symmetric(horizontal: 3), + child: Wrap( + direction: Axis.horizontal, + spacing: 3, + runSpacing: 3, + children: reactions + .map( + (reaction) => _ReactionChip( + emoji: reaction.$1, + records: reaction.$2, + onTap: () => onReactionTap(reaction.$1), + onLongPress: onReactionLongPress, + ), + ) + .toList(), + ), + ), + ); + } +} + +class _ReactionChip extends StatelessWidget { + final String emoji; + final List records; + final VoidCallback onTap; + final VoidCallback onLongPress; + + const _ReactionChip({ + required this.emoji, + required this.records, + required this.onTap, + required this.onLongPress, + }); + + @override + Widget build(BuildContext context) { + final colorScheme = Theme.of(context).colorScheme; + final sentByMe = records.any((x) => x.sentByMe()); + final moreThanOne = records.length > 1; + + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: onTap, + onLongPress: onLongPress, + child: Chip( + padding: const EdgeInsets.symmetric(horizontal: 6), + color: WidgetStatePropertyAll( + sentByMe ? colorScheme.secondaryContainer : colorScheme.surface, + ), + visualDensity: VisualDensity.compact, + labelPadding: + sentByMe ? EdgeInsets.symmetric(horizontal: 3) : EdgeInsets.zero, + shape: const StadiumBorder(side: BorderSide(color: Colors.transparent)), + avatar: _buildEmojiText(), + labelStyle: Theme.of(context).textTheme.labelLarge, + label: moreThanOne + ? Text( + records.length.toString(), + style: Theme.of(context).textTheme.labelLarge, + ) + : const SizedBox.shrink(), + ), + ); + } + + Widget _buildEmojiText() { + return Text( + emoji, + style: EmojiConfig.emojiTextStyle?.copyWith(fontSize: 18), + ); + } +} diff --git a/app/lib/features/chat_ng/widgets/reactions/reaction_detail_sheet.dart b/app/lib/features/chat_ng/widgets/reactions/reaction_detail_sheet.dart new file mode 100644 index 000000000000..c3bf796e2da9 --- /dev/null +++ b/app/lib/features/chat_ng/widgets/reactions/reaction_detail_sheet.dart @@ -0,0 +1,255 @@ +import 'package:acter/common/providers/room_providers.dart'; +import 'package:acter/common/themes/app_theme.dart'; +import 'package:acter/features/chat_ng/providers/chat_room_messages_provider.dart'; +import 'package:acter_avatar/acter_avatar.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_gen/gen_l10n/l10n.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; + +typedef ReactionData = ({ + List allUsers, + Map> userReactions, + Map> reactionUsers, +}); + +class ReactionDetailsSheet extends ConsumerStatefulWidget { + final String roomId; + final List reactions; + + const ReactionDetailsSheet({ + super.key, + required this.roomId, + required this.reactions, + }); + + @override + ConsumerState createState() => + _ReactionDetailsSheetState(); +} + +class _ReactionDetailsSheetState extends ConsumerState + with TickerProviderStateMixin { + late TabController _tabController; + late List _tabs; + late final ReactionData _reactionData; + + @override + void initState() { + super.initState(); + _reactionData = _processReactionData(); + } + + @override + void didChangeDependencies() { + super.didChangeDependencies(); + _initializeTabs(); + } + + @override + void didUpdateWidget(ReactionDetailsSheet oldWidget) { + super.didUpdateWidget(oldWidget); + + // Check if reactions have changed + final hasChanged = widget.reactions.length != oldWidget.reactions.length || + widget.reactions.any((reaction) { + final oldReaction = oldWidget.reactions.firstWhere( + (old) => old.$1 == reaction.$1, + orElse: () => (reaction.$1, []), + ); + return reaction.$2.length != oldReaction.$2.length; + }); + + if (hasChanged) { + setState(() { + _reactionData = _processReactionData(); + // dispose old one + _tabController.dispose(); + _initializeTabs(); + }); + } + } + + @override + void dispose() { + _tabController.dispose(); + super.dispose(); + } + + void _initializeTabs() { + final total = widget.reactions.fold( + 0, + (sum, reaction) => sum + reaction.$2.length, + ); + + _tabs = [ + Tab(child: Chip(label: Text(L10n.of(context).allReactionsCount(total)))), + ...widget.reactions.map( + (reaction) => Tab( + child: Chip( + avatar: Text(reaction.$1, style: EmojiConfig.emojiTextStyle), + label: Text(reaction.$2.length.toString()), + ), + ), + ), + ]; + + _tabController = TabController(length: _tabs.length, vsync: this); + } + + @override + Widget build(BuildContext context) { + return Column( + mainAxisSize: MainAxisSize.min, + children: [ + _buildTabBar(), + _buildTabBarView(), + ], + ); + } + + Widget _buildTabBar() { + return TabBar( + isScrollable: true, + padding: const EdgeInsets.all(24), + controller: _tabController, + overlayColor: WidgetStateProperty.all(Colors.transparent), + indicator: const BoxDecoration( + borderRadius: BorderRadius.all(Radius.circular(12)), + ), + indicatorSize: TabBarIndicatorSize.tab, + dividerColor: Colors.transparent, + tabs: _tabs, + ); + } + + Widget _buildTabBarView() { + return Flexible( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 14), + child: TabBarView( + controller: _tabController, + children: [ + _ReactionUsersList( + roomId: widget.roomId, + users: _reactionData.allUsers, + userReactions: _reactionData.userReactions, + ), + ...widget.reactions.map( + (reaction) => _ReactionUsersList( + roomId: widget.roomId, + users: _reactionData.reactionUsers[reaction.$1] ?? [], + userReactions: _reactionData.userReactions, + ), + ), + ], + ), + ), + ); + } + + ReactionData _processReactionData() { + // how many reactions per user + final userReactions = >{}; + // how many users of single reaction + final reactionUsers = >{}; + + for (final reaction in widget.reactions) { + final emoji = reaction.$1; + reactionUsers[emoji] = []; + + for (final record in reaction.$2) { + final userId = record.senderId().toString(); + userReactions.putIfAbsent(userId, () => []).add(emoji); + reactionUsers[emoji]?.add(userId); + } + } + + final allUsers = userReactions.keys.toList() + ..sort( + (a, b) => (userReactions[b]?.length ?? 0) + .compareTo(userReactions[a]?.length ?? 0), + ); + + return ( + allUsers: allUsers, + userReactions: userReactions, + reactionUsers: reactionUsers, + ); + } +} + +// Users list in reaction details +class _ReactionUsersList extends StatelessWidget { + final String roomId; + final List users; + final Map> userReactions; + + const _ReactionUsersList({ + required this.roomId, + required this.users, + required this.userReactions, + }); + + @override + Widget build(BuildContext context) { + return ListView.separated( + padding: const EdgeInsets.symmetric(vertical: 10), + shrinkWrap: true, + itemCount: users.length, + itemBuilder: (context, index) { + final userId = users[index]; + return ReactionUserItem( + roomId: roomId, + userId: userId, + emojis: userReactions[userId] ?? [], + ); + }, + separatorBuilder: (context, index) => const SizedBox(height: 12), + ); + } +} + +class ReactionUserItem extends ConsumerWidget { + final String roomId; + final String userId; + final List emojis; + + const ReactionUserItem({ + super.key, + required this.roomId, + required this.userId, + required this.emojis, + }); + + @override + Widget build(BuildContext context, WidgetRef ref) { + final theme = Theme.of(context); + final memberInfo = + ref.watch(memberAvatarInfoProvider((roomId: roomId, userId: userId))); + return ListTile( + leading: ActerAvatar( + options: AvatarOptions.DM( + AvatarInfo( + uniqueId: userId, + displayName: memberInfo.displayName, + avatar: memberInfo.avatar, + ), + ), + ), + title: Text(memberInfo.displayName ?? userId), + subtitle: memberInfo.displayName != null + ? Text(userId, style: theme.textTheme.labelLarge) + : null, + trailing: Wrap( + children: emojis + .map( + (emoji) => Padding( + padding: const EdgeInsets.symmetric(horizontal: 2), + child: Text(emoji, style: EmojiConfig.emojiTextStyle), + ), + ) + .toList(), + ), + ); + } +} diff --git a/app/lib/features/chat_ng/widgets/reactions/reaction_selector.dart b/app/lib/features/chat_ng/widgets/reactions/reaction_selector.dart new file mode 100644 index 000000000000..a8005e9fc533 --- /dev/null +++ b/app/lib/features/chat_ng/widgets/reactions/reaction_selector.dart @@ -0,0 +1,116 @@ +import 'package:acter/common/themes/app_theme.dart'; +import 'package:acter/common/utils/constants.dart'; +import 'package:acter/common/widgets/emoji_picker_widget.dart'; +import 'package:acter/features/chat_ng/actions/toggle_reaction_action.dart'; +import 'package:atlas_icons/atlas_icons.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; + +class ReactionSelector extends ConsumerWidget { + final double? size; + final String messageId; + final String roomId; + final bool isMe; + + const ReactionSelector({ + super.key, + required this.isMe, + required this.messageId, + required this.roomId, + this.size, + }); + + @override + Widget build(BuildContext context, WidgetRef ref) { + return ClipRRect( + borderRadius: BorderRadius.circular(20), + child: Container( + padding: const EdgeInsets.all(8), + margin: EdgeInsets.only( + bottom: 4, + left: isMe ? 0 : 8, + right: isMe ? 8 : 0, + ), + decoration: BoxDecoration( + borderRadius: const BorderRadius.all(Radius.circular(20)), + color: Theme.of(context).colorScheme.surface.withOpacity(0.8), + ), + child: _buildEmojiRow(context, ref), + ), + ); + } + + Widget _buildEmojiRow(BuildContext context, WidgetRef ref) { + return Row( + mainAxisSize: MainAxisSize.min, + children: [ + Wrap( + direction: Axis.horizontal, + spacing: 10.0, + children: [ + ..._buildEmojiButtons(context, ref), + _buildMoreButton(context, ref), + ], + ), + ], + ); + } + + List _buildEmojiButtons(BuildContext context, WidgetRef ref) { + return [ + heart, + thumbsUp, + prayHands, + faceWithTears, + clappingHands, + raisedHands, + astonishedFace, + ].map((emoji) => _buildEmojiButton(emoji, context, ref)).toList(); + } + + Widget _buildEmojiButton(String emoji, BuildContext context, WidgetRef ref) { + return InkWell( + onTap: () async { + await toggleReactionAction(ref, roomId, messageId, emoji); + if (context.mounted) Navigator.pop(context); + }, + child: Text( + emoji, + style: (EmojiConfig.emojiTextStyle ?? const TextStyle()) + .copyWith(fontSize: size ?? 28), + ), + ); + } + + Widget _buildMoreButton(BuildContext context, WidgetRef ref) { + return InkWell( + onTap: () => _showEmojiPicker(context, ref), + child: const Padding( + padding: EdgeInsets.only(top: 3), + child: Icon( + Atlas.dots_horizontal_thin, + size: 28, + ), + ), + ); + } + + void _showEmojiPicker(BuildContext context, WidgetRef ref) { + showModalBottomSheet( + context: context, + enableDrag: false, + builder: (context) => EmojiPickerWidget( + withBoarder: true, + onEmojiSelected: (category, emoji) async { + await toggleReactionAction(ref, roomId, messageId, emoji.emoji); + if (context.mounted) { + // we have overlays opened, dismiss both of them + Navigator.pop(context); + Navigator.pop(context); + } + }, + onClosePicker: () => Navigator.pop(context), + ), + ); + } +} diff --git a/app/lib/features/chat_ng/widgets/reactions/reactions_list.dart b/app/lib/features/chat_ng/widgets/reactions/reactions_list.dart new file mode 100644 index 000000000000..496954339b54 --- /dev/null +++ b/app/lib/features/chat_ng/widgets/reactions/reactions_list.dart @@ -0,0 +1,42 @@ +import 'package:acter/features/chat_ng/actions/toggle_reaction_action.dart'; +import 'package:acter/features/chat_ng/providers/chat_room_messages_provider.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reaction_chips_widget.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reaction_detail_sheet.dart'; +import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart' + show RoomEventItem; +import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; + +class ReactionsList extends ConsumerWidget { + final String roomId; + final String messageId; + final RoomEventItem item; + const ReactionsList({ + super.key, + required this.roomId, + required this.messageId, + required this.item, + }); + + @override + Widget build(BuildContext context, WidgetRef ref) { + final reactions = ref.watch(messageReactionsProvider(item)); + if (reactions.isEmpty) return const SizedBox.shrink(); + return ReactionChipsWidget( + reactions: reactions, + onReactionTap: (emoji) => + toggleReactionAction(ref, roomId, messageId, emoji), + onReactionLongPress: () => showReactionsSheet(context, reactions), + ); + } + + void showReactionsSheet(BuildContext context, List reactions) { + showModalBottomSheet( + context: context, + builder: (context) => ReactionDetailsSheet( + roomId: roomId, + reactions: reactions, + ), + ); + } +} diff --git a/app/lib/l10n/app_en.arb b/app/lib/l10n/app_en.arb index 0b48ce9ca3f4..e43e2178df1a 100644 --- a/app/lib/l10n/app_en.arb +++ b/app/lib/l10n/app_en.arb @@ -41,6 +41,7 @@ "@all": {}, "allMessages": "All Messages", "@allMessages": {}, + "allReactionsCount": "All {total}", "alreadyConfirmed": "Already confirmed", "@alreadyConfirmed": {}, "analyticsTitle": "Help us help you", diff --git a/app/pubspec.yaml b/app/pubspec.yaml index 06a45dcef913..521ac6762502 100644 --- a/app/pubspec.yaml +++ b/app/pubspec.yaml @@ -78,8 +78,8 @@ dependencies: path: ../packages/rust_sdk # State Management - flutter_riverpod: ^2.5.1 - riverpod: ^2.5.1 + flutter_riverpod: ^2.6.1 + riverpod: ^2.6.1 # Routing go_router: ^14.1.4 diff --git a/app/test/common/mock_data/mock_avatar_info.dart b/app/test/common/mock_data/mock_avatar_info.dart index bc57793329c1..4163f09db150 100644 --- a/app/test/common/mock_data/mock_avatar_info.dart +++ b/app/test/common/mock_data/mock_avatar_info.dart @@ -4,9 +4,12 @@ import 'package:mocktail/mocktail.dart'; class MockAvatarInfo extends Mock implements AvatarInfo { @override - String get uniqueId => 'mockUniqueId'; + final String uniqueId; + + String get userId => uniqueId; @override TooltipStyle get tooltip => TooltipStyle.Combined; + MockAvatarInfo({required this.uniqueId}); } diff --git a/app/test/common/mock_data/mock_user_id.dart b/app/test/common/mock_data/mock_user_id.dart index 4ed746f9f982..b44daf3e13c4 100644 --- a/app/test/common/mock_data/mock_user_id.dart +++ b/app/test/common/mock_data/mock_user_id.dart @@ -1,4 +1,14 @@ import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart'; import 'package:mocktail/mocktail.dart'; -class MockUserId extends Mock implements UserId {} +class MockUserId extends Mock implements UserId { + final String _value; + + MockUserId(this._value); + + @override + String toString() => _value; +} + +// Utility function to create mock user IDs from str +MockUserId createMockUserId(String value) => MockUserId(value); diff --git a/app/test/features/chat_ng/reactions/reactions_test.dart b/app/test/features/chat_ng/reactions/reactions_test.dart new file mode 100644 index 000000000000..57568b1d005c --- /dev/null +++ b/app/test/features/chat_ng/reactions/reactions_test.dart @@ -0,0 +1,263 @@ +import 'package:acter/common/providers/room_providers.dart'; +import 'package:acter/common/providers/sdk_provider.dart'; +import 'package:acter/features/chat_ng/providers/chat_room_messages_provider.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reaction_chips_widget.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reaction_detail_sheet.dart'; +import 'package:acter/features/chat_ng/widgets/reactions/reactions_list.dart'; +import 'package:acter_flutter_sdk/acter_flutter_sdk_ffi.dart' + show ReactionRecord, UserId; +import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:mocktail/mocktail.dart'; + +import '../../../common/mock_data/mock_avatar_info.dart'; +import '../../../common/mock_data/mock_user_id.dart'; +import '../../../helpers/mock_a3sdk.dart'; +import '../../../helpers/test_util.dart'; +import '../messages/chat_message_test.dart'; + +class MockReactionRecord extends Mock implements ReactionRecord { + final UserId _senderId; + + MockReactionRecord(this._senderId); + + @override + bool sentByMe() => false; + @override + UserId senderId() => _senderId; +} + +void main() { + group('Reactions widgets test', () { + group('tab initialization tests', () { + testWidgets('creates correct number of initial tabs', (tester) async { + final reactions = [ + ('👍', [MockReactionRecord(createMockUserId('user-1'))]), + ( + '❤️', + [ + MockReactionRecord(createMockUserId('user-2')), + MockReactionRecord(createMockUserId('user-3')), + ] + ), + ]; + + await tester.pumpProviderWidget( + overrides: [ + memberAvatarInfoProvider.overrideWith( + (ref, param) => MockAvatarInfo(uniqueId: param.userId), + ), + ], + child: MaterialApp( + home: Scaffold( + body: ReactionDetailsSheet( + roomId: 'test-room', + reactions: reactions, + ), + ), + ), + ); + + // Verify "All" tab plus one per reaction + expect(find.byType(Tab), findsNWidgets(3)); + expect(find.text('All 3'), findsOneWidget); + expect(find.text('1'), findsOneWidget); // 👍 count + expect(find.text('2'), findsOneWidget); // ❤️ count + }); + + testWidgets('handles empty reactions list', (tester) async { + await tester.pumpProviderWidget( + child: MaterialApp( + home: Scaffold( + body: ReactionDetailsSheet( + roomId: 'test-room', + reactions: [], + ), + ), + ), + ); + + expect(find.byType(Tab), findsNWidgets(1)); // Only "All" tab + expect(find.text('All 0'), findsOneWidget); + }); + }); + + group('reaction update tests', () { + final mockEvent = MockRoomEventItem(mockSender: 'user-1'); + final reactionsNotifier = StateController>([ + ('👍', [MockReactionRecord(createMockUserId('user-1'))]), + ]); + final overrides = [ + sdkProvider.overrideWith((ref) => MockActerSdk()), + memberAvatarInfoProvider.overrideWith( + (ref, param) => MockAvatarInfo(uniqueId: param.userId), + ), + messageReactionsProvider(mockEvent).overrideWith( + (ref) => reactionsNotifier.state, + ), + ]; + // Helper to open bottom sheet + Future openReactionsDetailSheet( + List reactions, + WidgetTester tester, + ) async { + await tester.tap(find.byType(ReactionChipsWidget)); + await tester.pumpAndSettle(); + + // Show bottom sheet directly + await tester.pump(); + showModalBottomSheet( + context: tester.element(find.byType(ReactionsList)), + builder: (context) => ReactionDetailsSheet( + roomId: 'test-room', + reactions: reactions, + ), + ); + await tester.pumpAndSettle(); + } + + testWidgets('updates tabs when reactions change', (tester) async { + await tester.pumpProviderWidget( + overrides: overrides, + child: MaterialApp( + home: Scaffold( + body: ReactionsList( + roomId: 'test-room', + messageId: 'message-1', + item: mockEvent, + ), + ), + ), + ); + + // First time opening with initial state + await openReactionsDetailSheet(reactionsNotifier.state, tester); + expect(find.byType(Tab), findsNWidgets(2)); + expect(find.text('All 1'), findsOneWidget); + + // Close sheet + await tester.tapAt(const Offset(20, 20)); + await tester.pumpAndSettle(); + + // Update reactions + final updatedReactions = [ + ('👍', [MockReactionRecord(createMockUserId('user-1'))]), + ('❤️', [MockReactionRecord(createMockUserId('user-2'))]), + ]; + reactionsNotifier.state = updatedReactions; + await tester.pumpAndSettle(); + + // Open bottom sheet again with updated state + await openReactionsDetailSheet(reactionsNotifier.state, tester); + expect( + find.byType(Tab), + findsNWidgets(3), + reason: 'Should have 3 tabs after adding a new reaction', + ); + expect( + find.text('All 2'), + findsOneWidget, + reason: 'Should show total of 2 reactions', + ); + }); + + testWidgets('updates when reaction count changes', (tester) async { + // reset notifier state before previous test run + reactionsNotifier.state = [ + ('👍', [MockReactionRecord(createMockUserId('user-1'))]), + ]; + await tester.pumpProviderWidget( + overrides: overrides, + child: MaterialApp( + home: Scaffold( + body: ReactionsList( + roomId: 'test-room', + messageId: 'message-1', + item: mockEvent, + ), + ), + ), + ); + + // Show initial state + await openReactionsDetailSheet(reactionsNotifier.state, tester); + expect(find.text('1'), findsOneWidget); + + // Close bottom sheet + await tester.tapAt(const Offset(20, 20)); + await tester.pumpAndSettle(); + + // Update state with second user + reactionsNotifier.state = [ + ( + '👍', + [ + MockReactionRecord(createMockUserId('user-1')), + MockReactionRecord(createMockUserId('user-2')), + ] + ), + ]; + await tester.pumpAndSettle(); + + // Show updated state + await openReactionsDetailSheet(reactionsNotifier.state, tester); + expect(find.text('2'), findsOneWidget); + }); + }); + + group('user list tests', () { + testWidgets('displays correct users count in each tab', (tester) async { + final reactions = [ + ( + '👍', + [ + MockReactionRecord(createMockUserId('user-1')), + MockReactionRecord(createMockUserId('user-2')), + ] + ), + ( + '❤️', + [ + MockReactionRecord(createMockUserId('user-2')), + MockReactionRecord(createMockUserId('user-3')), + ] + ), + ]; + + await tester.pumpProviderWidget( + overrides: [ + memberAvatarInfoProvider.overrideWith( + (ref, param) => MockAvatarInfo(uniqueId: param.userId), + ), + ], + child: MaterialApp( + home: Scaffold( + body: ReactionDetailsSheet( + roomId: 'test-room', + reactions: reactions, + ), + ), + ), + ); + + // Check "All" tab content + expect( + find.byType(ReactionUserItem), + findsNWidgets(3), + ); // All unique users + + // Check 👍 tab + await tester.tap(find.text('2').first); // First tab with count 2 + await tester.pumpAndSettle(); + + expect(find.byType(ReactionUserItem), findsNWidgets(2)); + + // Check ❤️ tab + await tester.tap(find.text('2').last); // Second tab with count 2 + await tester.pumpAndSettle(); + expect(find.byType(ReactionUserItem), findsNWidgets(2)); + }); + }); + }); +} diff --git a/app/test/features/comments/add_comment_test.dart b/app/test/features/comments/add_comment_test.dart index f5f1e48a15fa..5b08bd6d9800 100644 --- a/app/test/features/comments/add_comment_test.dart +++ b/app/test/features/comments/add_comment_test.dart @@ -14,7 +14,7 @@ void main() { setUp(() { mockCommentsManager = MockCommentsManager(); - mockAvatarInfo = MockAvatarInfo(); + mockAvatarInfo = MockAvatarInfo(uniqueId: 'user-1'); }); group('Add Comment', () { testWidgets('should display avatar and comment input', diff --git a/app/test/features/comments/comment_item_test.dart b/app/test/features/comments/comment_item_test.dart index 41bd919da66e..bd1fdf95277f 100644 --- a/app/test/features/comments/comment_item_test.dart +++ b/app/test/features/comments/comment_item_test.dart @@ -21,11 +21,11 @@ void main() { setUp(() { mockCommentsManager = MockCommentsManager(); mockComment = MockComment( - fakeSender: MockUserId(), + fakeSender: MockUserId('user-1'), fakeMsgContent: MockMsgContent(bodyText: 'This is a test message'), fakeOriginServerTs: DateTime.now().millisecondsSinceEpoch, ); - mockAvatarInfo = MockAvatarInfo(); + mockAvatarInfo = MockAvatarInfo(uniqueId: 'user-1'); // Mock the values expected by the widget when(() => mockCommentsManager.roomIdStr()).thenReturn('roomId'); diff --git a/app/test/features/comments/comment_list_test.dart b/app/test/features/comments/comment_list_test.dart index 7b8c6d193ab3..39e6c008bdc3 100644 --- a/app/test/features/comments/comment_list_test.dart +++ b/app/test/features/comments/comment_list_test.dart @@ -90,17 +90,17 @@ void main() { // Arrange final mockComment1 = MockComment( - fakeSender: MockUserId(), + fakeSender: MockUserId('user-1'), fakeMsgContent: MockMsgContent(bodyText: 'message 1'), fakeOriginServerTs: DateTime.now().millisecondsSinceEpoch, ); final mockComment2 = MockComment( - fakeSender: MockUserId(), + fakeSender: MockUserId('user-2'), fakeMsgContent: MockMsgContent(bodyText: 'message 2'), fakeOriginServerTs: DateTime.now().millisecondsSinceEpoch, ); final mockComment3 = MockComment( - fakeSender: MockUserId(), + fakeSender: MockUserId('user-3'), fakeMsgContent: MockMsgContent(bodyText: 'message 3'), fakeOriginServerTs: DateTime.now().millisecondsSinceEpoch, );