/* * Copyright (c) 2019-2020, Sergey Bugaev * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include namespace Markdown { Text::Style CodeBlock::style() const { if (m_style_spec.spans().is_empty()) return {}; return m_style_spec.spans()[0].style; } String CodeBlock::style_language() const { if (m_style_spec.spans().is_empty()) return {}; return m_style_spec.spans()[0].text; } String CodeBlock::render_to_html() const { StringBuilder builder; String style_language = this->style_language(); Text::Style style = this->style(); if (style.strong) builder.append(""); if (style.emph) builder.append(""); if (style_language.is_null()) builder.append(""); else builder.appendf("", style_language.characters()); builder.append(escape_html_entities(m_code)); builder.append(""); if (style.emph) builder.append(""); if (style.strong) builder.append(""); builder.append('\n'); return builder.build(); } String CodeBlock::render_for_terminal() const { StringBuilder builder; Text::Style style = this->style(); bool needs_styling = style.strong || style.emph; if (needs_styling) { builder.append("\033["); bool first = true; if (style.strong) { builder.append('1'); first = false; } if (style.emph) { if (!first) builder.append(';'); builder.append('4'); } builder.append('m'); } builder.append(m_code); if (needs_styling) builder.append("\033[0m"); builder.append("\n\n"); return builder.build(); } OwnPtr CodeBlock::parse(Vector::ConstIterator& lines) { if (lines.is_end()) return nullptr; constexpr auto tick_tick_tick = "```"; StringView line = *lines; if (!line.starts_with(tick_tick_tick)) return nullptr; // 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. StringView style_spec = line.substring_view(3, line.length() - 3); auto spec = Text::parse(style_spec); if (!spec.has_value()) return nullptr; ++lines; bool first = true; StringBuilder builder; while (true) { if (lines.is_end()) break; line = *lines; ++lines; if (line == tick_tick_tick) break; if (!first) builder.append('\n'); builder.append(line); first = false; } return make(move(spec.value()), builder.build()); } }