|
|
|
@@ -15,6 +15,7 @@ import 'package:island/widgets/content/cloud_files.dart';
|
|
|
|
|
import 'package:island/widgets/content/markdown_latex.dart';
|
|
|
|
|
import 'package:markdown/markdown.dart' as markdown;
|
|
|
|
|
import 'package:markdown_widget/markdown_widget.dart';
|
|
|
|
|
import 'package:material_symbols_icons/symbols.dart';
|
|
|
|
|
import 'package:styled_widget/styled_widget.dart';
|
|
|
|
|
import 'package:url_launcher/url_launcher.dart';
|
|
|
|
|
|
|
|
|
@@ -62,6 +63,31 @@ class MarkdownTextContent extends HookConsumerWidget {
|
|
|
|
|
final config =
|
|
|
|
|
isDark ? MarkdownConfig.darkConfig : MarkdownConfig.defaultConfig;
|
|
|
|
|
|
|
|
|
|
final onMentionTap = useCallback((String type, String id) {
|
|
|
|
|
final fullPath = '/$type/$id';
|
|
|
|
|
context.push(fullPath);
|
|
|
|
|
}, [context]);
|
|
|
|
|
|
|
|
|
|
final mentionGenerator = MentionChipGenerator(
|
|
|
|
|
backgroundColor: Theme.of(context).colorScheme.secondary,
|
|
|
|
|
foregroundColor: Theme.of(context).colorScheme.onSecondary,
|
|
|
|
|
onTap: onMentionTap,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
final highlightGenerator = HighlightGenerator(
|
|
|
|
|
highlightColor: Theme.of(context).colorScheme.primaryContainer,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
final spoilerRevealed = useState(false);
|
|
|
|
|
|
|
|
|
|
final spoilerGenerator = SpoilerGenerator(
|
|
|
|
|
backgroundColor: Theme.of(context).colorScheme.tertiary,
|
|
|
|
|
foregroundColor: Theme.of(context).colorScheme.onTertiary,
|
|
|
|
|
outlineColor: Theme.of(context).colorScheme.outline,
|
|
|
|
|
revealed: spoilerRevealed.value,
|
|
|
|
|
onToggle: () => spoilerRevealed.value = !spoilerRevealed.value,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return MarkdownBlock(
|
|
|
|
|
data: content,
|
|
|
|
|
selectable: isSelectable,
|
|
|
|
@@ -202,6 +228,7 @@ class MarkdownTextContent extends HookConsumerWidget {
|
|
|
|
|
generator: MarkdownTextContent.buildGenerator(
|
|
|
|
|
isDark: isDark,
|
|
|
|
|
linesMargin: linesMargin,
|
|
|
|
|
generators: [mentionGenerator, highlightGenerator, spoilerGenerator],
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
@@ -209,11 +236,14 @@ class MarkdownTextContent extends HookConsumerWidget {
|
|
|
|
|
static MarkdownGenerator buildGenerator({
|
|
|
|
|
bool isDark = false,
|
|
|
|
|
EdgeInsets? linesMargin,
|
|
|
|
|
List<dynamic> generators = const [],
|
|
|
|
|
}) {
|
|
|
|
|
return MarkdownGenerator(
|
|
|
|
|
generators: [latexGenerator],
|
|
|
|
|
generators: [latexGenerator, ...generators],
|
|
|
|
|
inlineSyntaxList: [
|
|
|
|
|
_MetionInlineSyntax(),
|
|
|
|
|
_HighlightInlineSyntax(),
|
|
|
|
|
_SpoilerInlineSyntax(),
|
|
|
|
|
_StickerInlineSyntax(),
|
|
|
|
|
LatexSyntax(isDark),
|
|
|
|
|
],
|
|
|
|
@@ -237,9 +267,12 @@ class _MetionInlineSyntax extends markdown.InlineSyntax {
|
|
|
|
|
"c" => 'chat',
|
|
|
|
|
_ => '',
|
|
|
|
|
};
|
|
|
|
|
final anchor = markdown.Element.text('a', alias)
|
|
|
|
|
..attributes['href'] = Uri.encodeFull('solian://$type/${parts.last}');
|
|
|
|
|
parser.addNode(anchor);
|
|
|
|
|
final element =
|
|
|
|
|
markdown.Element('mention-chip', [markdown.Text(alias)])
|
|
|
|
|
..attributes['alias'] = alias
|
|
|
|
|
..attributes['type'] = type
|
|
|
|
|
..attributes['id'] = parts.last;
|
|
|
|
|
parser.addNode(element);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
@@ -258,3 +291,260 @@ class _StickerInlineSyntax extends markdown.InlineSyntax {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _HighlightInlineSyntax extends markdown.InlineSyntax {
|
|
|
|
|
_HighlightInlineSyntax() : super(r'==([^=]+)==');
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
bool onMatch(markdown.InlineParser parser, Match match) {
|
|
|
|
|
final text = match[1]!;
|
|
|
|
|
final element = markdown.Element('highlight', [markdown.Text(text)]);
|
|
|
|
|
parser.addNode(element);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _SpoilerInlineSyntax extends markdown.InlineSyntax {
|
|
|
|
|
_SpoilerInlineSyntax() : super(r'=!([^!]+)!=');
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
bool onMatch(markdown.InlineParser parser, Match match) {
|
|
|
|
|
final text = match[1]!;
|
|
|
|
|
final element = markdown.Element('spoiler', [markdown.Text(text)]);
|
|
|
|
|
parser.addNode(element);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class MentionSpanNodeGenerator {
|
|
|
|
|
final Color backgroundColor;
|
|
|
|
|
final Color foregroundColor;
|
|
|
|
|
final void Function(String type, String id) onTap;
|
|
|
|
|
|
|
|
|
|
MentionSpanNodeGenerator({
|
|
|
|
|
required this.backgroundColor,
|
|
|
|
|
required this.foregroundColor,
|
|
|
|
|
required this.onTap,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
SpanNode? call(
|
|
|
|
|
String tag,
|
|
|
|
|
Map<String, String> attributes,
|
|
|
|
|
List<SpanNode> children,
|
|
|
|
|
) {
|
|
|
|
|
if (tag == 'mention-chip') {
|
|
|
|
|
return MentionChipSpanNode(
|
|
|
|
|
attributes: attributes,
|
|
|
|
|
backgroundColor: backgroundColor,
|
|
|
|
|
foregroundColor: foregroundColor,
|
|
|
|
|
onTap: onTap,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class MentionChipGenerator extends SpanNodeGeneratorWithTag {
|
|
|
|
|
MentionChipGenerator({
|
|
|
|
|
required Color backgroundColor,
|
|
|
|
|
required Color foregroundColor,
|
|
|
|
|
required void Function(String type, String id) onTap,
|
|
|
|
|
}) : super(
|
|
|
|
|
tag: 'mention-chip',
|
|
|
|
|
generator: (
|
|
|
|
|
markdown.Element element,
|
|
|
|
|
MarkdownConfig config,
|
|
|
|
|
WidgetVisitor visitor,
|
|
|
|
|
) {
|
|
|
|
|
return MentionChipSpanNode(
|
|
|
|
|
attributes: element.attributes,
|
|
|
|
|
backgroundColor: backgroundColor,
|
|
|
|
|
foregroundColor: foregroundColor,
|
|
|
|
|
onTap: onTap,
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class MentionChipSpanNode extends SpanNode {
|
|
|
|
|
final Map<String, String> attributes;
|
|
|
|
|
final Color backgroundColor;
|
|
|
|
|
final Color foregroundColor;
|
|
|
|
|
final void Function(String type, String id) onTap;
|
|
|
|
|
|
|
|
|
|
MentionChipSpanNode({
|
|
|
|
|
required this.attributes,
|
|
|
|
|
required this.backgroundColor,
|
|
|
|
|
required this.foregroundColor,
|
|
|
|
|
required this.onTap,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
InlineSpan build() {
|
|
|
|
|
final alias = attributes['alias'] ?? '';
|
|
|
|
|
final type = attributes['type'] ?? '';
|
|
|
|
|
final id = attributes['id'] ?? '';
|
|
|
|
|
|
|
|
|
|
final parts = alias.substring(1).split('/');
|
|
|
|
|
|
|
|
|
|
return WidgetSpan(
|
|
|
|
|
alignment: PlaceholderAlignment.middle,
|
|
|
|
|
child: InkWell(
|
|
|
|
|
onTap: () => onTap(type, id),
|
|
|
|
|
child: Container(
|
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
|
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
|
color: backgroundColor.withOpacity(0.1),
|
|
|
|
|
borderRadius: BorderRadius.circular(32),
|
|
|
|
|
),
|
|
|
|
|
child: Row(
|
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
|
spacing: 6,
|
|
|
|
|
children: [
|
|
|
|
|
Container(
|
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
|
color: backgroundColor.withOpacity(0.5),
|
|
|
|
|
borderRadius: const BorderRadius.all(Radius.circular(32)),
|
|
|
|
|
),
|
|
|
|
|
child: Icon(
|
|
|
|
|
switch (parts.first.isEmpty ? 'u' : parts.first) {
|
|
|
|
|
'c' => Symbols.forum_rounded,
|
|
|
|
|
'r' => Symbols.group_rounded,
|
|
|
|
|
'u' => Symbols.person_rounded,
|
|
|
|
|
'p' => Symbols.edit_rounded,
|
|
|
|
|
_ => Symbols.person_rounded,
|
|
|
|
|
},
|
|
|
|
|
size: 14,
|
|
|
|
|
color: foregroundColor,
|
|
|
|
|
fill: 1,
|
|
|
|
|
).padding(all: 2),
|
|
|
|
|
),
|
|
|
|
|
Text(
|
|
|
|
|
parts.last,
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
color: backgroundColor,
|
|
|
|
|
fontSize: 14,
|
|
|
|
|
fontWeight: FontWeight.w500,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class HighlightGenerator extends SpanNodeGeneratorWithTag {
|
|
|
|
|
HighlightGenerator({required Color highlightColor})
|
|
|
|
|
: super(
|
|
|
|
|
tag: 'highlight',
|
|
|
|
|
generator: (
|
|
|
|
|
markdown.Element element,
|
|
|
|
|
MarkdownConfig config,
|
|
|
|
|
WidgetVisitor visitor,
|
|
|
|
|
) {
|
|
|
|
|
return HighlightSpanNode(
|
|
|
|
|
text: element.textContent,
|
|
|
|
|
highlightColor: highlightColor,
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class HighlightSpanNode extends SpanNode {
|
|
|
|
|
final String text;
|
|
|
|
|
final Color highlightColor;
|
|
|
|
|
|
|
|
|
|
HighlightSpanNode({required this.text, required this.highlightColor});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
InlineSpan build() {
|
|
|
|
|
return TextSpan(
|
|
|
|
|
text: text,
|
|
|
|
|
style: TextStyle(backgroundColor: highlightColor),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class SpoilerGenerator extends SpanNodeGeneratorWithTag {
|
|
|
|
|
SpoilerGenerator({
|
|
|
|
|
required Color backgroundColor,
|
|
|
|
|
required Color foregroundColor,
|
|
|
|
|
required Color outlineColor,
|
|
|
|
|
required bool revealed,
|
|
|
|
|
required VoidCallback onToggle,
|
|
|
|
|
}) : super(
|
|
|
|
|
tag: 'spoiler',
|
|
|
|
|
generator: (
|
|
|
|
|
markdown.Element element,
|
|
|
|
|
MarkdownConfig config,
|
|
|
|
|
WidgetVisitor visitor,
|
|
|
|
|
) {
|
|
|
|
|
return SpoilerSpanNode(
|
|
|
|
|
text: element.textContent,
|
|
|
|
|
backgroundColor: backgroundColor,
|
|
|
|
|
foregroundColor: foregroundColor,
|
|
|
|
|
outlineColor: outlineColor,
|
|
|
|
|
revealed: revealed,
|
|
|
|
|
onToggle: onToggle,
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class SpoilerSpanNode extends SpanNode {
|
|
|
|
|
final String text;
|
|
|
|
|
final Color backgroundColor;
|
|
|
|
|
final Color foregroundColor;
|
|
|
|
|
final Color outlineColor;
|
|
|
|
|
final bool revealed;
|
|
|
|
|
final VoidCallback onToggle;
|
|
|
|
|
|
|
|
|
|
SpoilerSpanNode({
|
|
|
|
|
required this.text,
|
|
|
|
|
required this.backgroundColor,
|
|
|
|
|
required this.foregroundColor,
|
|
|
|
|
required this.outlineColor,
|
|
|
|
|
required this.revealed,
|
|
|
|
|
required this.onToggle,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
InlineSpan build() {
|
|
|
|
|
return WidgetSpan(
|
|
|
|
|
child: InkWell(
|
|
|
|
|
onTap: onToggle,
|
|
|
|
|
child: Container(
|
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 4, vertical: 2),
|
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
|
color: revealed ? Colors.transparent : backgroundColor,
|
|
|
|
|
border: revealed ? Border.all(color: outlineColor, width: 1) : null,
|
|
|
|
|
borderRadius: BorderRadius.circular(4),
|
|
|
|
|
),
|
|
|
|
|
child:
|
|
|
|
|
revealed
|
|
|
|
|
? Row(
|
|
|
|
|
spacing: 6,
|
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
|
children: [Icon(Symbols.visibility, size: 18), Text(text)],
|
|
|
|
|
)
|
|
|
|
|
: Row(
|
|
|
|
|
spacing: 6,
|
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
|
children: [
|
|
|
|
|
Icon(
|
|
|
|
|
Symbols.visibility_off,
|
|
|
|
|
color: foregroundColor,
|
|
|
|
|
size: 18,
|
|
|
|
|
),
|
|
|
|
|
Text(text, style: TextStyle(color: foregroundColor)),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|