diff --git a/lib/model/content.dart b/lib/model/content.dart index f4d0faf44c..257d49e9ff 100644 --- a/lib/model/content.dart +++ b/lib/model/content.dart @@ -1016,6 +1016,7 @@ class _ZulipContentParser { assert(_debugParserContext == _ParserContext.block); final List result = []; final List currentParagraph = []; + List imageNodes = []; void consumeParagraph() { final parsed = parseBlockInline(currentParagraph); result.add(ParagraphNode( @@ -1029,13 +1030,36 @@ class _ZulipContentParser { if (node is dom.Text && (node.text == '\n')) continue; if (_isPossibleInlineNode(node)) { + if (imageNodes.isNotEmpty) { + result.add(ImageNodeList(imageNodes)); + imageNodes = []; + // In a context where paragraphs are implicit it + // should be impossible to have more paragraph + // content after image previews. + result.add(ParagraphNode( + wasImplicit: true, + links: null, + nodes: [UnimplementedInlineContentNode(htmlNode: node)] + )); + continue; + } currentParagraph.add(node); continue; } if (currentParagraph.isNotEmpty) consumeParagraph(); - result.add(parseBlockContent(node)); + final block = parseBlockContent(node); + if (block is ImageNode) { + imageNodes.add(block); + continue; + } + if (imageNodes.isNotEmpty) { + result.add(ImageNodeList(imageNodes)); + imageNodes = []; + } + result.add(block); } if (currentParagraph.isNotEmpty) consumeParagraph(); + if (imageNodes.isNotEmpty) result.add(ImageNodeList(imageNodes)); return result; } diff --git a/lib/widgets/content.dart b/lib/widgets/content.dart index f22108ec15..e2a45923eb 100644 --- a/lib/widgets/content.dart +++ b/lib/widgets/content.dart @@ -87,6 +87,10 @@ class BlockContentList extends StatelessWidget { } else if (node is ImageNodeList) { return MessageImageList(node: node); } else if (node is ImageNode) { + assert(false, + "[ImageNode] not allowed in [BlockContentList]. " + "It should be wrapped in [ImageNodeList]." + ); return MessageImage(node: node); } else if (node is UnimplementedBlockContentNode) { return Text.rich(_errorUnimplemented(node)); diff --git a/test/model/content_test.dart b/test/model/content_test.dart index 8641e8a7ce..e40aeb8cf3 100644 --- a/test/model/content_test.dart +++ b/test/model/content_test.dart @@ -508,6 +508,33 @@ void main() { ]), ]); + testParse('content after image cluster', + '

content ' + 'icon.png ' + 'icon.png

\n' + '
' + '' + '
' + '
' + '' + '
' + '

more content

', + const [ + 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'), + ]), + ]); + testParse('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" '

' @@ -554,6 +581,71 @@ void main() { ImageNode(srcUrl: 'https://uploads.zulipusercontent.net/51b70540cf6a5b3c8a0b919c893b8abddd447e88/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e673f763d33'), ]), ]); + + // TODO: maybe delete this + testParse('image as immediate child in implicit paragraph', + // "* https://chat.zulip.org/user_avatars/2/realm/icon.png" + '

', + const [ + ListNode(ListStyle.unordered, [[ + ImageNodeList([ + ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png'), + ]), + ]]), + ]); + + testParse('image cluster in implicit paragraph', + // "* [icon.png](https://chat.zulip.org/user_avatars/2/realm/icon.png) [icon.png](https://chat.zulip.org/user_avatars/2/realm/icon.png?version=2)" + '', + const [ + ListNode(ListStyle.unordered, [[ + ParagraphNode(wasImplicit: true, links: null, nodes: [ + 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'), + ]), + ]]), + ]); + + testParse('impossible content after image cluster in implicit paragraph', + '', + [ + ListNode(ListStyle.unordered, [[ + const ParagraphNode(wasImplicit: true, links: null, nodes: [ + LinkNode(url: 'https://chat.zulip.org/user_avatars/2/realm/icon.png', nodes: [TextNode('icon.png')]), + TextNode(' '), + ]), + const ImageNodeList([ + ImageNode(srcUrl: 'https://chat.zulip.org/user_avatars/2/realm/icon.png'), + ]), + ParagraphNode(wasImplicit: true, links: null, nodes: [ + inlineUnimplemented('Some content'), + ]) + ]]), + ]); }); testParse('parse nested lists, quotes, headings, code blocks', diff --git a/test/widgets/content_test.dart b/test/widgets/content_test.dart index 61611df677..aa5cd45b86 100644 --- a/test/widgets/content_test.dart +++ b/test/widgets/content_test.dart @@ -388,6 +388,42 @@ void main() { 'https://uploads.zulipusercontent.net/51b70540cf6a5b3c8a0b919c893b8abddd447e88/68747470733a2f2f656e2e77696b6970656469612e6f72672f7374617469632f696d616765732f69636f6e732f77696b6970656469612e706e673f763d33', ]); }); + + testWidgets('image as immediate child in list item', (tester) async { + // "* https://chat.zulip.org/user_avatars/2/realm/icon.png" + await prepareContent(tester, + ''); + final images = tester.widgetList(find.byType(RealmContentNetworkImage)); + check(images.map((i) => i.src.toString()).toList()) + .deepEquals([ + 'https://chat.zulip.org/user_avatars/2/realm/icon.png', + ]); + }); + + testWidgets('image cluster in list item', (tester) async { + // "* [icon.png](https://chat.zulip.org/user_avatars/2/realm/icon.png) [icon.png](https://chat.zulip.org/user_avatars/2/realm/icon.png?version=2)" + await prepareContent(tester, + ''); + final images = tester.widgetList(find.byType(RealmContentNetworkImage)); + check(images.map((i) => i.src.toString()).toList()) + .deepEquals([ + 'https://chat.zulip.org/user_avatars/2/realm/icon.png', + 'https://chat.zulip.org/user_avatars/2/realm/icon.png?version=2', + ]); + }); }); group('RealmContentNetworkImage', () {