diff --git a/lib/widgets/message_list.dart b/lib/widgets/message_list.dart index aefd29be879..084fe029eb1 100644 --- a/lib/widgets/message_list.dart +++ b/lib/widgets/message_list.dart @@ -556,6 +556,7 @@ class StreamMessageRecipientHeader extends StatelessWidget { final subscription = store.subscriptions[message.streamId]; final Color backgroundColor; final Color contrastingColor; + final Color iconColor; if (subscription != null) { final swatch = subscription.colorSwatch(); backgroundColor = swatch.barBackground; @@ -563,9 +564,11 @@ class StreamMessageRecipientHeader extends StatelessWidget { (ThemeData.estimateBrightnessForColor(swatch.barBackground) == Brightness.dark) ? Colors.white : Colors.black; + iconColor = swatch.iconOnBarBackground; } else { backgroundColor = _kFallbackStreamColor; contrastingColor = Colors.black; + iconColor = Colors.black; } final textStyle = TextStyle( color: contrastingColor, @@ -589,8 +592,15 @@ class StreamMessageRecipientHeader extends StatelessWidget { child: Row( crossAxisAlignment: CrossAxisAlignment.center, children: [ - const SizedBox(width: 16), - // TODO globe/lock icons for web-public and private streams + Padding( + // Figma specifies 5px horizontal spacing but 6px matches + // more accurately what we see in Figma. + // Bottom padding added here to shift icon up to + // match alignment with text visually. + padding: const EdgeInsets.only(left: 6, right: 6, bottom: 3), + child: Icon(size: 16, color: iconColor, + // A null [Icon.icon] makes a blank space. + (stream != null) ? iconDataForStream(stream) : null)), Padding( padding: const EdgeInsets.symmetric(vertical: 11), child: Text(streamName, diff --git a/test/widgets/message_list_test.dart b/test/widgets/message_list_test.dart index ba1bdec466c..048233c24db 100644 --- a/test/widgets/message_list_test.dart +++ b/test/widgets/message_list_test.dart @@ -14,6 +14,7 @@ import 'package:zulip/model/localizations.dart'; import 'package:zulip/model/narrow.dart'; import 'package:zulip/model/store.dart'; import 'package:zulip/widgets/content.dart'; +import 'package:zulip/widgets/icons.dart'; import 'package:zulip/widgets/message_list.dart'; import 'package:zulip/widgets/sticky_header.dart'; import 'package:zulip/widgets/store.dart'; @@ -249,6 +250,54 @@ void main() { matching: find.byType(ColoredBox), ))).color.equals(swatch.barBackground); }); + + testWidgets('color of stream icon', (tester) async { + final stream = eg.stream(isWebPublic: true); + final subscription = eg.subscription(stream, color: Colors.red.value); + final swatch = subscription.colorSwatch(); + await setupMessageListPage(tester, + messages: [eg.streamMessage(stream: subscription)], + subscriptions: [subscription]); + await tester.pump(); + check(tester.widget(find.byIcon(ZulipIcons.globe))) + .color.equals(swatch.iconOnBarBackground); + }); + + testWidgets('normal streams show hash icon', (tester) async { + final stream = eg.stream(isWebPublic: false, inviteOnly: false); + await setupMessageListPage(tester, + messages: [eg.streamMessage(stream: stream)], + streams: [stream]); + await tester.pump(); + check(find.descendant( + of: find.byType(StreamMessageRecipientHeader), + matching: find.byIcon(ZulipIcons.hash_sign), + ).evaluate()).length.equals(1); + }); + + testWidgets('public streams show globe icon', (tester) async { + final stream = eg.stream(isWebPublic: true); + await setupMessageListPage(tester, + messages: [eg.streamMessage(stream: stream)], + streams: [stream]); + await tester.pump(); + check(find.descendant( + of: find.byType(StreamMessageRecipientHeader), + matching: find.byIcon(ZulipIcons.globe), + ).evaluate()).length.equals(1); + }); + + testWidgets('private streams show lock icon', (tester) async { + final stream = eg.stream(inviteOnly: true); + await setupMessageListPage(tester, + messages: [eg.streamMessage(stream: stream)], + streams: [stream]); + await tester.pump(); + check(find.descendant( + of: find.byType(StreamMessageRecipientHeader), + matching: find.byIcon(ZulipIcons.lock), + ).evaluate()).length.equals(1); + }); }); testWidgets('show stream name from message when stream unknown', (tester) async {