2020-01-18 11:38:21 +03:00
|
|
|
/*
|
2020-01-24 16:45:29 +03:00
|
|
|
* Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
|
2022-04-22 08:34:09 +03:00
|
|
|
* Copyright (c) 2022, Peter Elliott <pelliott@serenityos.org>
|
2020-01-18 11:38:21 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 11:38:21 +03:00
|
|
|
*/
|
|
|
|
|
2022-12-23 12:25:00 +03:00
|
|
|
#include <AK/Forward.h>
|
2019-09-21 00:46:18 +03:00
|
|
|
#include <AK/StringBuilder.h>
|
2020-10-31 21:13:17 +03:00
|
|
|
#include <LibJS/MarkupGenerator.h>
|
2020-04-28 22:04:25 +03:00
|
|
|
#include <LibMarkdown/CodeBlock.h>
|
2021-09-10 22:36:29 +03:00
|
|
|
#include <LibMarkdown/Visitor.h>
|
2021-09-11 10:37:28 +03:00
|
|
|
#include <LibRegex/Regex.h>
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2020-04-28 22:04:25 +03:00
|
|
|
namespace Markdown {
|
|
|
|
|
2022-12-04 21:02:33 +03:00
|
|
|
DeprecatedString CodeBlock::render_to_html(bool) const
|
2019-09-21 00:46:18 +03:00
|
|
|
{
|
|
|
|
StringBuilder builder;
|
|
|
|
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("<pre>"sv);
|
2021-06-09 17:03:27 +03:00
|
|
|
|
2021-09-11 10:37:28 +03:00
|
|
|
if (m_style.length() >= 2)
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("<strong>"sv);
|
2021-09-11 10:37:28 +03:00
|
|
|
else if (m_style.length() >= 2)
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("<em>"sv);
|
2021-09-11 10:37:28 +03:00
|
|
|
|
2021-09-07 04:11:46 +03:00
|
|
|
if (m_language.is_empty())
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("<code>"sv);
|
2019-09-21 00:46:18 +03:00
|
|
|
else
|
2021-09-11 10:37:28 +03:00
|
|
|
builder.appendff("<code class=\"language-{}\">", escape_html_entities(m_language));
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2022-12-15 18:29:21 +03:00
|
|
|
if (m_language == "js") {
|
|
|
|
auto html_or_error = JS::MarkupGenerator::html_from_source(m_code);
|
|
|
|
if (html_or_error.is_error()) {
|
|
|
|
warnln("Could not render js code to html: {}", html_or_error.error());
|
|
|
|
builder.append(escape_html_entities(m_code));
|
|
|
|
} else {
|
|
|
|
builder.append(html_or_error.release_value());
|
|
|
|
}
|
|
|
|
} else {
|
2020-10-31 21:13:17 +03:00
|
|
|
builder.append(escape_html_entities(m_code));
|
2022-12-15 18:29:21 +03:00
|
|
|
}
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("</code>"sv);
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2021-09-11 10:37:28 +03:00
|
|
|
if (m_style.length() >= 2)
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("</strong>"sv);
|
2021-09-11 10:37:28 +03:00
|
|
|
else if (m_style.length() >= 2)
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("</em>"sv);
|
2021-09-11 10:37:28 +03:00
|
|
|
|
2022-07-11 20:32:29 +03:00
|
|
|
builder.append("</pre>\n"sv);
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2023-01-26 21:58:09 +03:00
|
|
|
return builder.to_deprecated_string();
|
2019-09-21 00:46:18 +03:00
|
|
|
}
|
|
|
|
|
2022-12-23 12:25:00 +03:00
|
|
|
Vector<DeprecatedString> CodeBlock::render_lines_for_terminal(size_t) const
|
2019-09-21 00:46:18 +03:00
|
|
|
{
|
2022-12-23 12:25:00 +03:00
|
|
|
Vector<DeprecatedString> lines;
|
|
|
|
|
|
|
|
// Do not indent too much if we are in the synopsis
|
|
|
|
auto indentation = " "sv;
|
|
|
|
if (m_current_section != nullptr) {
|
|
|
|
auto current_section_name = m_current_section->render_lines_for_terminal()[0];
|
|
|
|
if (current_section_name.contains("SYNOPSIS"sv))
|
|
|
|
indentation = " "sv;
|
2022-02-17 00:12:30 +03:00
|
|
|
}
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2022-12-23 12:25:00 +03:00
|
|
|
for (auto const& line : m_code.split('\n'))
|
|
|
|
lines.append(DeprecatedString::formatted("{}{}", indentation, line));
|
|
|
|
lines.append("");
|
|
|
|
|
|
|
|
return lines;
|
2019-09-21 00:46:18 +03:00
|
|
|
}
|
|
|
|
|
2021-09-10 22:36:29 +03:00
|
|
|
RecursionDecision CodeBlock::walk(Visitor& visitor) const
|
|
|
|
{
|
|
|
|
RecursionDecision rd = visitor.visit(*this);
|
|
|
|
if (rd != RecursionDecision::Recurse)
|
|
|
|
return rd;
|
|
|
|
|
|
|
|
rd = visitor.visit(m_code);
|
|
|
|
if (rd != RecursionDecision::Recurse)
|
|
|
|
return rd;
|
|
|
|
|
|
|
|
// Don't recurse on m_language and m_style.
|
|
|
|
|
|
|
|
// Normalize return value.
|
|
|
|
return RecursionDecision::Continue;
|
|
|
|
}
|
|
|
|
|
2022-04-26 02:48:13 +03:00
|
|
|
static Regex<ECMA262> open_fence_re("^ {0,3}(([\\`\\~])\\2{2,})\\s*([\\*_]*)\\s*([^\\*_\\s]*).*$");
|
|
|
|
static Regex<ECMA262> close_fence_re("^ {0,3}(([\\`\\~])\\2{2,})\\s*$");
|
2022-04-22 08:34:09 +03:00
|
|
|
|
|
|
|
static Optional<int> line_block_prefix(StringView const& line)
|
|
|
|
{
|
|
|
|
int characters = 0;
|
|
|
|
int indents = 0;
|
|
|
|
|
|
|
|
for (char ch : line) {
|
|
|
|
if (indents == 4)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (ch == ' ') {
|
|
|
|
++characters;
|
|
|
|
++indents;
|
|
|
|
} else if (ch == '\t') {
|
|
|
|
++characters;
|
|
|
|
indents = 4;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (indents == 4)
|
|
|
|
return characters;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-07-31 22:53:25 +03:00
|
|
|
OwnPtr<CodeBlock> CodeBlock::parse(LineIterator& lines, Heading* current_section)
|
2019-09-21 00:46:18 +03:00
|
|
|
{
|
|
|
|
if (lines.is_end())
|
2021-01-11 02:29:28 +03:00
|
|
|
return {};
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2022-04-22 08:34:09 +03:00
|
|
|
StringView line = *lines;
|
2022-04-26 02:48:13 +03:00
|
|
|
if (open_fence_re.match(line).success)
|
2022-07-31 22:53:25 +03:00
|
|
|
return parse_backticks(lines, current_section);
|
2019-09-21 00:46:18 +03:00
|
|
|
|
2022-04-22 08:34:09 +03:00
|
|
|
if (line_block_prefix(line).has_value())
|
|
|
|
return parse_indent(lines);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-07-31 22:53:25 +03:00
|
|
|
OwnPtr<CodeBlock> CodeBlock::parse_backticks(LineIterator& lines, Heading* current_section)
|
2022-04-22 08:34:09 +03:00
|
|
|
{
|
2019-09-21 00:46:18 +03:00
|
|
|
StringView line = *lines;
|
|
|
|
|
2021-09-11 10:37:28 +03:00
|
|
|
// Our Markdown extension: we allow
|
|
|
|
// specifying a style and a language
|
|
|
|
// for a code block, like so:
|
|
|
|
//
|
|
|
|
// ```**sh**
|
|
|
|
// $ echo hello friends!
|
|
|
|
// ````
|
|
|
|
//
|
|
|
|
// The code block will be made bold,
|
|
|
|
// and if possible syntax-highlighted
|
|
|
|
// as appropriate for a shell script.
|
2022-04-26 02:48:13 +03:00
|
|
|
|
|
|
|
auto matches = open_fence_re.match(line).capture_group_matches[0];
|
|
|
|
auto fence = matches[0].view.string_view();
|
|
|
|
auto style = matches[2].view.string_view();
|
|
|
|
auto language = matches[3].view.string_view();
|
2019-09-21 00:46:18 +03:00
|
|
|
|
|
|
|
++lines;
|
|
|
|
|
|
|
|
StringBuilder builder;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if (lines.is_end())
|
|
|
|
break;
|
|
|
|
line = *lines;
|
|
|
|
++lines;
|
2022-04-26 02:48:13 +03:00
|
|
|
|
|
|
|
auto close_match = close_fence_re.match(line);
|
|
|
|
if (close_match.success) {
|
|
|
|
auto close_fence = close_match.capture_group_matches[0][0].view.string_view();
|
|
|
|
if (close_fence[0] == fence[0] && close_fence.length() >= fence.length())
|
|
|
|
break;
|
|
|
|
}
|
2019-09-21 00:46:18 +03:00
|
|
|
builder.append(line);
|
2022-04-26 02:57:58 +03:00
|
|
|
builder.append('\n');
|
2019-09-21 00:46:18 +03:00
|
|
|
}
|
|
|
|
|
2023-01-26 21:58:09 +03:00
|
|
|
return make<CodeBlock>(language, style, builder.to_deprecated_string(), current_section);
|
2019-09-21 00:46:18 +03:00
|
|
|
}
|
2020-04-28 22:04:25 +03:00
|
|
|
|
2022-04-22 08:34:09 +03:00
|
|
|
OwnPtr<CodeBlock> CodeBlock::parse_indent(LineIterator& lines)
|
|
|
|
{
|
|
|
|
StringBuilder builder;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if (lines.is_end())
|
|
|
|
break;
|
|
|
|
StringView line = *lines;
|
|
|
|
|
|
|
|
auto prefix_length = line_block_prefix(line);
|
|
|
|
if (!prefix_length.has_value())
|
|
|
|
break;
|
|
|
|
|
|
|
|
line = line.substring_view(prefix_length.value());
|
|
|
|
++lines;
|
|
|
|
|
|
|
|
builder.append(line);
|
2022-04-26 02:57:58 +03:00
|
|
|
builder.append('\n');
|
2022-04-22 08:34:09 +03:00
|
|
|
}
|
|
|
|
|
2023-01-26 21:58:09 +03:00
|
|
|
return make<CodeBlock>("", "", builder.to_deprecated_string(), nullptr);
|
2022-04-22 08:34:09 +03:00
|
|
|
}
|
2020-04-28 22:04:25 +03:00
|
|
|
}
|