diff --git a/lib/model/content.dart b/lib/model/content.dart
index 925a3e64f3..f4d0faf44c 100644
--- a/lib/model/content.dart
+++ b/lib/model/content.dart
@@ -309,6 +309,17 @@ class MathBlockNode extends BlockContentNode {
}
}
+class ImageNodeList extends BlockContentNode {
+ const ImageNodeList(this.images, {super.debugHtmlNode});
+
+ final List images;
+
+ @override
+ List debugDescribeChildren() {
+ return images.map((node) => node.toDiagnosticsNode()).toList();
+ }
+}
+
class ImageNode extends BlockContentNode {
const ImageNode({super.debugHtmlNode, required this.srcUrl});
@@ -1031,13 +1042,26 @@ class _ZulipContentParser {
List parseBlockContentList(dom.NodeList nodes) {
assert(_debugParserContext == _ParserContext.block);
- final acceptedNodes = nodes.where((node) {
+ final List result = [];
+ List imageNodes = [];
+ for (final node in nodes) {
// We get a bunch of newline Text nodes between paragraphs.
// A browser seems to ignore these; let's do the same.
- if (node is dom.Text && (node.text == '\n')) return false;
- return true;
- });
- return acceptedNodes.map(parseBlockContent).toList(growable: false);
+ if (node is dom.Text && (node.text == '\n')) continue;
+
+ final block = parseBlockContent(node);
+ if (block is ImageNode) {
+ imageNodes.add(block);
+ continue;
+ }
+ if (imageNodes.isNotEmpty) {
+ result.add(ImageNodeList(imageNodes));
+ imageNodes = [];
+ }
+ result.add(block);
+ }
+ if (imageNodes.isNotEmpty) result.add(ImageNodeList(imageNodes));
+ return result;
}
ZulipContent parse(String html) {
diff --git a/lib/widgets/content.dart b/lib/widgets/content.dart
index db6dd5b6d6..f22108ec15 100644
--- a/lib/widgets/content.dart
+++ b/lib/widgets/content.dart
@@ -84,6 +84,8 @@ class BlockContentList extends StatelessWidget {
return CodeBlock(node: node);
} else if (node is MathBlockNode) {
return MathBlock(node: node);
+ } else if (node is ImageNodeList) {
+ return MessageImageList(node: node);
} else if (node is ImageNode) {
return MessageImage(node: node);
} else if (node is UnimplementedBlockContentNode) {
@@ -230,6 +232,18 @@ class ListItemWidget extends StatelessWidget {
}
}
+class MessageImageList extends StatelessWidget {
+ const MessageImageList({super.key, required this.node});
+
+ final ImageNodeList node;
+
+ @override
+ Widget build(BuildContext context) {
+ return Wrap(
+ children: node.images.map((imageNode) => MessageImage(node: imageNode)).toList());
+ }
+}
+
class MessageImage extends StatelessWidget {
const MessageImage({super.key, required this.node});
@@ -239,7 +253,6 @@ class MessageImage extends StatelessWidget {
Widget build(BuildContext context) {
final message = InheritedMessage.of(context);
- // TODO(#193) multiple images in a row
// TODO image hover animation
final src = node.srcUrl;
@@ -251,7 +264,7 @@ class MessageImage extends StatelessWidget {
Navigator.of(context).push(getLightboxRoute(
context: context, message: message, src: resolvedSrc));
},
- child: Align(
+ child: UnconstrainedBox(
alignment: Alignment.centerLeft,
child: Padding(
// TODO clean up this padding by imitating web less precisely;
diff --git a/test/model/content_test.dart b/test/model/content_test.dart
index 9d8c4f22d0..8d22bcef44 100644
--- a/test/model/content_test.dart
+++ b/test/model/content_test.dart
@@ -256,6 +256,115 @@ class ContentExample {
'λ'
'
\n
\n',
[QuotationNode([MathBlockNode(texSource: r'\lambda')])]);
+
+ static const imageSingle = ContentExample(
+ 'single image',
+ "https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3",
+ '', [
+ ImageNodeList([
+ ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3'),
+ ]),
+ ]);
+
+ static const imageCluster = ContentExample(
+ 'multiple images',
+ "https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3\nhttps://chat.zulip.org/user_avatars/2/realm/icon.png?version=4",
+ ''
+ 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3
\n'
+ 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=4
\n'
+ ''
+ '', [
+ ParagraphNode(links: null, nodes: [
+ LinkNode(url: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3', nodes: [TextNode('https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3')]),
+ LineBreakInlineNode(),
+ TextNode('\n'),
+ LinkNode(url: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=4', nodes: [TextNode('https://chat.zulip.org/user_avatars/2/realm/icon.png?version=4')]),
+ ]),
+ ImageNodeList([
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/f535ba07f95b99a83aa48e44fd62bbb6c6cf6615/68747470733a2f2f636861742e7a756c69702e6f72672f757365725f617661746172732f322f7265616c6d2f69636f6e2e706e673f76657273696f6e3d33'),
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/8f63bc2632a0e41be3f457d86c077e61b4a03e7e/68747470733a2f2f636861742e7a756c69702e6f72672f757365725f617661746172732f322f7265616c6d2f69636f6e2e706e673f76657273696f6e3d34'),
+ ]),
+ ]);
+
+ static const imageClusterThenContent = ContentExample(
+ 'content after image cluster',
+ "https://chat.zulip.org/user_avatars/2/realm/icon.png\nhttps://chat.zulip.org/user_avatars/2/realm/icon.png?version=2\n\nmore content",
+ 'content '
+ 'icon.png '
+ 'icon.png
\n'
+ ''
+ ''
+ 'more content
', [
+ ParagraphNode(links: null, nodes: [
+ TextNode('content '),
+ LinkNode(url: 'https://chat.zulip.org/user_avatars/2/realm/icon.png', nodes: [TextNode('icon.png')]),
+ TextNode(' '),
+ LinkNode(url: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=2', nodes: [TextNode('icon.png')]),
+ ]),
+ ImageNodeList([
+ ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png'),
+ ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=2'),
+ ]),
+ ParagraphNode(links: null, nodes: [
+ TextNode('more content'),
+ ]),
+ ]);
+
+ static const imageMultipleClusters = ContentExample(
+ 'multiple clusters of images',
+ "https://en.wikipedia.org/static/images/icons/wikipedia.png\nhttps://en.wikipedia.org/static/images/icons/wikipedia.png?v=1\n\nTest\n\nhttps://en.wikipedia.org/static/images/icons/wikipedia.png?v=2\nhttps://en.wikipedia.org/static/images/icons/wikipedia.png?v=3",
+ ''
+ 'https://en.wikipedia.org/static/images/icons/wikipedia.png
\n' 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=1
\n'
+ ''
+ ''
+ 'Test
\n'
+ ''
+ 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=2
\n'
+ 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=3
\n'
+ ''
+ '', [
+ ParagraphNode(links: null, nodes: [
+ LinkNode(url: 'https://en.wikipedia.org/static/images/icons/wikipedia.png', nodes: [TextNode('https://en.wikipedia.org/static/images/icons/wikipedia.png')]),
+ LineBreakInlineNode(),
+ TextNode('\n'),
+ LinkNode(url: 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=1', nodes: [TextNode('https://en.wikipedia.org/static/images/icons/wikipedia.png?v=1')]),
+ ]),
+ ImageNodeList([
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/34b2695ca83af76204b0b25a8f2019ee35ec38fa/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e67'),
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/d200fb112aaccbff9df767373a201fa59601f362/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e673f763d31'),
+ ]),
+ ParagraphNode(links: null, nodes: [
+ TextNode('Test'),
+ ]),
+ ParagraphNode(links: null, nodes: [
+ LinkNode(url: 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=2', nodes: [TextNode('https://en.wikipedia.org/static/images/icons/wikipedia.png?v=2')]),
+ LineBreakInlineNode(),
+ TextNode('\n'),
+ LinkNode(url: 'https://en.wikipedia.org/static/images/icons/wikipedia.png?v=3', nodes: [TextNode('https://en.wikipedia.org/static/images/icons/wikipedia.png?v=3')]),
+ ]),
+ ImageNodeList([
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/c4db87e81348dac94eacaa966b46d968b34029cc/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e673f763d32'),
+ ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/51b70540cf6a5b3c8a0b919c893b8abddd447e88/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e673f763d33'),
+ ]),
+ ]);
}
UnimplementedBlockContentNode blockUnimplemented(String html) {
@@ -576,14 +685,10 @@ void main() {
testParseExample(ContentExample.mathBlock);
testParseExample(ContentExample.mathBlockInQuote);
- testParse('parse image',
- // "https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3"
- '', const [
- ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=3'),
- ]);
+ testParseExample(ContentExample.imageSingle);
+ testParseExample(ContentExample.imageCluster);
+ testParseExample(ContentExample.imageClusterThenContent);
+ testParseExample(ContentExample.imageMultipleClusters);
testParse('parse nested lists, quotes, headings, code blocks',
// "1. > ###### two\n > * three\n\n four"
diff --git a/test/widgets/content_test.dart b/test/widgets/content_test.dart
index 9e8b9789b6..c13d886673 100644
--- a/test/widgets/content_test.dart
+++ b/test/widgets/content_test.dart
@@ -252,6 +252,77 @@ void main() {
tester.widget(find.textContaining(RegExp(r'^(Tue, Jan 30|Wed, Jan 31), 2024, \d+:\d\d [AP]M$')));
});
+ group('MessageImages', () {
+ final message = eg.streamMessage();
+
+ Future prepareContent(WidgetTester tester, String html) async {
+ addTearDown(testBinding.reset);
+
+ await testBinding.globalStore.add(eg.selfAccount, eg.initialSnapshot());
+ final httpClient = FakeImageHttpClient();
+
+ debugNetworkImageHttpClientProvider = () => httpClient;
+ httpClient.request.response
+ ..statusCode = HttpStatus.ok
+ ..content = kSolidBlueAvatar;
+
+ await tester.pumpWidget(
+ MaterialApp(
+ home: Directionality(
+ textDirection: TextDirection.ltr,
+ child: GlobalStoreWidget(
+ child: PerAccountStoreWidget(
+ accountId: eg.selfAccount.id,
+ child: MessageContent(
+ message: message,
+ content: parseContent(html)))))));
+ await tester.pump(); // global store
+ await tester.pump(); // per-account store
+ debugNetworkImageHttpClientProvider = null;
+ }
+
+ testWidgets('single image', (tester) async {
+ const example = ContentExample.imageSingle;
+ await prepareContent(tester, example.html);
+ final expectedImages = (example.expectedNodes[0] as ImageNodeList).images;
+ final images = tester.widgetList(
+ find.byType(RealmContentNetworkImage));
+ check(images.map((i) => i.src.toString()).toList())
+ .deepEquals(expectedImages.map((n) => n.srcUrl));
+ });
+
+ testWidgets('multiple images', (tester) async {
+ const example = ContentExample.imageCluster;
+ await prepareContent(tester, example.html);
+ final expectedImages = (example.expectedNodes[1] as ImageNodeList).images;
+ final images = tester.widgetList(
+ find.byType(RealmContentNetworkImage));
+ check(images.map((i) => i.src.toString()).toList())
+ .deepEquals(expectedImages.map((n) => n.srcUrl));
+ });
+
+ testWidgets('content after image cluster', (tester) async {
+ const example = ContentExample.imageClusterThenContent;
+ await prepareContent(tester, example.html);
+ final expectedImages = (example.expectedNodes[1] as ImageNodeList).images;
+ final images = tester.widgetList(
+ find.byType(RealmContentNetworkImage));
+ check(images.map((i) => i.src.toString()).toList())
+ .deepEquals(expectedImages.map((n) => n.srcUrl));
+ });
+
+ testWidgets('multiple clusters of images', (tester) async {
+ const example = ContentExample.imageMultipleClusters;
+ await prepareContent(tester, example.html);
+ final expectedImages = (example.expectedNodes[1] as ImageNodeList).images
+ + (example.expectedNodes[4] as ImageNodeList).images;
+ final images = tester.widgetList(
+ find.byType(RealmContentNetworkImage));
+ check(images.map((i) => i.src.toString()).toList())
+ .deepEquals(expectedImages.map((n) => n.srcUrl));
+ });
+ });
+
group('RealmContentNetworkImage', () {
final authHeaders = authHeader(email: eg.selfAccount.email, apiKey: eg.selfAccount.apiKey);