|             Line data    Source code 
       1              : /*
       2              :  *   Famedly Matrix SDK
       3              :  *   Copyright (C) 2020, 2021 Famedly GmbH
       4              :  *
       5              :  *   This program is free software: you can redistribute it and/or modify
       6              :  *   it under the terms of the GNU Affero General Public License as
       7              :  *   published by the Free Software Foundation, either version 3 of the
       8              :  *   License, or (at your option) any later version.
       9              :  *
      10              :  *   This program is distributed in the hope that it will be useful,
      11              :  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
      12              :  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
      13              :  *   GNU Affero General Public License for more details.
      14              :  *
      15              :  *   You should have received a copy of the GNU Affero General Public License
      16              :  *   along with this program.  If not, see <https://www.gnu.org/licenses/>.
      17              :  */
      18              : 
      19              : import 'dart:convert';
      20              : 
      21              : import 'package:markdown/markdown.dart';
      22              : 
      23              : const htmlAttrEscape = HtmlEscape(HtmlEscapeMode.attribute);
      24              : 
      25              : class SpoilerSyntax extends DelimiterSyntax {
      26           11 :   SpoilerSyntax()
      27           11 :       : super(
      28              :           r'\|\|',
      29              :           requiresDelimiterRun: true,
      30           22 :           tags: [DelimiterTag('span', 2)],
      31              :         );
      32              : 
      33            2 :   @override
      34              :   Iterable<Node>? close(
      35              :     InlineParser parser,
      36              :     Delimiter opener,
      37              :     Delimiter closer, {
      38              :     required String tag,
      39              :     required List<Node> Function() getChildren,
      40              :   }) {
      41            2 :     final children = getChildren();
      42            2 :     final newChildren = <Node>[];
      43              :     var searchingForReason = true;
      44              :     var reason = '';
      45            4 :     for (final child in children) {
      46              :       // If we already found a reason, let's just use our child nodes as-is
      47              :       if (!searchingForReason) {
      48            2 :         newChildren.add(child);
      49              :         continue;
      50              :       }
      51            2 :       if (child is Text) {
      52            4 :         final ix = child.text.indexOf('|');
      53            2 :         if (ix > 0) {
      54            6 :           reason += child.text.substring(0, ix);
      55           10 :           newChildren.add(Text(child.text.substring(ix + 1)));
      56              :           searchingForReason = false;
      57              :         } else {
      58            4 :           reason += child.text;
      59              :         }
      60              :       } else {
      61              :         // if we don't have a text node as reason we just want to cancel this whole thing
      62              :         break;
      63              :       }
      64              :     }
      65              :     // if we were still searching for a reason that means there was none - use the original children!
      66              :     final element =
      67            2 :         Element('span', searchingForReason ? children : newChildren);
      68            4 :     element.attributes['data-mx-spoiler'] =
      69            2 :         searchingForReason ? '' : htmlAttrEscape.convert(reason);
      70            2 :     return <Node>[element];
      71              :   }
      72              : }
      73              : 
      74              : class EmoteSyntax extends InlineSyntax {
      75              :   final Map<String, Map<String, String>> Function()? getEmotePacks;
      76              :   Map<String, Map<String, String>>? emotePacks;
      77           22 :   EmoteSyntax(this.getEmotePacks) : super(r':(?:([-\w]+)~)?([-\w]+):');
      78              : 
      79            2 :   @override
      80              :   bool onMatch(InlineParser parser, Match match) {
      81            6 :     final emotePacks = this.emotePacks ??= getEmotePacks?.call() ?? {};
      82            2 :     final pack = match[1] ?? '';
      83            2 :     final emote = match[2];
      84              :     String? mxc;
      85            2 :     if (pack.isEmpty) {
      86              :       // search all packs
      87            4 :       for (final emotePack in emotePacks.values) {
      88            2 :         mxc = emotePack[emote];
      89              :         if (mxc != null) {
      90              :           break;
      91              :         }
      92              :       }
      93              :     } else {
      94            4 :       mxc = emotePacks[pack]?[emote];
      95              :     }
      96              :     if (mxc == null) {
      97              :       // emote not found. Insert the whole thing as plain text
      98            6 :       parser.addNode(Text(match[0]!));
      99              :       return true;
     100              :     }
     101            2 :     final element = Element.empty('img');
     102            4 :     element.attributes['data-mx-emoticon'] = '';
     103            6 :     element.attributes['src'] = htmlAttrEscape.convert(mxc);
     104            8 :     element.attributes['alt'] = htmlAttrEscape.convert(':$emote:');
     105            8 :     element.attributes['title'] = htmlAttrEscape.convert(':$emote:');
     106            4 :     element.attributes['height'] = '32';
     107            4 :     element.attributes['vertical-align'] = 'middle';
     108            2 :     parser.addNode(element);
     109              :     return true;
     110              :   }
     111              : }
     112              : 
     113              : class InlineLatexSyntax extends DelimiterSyntax {
     114           22 :   InlineLatexSyntax() : super(r'\$([^\s$]([^\$]*[^\s$])?)\$');
     115              : 
     116            2 :   @override
     117              :   bool onMatch(InlineParser parser, Match match) {
     118              :     final element =
     119           10 :         Element('span', [Element.text('code', htmlEscape.convert(match[1]!))]);
     120            8 :     element.attributes['data-mx-maths'] = htmlAttrEscape.convert(match[1]!);
     121            2 :     parser.addNode(element);
     122              :     return true;
     123              :   }
     124              : }
     125              : 
     126              : // We also want to allow single-lines of like "$$latex$$"
     127              : class BlockLatexSyntax extends BlockSyntax {
     128           11 :   @override
     129           11 :   RegExp get pattern => RegExp(r'^[ ]{0,3}\$\$(.*)$');
     130              : 
     131              :   final endPattern = RegExp(r'^(.*)\$\$\s*$');
     132              : 
     133            0 :   @override
     134              :   List<Line?> parseChildLines(BlockParser parser) {
     135            0 :     final childLines = <Line>[];
     136              :     var first = true;
     137            0 :     while (!parser.isDone) {
     138            0 :       final match = endPattern.firstMatch(parser.current.content);
     139            0 :       if (match == null || (first && match[1]!.trim().isEmpty)) {
     140            0 :         childLines.add(parser.current);
     141            0 :         parser.advance();
     142              :       } else {
     143            0 :         childLines.add(Line(match[1]!));
     144            0 :         parser.advance();
     145              :         break;
     146              :       }
     147              :       first = false;
     148              :     }
     149              :     return childLines;
     150              :   }
     151              : 
     152            0 :   @override
     153              :   Node parse(BlockParser parser) {
     154            0 :     final childLines = parseChildLines(parser);
     155              :     // we use .substring(2) as childLines will *always* contain the first two '$$'
     156            0 :     final latex = childLines.join('\n').trim().substring(2).trim();
     157            0 :     final element = Element('div', [
     158            0 :       Element('pre', [Element.text('code', htmlEscape.convert(latex))]),
     159              :     ]);
     160            0 :     element.attributes['data-mx-maths'] = htmlAttrEscape.convert(latex);
     161              :     return element;
     162              :   }
     163              : }
     164              : 
     165              : class PillSyntax extends InlineSyntax {
     166           11 :   PillSyntax()
     167           11 :       : super(
     168              :           r'([@#!][^\s:]*:(?:[^\s]+\.\w+|[\d\.]+|\[[a-fA-F0-9:]+\])(?::\d+)?)',
     169              :         );
     170              : 
     171            2 :   @override
     172              :   bool onMatch(InlineParser parser, Match match) {
     173            4 :     if (match.start > 0 &&
     174           12 :         !RegExp(r'[\s.!?:;\(]').hasMatch(match.input[match.start - 1])) {
     175            0 :       parser.addNode(Text(match[0]!));
     176              :       return true;
     177              :     }
     178            2 :     final identifier = match[1]!;
     179            4 :     final element = Element.text('a', htmlEscape.convert(identifier));
     180            4 :     element.attributes['href'] =
     181            4 :         htmlAttrEscape.convert('https://matrix.to/#/$identifier');
     182            2 :     parser.addNode(element);
     183              :     return true;
     184              :   }
     185              : }
     186              : 
     187              : class MentionSyntax extends InlineSyntax {
     188              :   final String? Function(String)? getMention;
     189           22 :   MentionSyntax(this.getMention) : super(r'(@(?:\[[^\]:]+\]|\w+)(?:#\w+)?)');
     190              : 
     191            2 :   @override
     192              :   bool onMatch(InlineParser parser, Match match) {
     193            6 :     final mention = getMention?.call(match[1]!);
     194            4 :     if ((match.start > 0 &&
     195           12 :             !RegExp(r'[\s.!?:;\(]').hasMatch(match.input[match.start - 1])) ||
     196              :         mention == null) {
     197            6 :       parser.addNode(Text(match[0]!));
     198              :       return true;
     199              :     }
     200            6 :     final element = Element.text('a', htmlEscape.convert(match[1]!));
     201            4 :     element.attributes['href'] =
     202            4 :         htmlAttrEscape.convert('https://matrix.to/#/$mention');
     203            2 :     parser.addNode(element);
     204              :     return true;
     205              :   }
     206              : }
     207              : 
     208           11 : String markdown(
     209              :   String text, {
     210              :   Map<String, Map<String, String>> Function()? getEmotePacks,
     211              :   String? Function(String)? getMention,
     212              :   bool convertLinebreaks = true,
     213              : }) {
     214           11 :   var ret = markdownToHtml(
     215           11 :     text.replaceNewlines(),
     216           11 :     extensionSet: ExtensionSet.gitHubFlavored,
     217           11 :     blockSyntaxes: [
     218           11 :       BlockLatexSyntax(),
     219              :     ],
     220           11 :     inlineSyntaxes: [
     221           11 :       StrikethroughSyntax(),
     222           11 :       SpoilerSyntax(),
     223           11 :       EmoteSyntax(getEmotePacks),
     224           11 :       PillSyntax(),
     225           11 :       MentionSyntax(getMention),
     226           11 :       InlineLatexSyntax(),
     227              :     ],
     228              :   );
     229              : 
     230           33 :   var stripPTags = '<p>'.allMatches(ret).length <= 1;
     231              :   if (stripPTags) {
     232              :     const otherBlockTags = {
     233              :       'table',
     234              :       'pre',
     235              :       'ol',
     236              :       'ul',
     237              :       'h1',
     238              :       'h2',
     239              :       'h3',
     240              :       'h4',
     241              :       'h5',
     242              :       'h6',
     243              :       'blockquote',
     244              :       'div',
     245              :     };
     246           18 :     for (final tag in otherBlockTags) {
     247              :       // we check for the close tag as the opening one might have attributes
     248           18 :       if (ret.contains('</$tag>')) {
     249              :         stripPTags = false;
     250              :         break;
     251              :       }
     252              :     }
     253              :   }
     254              :   ret = ret
     255           11 :       .trim()
     256              :       // Remove trailing linebreaks
     257           22 :       .replaceAll(RegExp(r'(<br />)+$'), '');
     258              :   if (convertLinebreaks) {
     259              :     // Only convert linebreaks which are not in <pre> blocks
     260            9 :     ret = ret.convertLinebreaksToBr('p');
     261              :     // Delete other linebreaks except for pre blocks:
     262            9 :     ret = ret.convertLinebreaksToBr('pre', exclude: true, replaceWith: '');
     263              :   }
     264              : 
     265              :   if (stripPTags) {
     266           18 :     ret = ret.replaceAll('<p>', '').replaceAll('</p>', '');
     267              :   }
     268              : 
     269              :   return ret;
     270              : }
     271              : 
     272              : extension on String {
     273           11 :   String replaceNewlines() {
     274              :     // RegEx for at least 3 following \n
     275           11 :     final regExp = RegExp(r'(\n{3,})');
     276              : 
     277           13 :     return replaceAllMapped(regExp, (match) {
     278            2 :       final newLineGroup = match.group(0)!;
     279              :       return newLineGroup
     280            2 :           .replaceAll('\n', '<br/>')
     281            2 :           .replaceFirst('<br/><br/>', '\n\n');
     282              :     });
     283              :   }
     284              : 
     285            9 :   String convertLinebreaksToBr(
     286              :     String tagName, {
     287              :     bool exclude = false,
     288              :     String replaceWith = '<br/>',
     289              :   }) {
     290           18 :     final parts = split('$tagName>');
     291              :     var convertLinebreaks = exclude;
     292           27 :     for (var i = 0; i < parts.length; i++) {
     293           27 :       if (convertLinebreaks) parts[i] = parts[i].replaceAll('\n', replaceWith);
     294              :       convertLinebreaks = !convertLinebreaks;
     295              :     }
     296           18 :     return parts.join('$tagName>');
     297              :   }
     298              : }
         |