Text.h 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*
  2. * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
  3. * Copyright (c) 2021, Peter Elliott <pelliott@serenityos.org>
  4. * Copyright (c) 2022, the SerenityOS developers.
  5. *
  6. * SPDX-License-Identifier: BSD-2-Clause
  7. */
  8. #pragma once
  9. #include <AK/Noncopyable.h>
  10. #include <AK/NonnullOwnPtrVector.h>
  11. #include <AK/OwnPtr.h>
  12. #include <AK/RecursionDecision.h>
  13. #include <AK/String.h>
  14. #include <LibMarkdown/Forward.h>
  15. namespace Markdown {
  16. class Text final {
  17. public:
  18. class Node {
  19. public:
  20. virtual void render_to_html(StringBuilder& builder) const = 0;
  21. virtual void render_for_terminal(StringBuilder& builder) const = 0;
  22. virtual size_t terminal_length() const = 0;
  23. virtual RecursionDecision walk(Visitor&) const = 0;
  24. virtual ~Node() = default;
  25. };
  26. class EmphasisNode : public Node {
  27. public:
  28. bool strong;
  29. NonnullOwnPtr<Node> child;
  30. EmphasisNode(bool strong, NonnullOwnPtr<Node> child)
  31. : strong(strong)
  32. , child(move(child))
  33. {
  34. }
  35. virtual void render_to_html(StringBuilder& builder) const override;
  36. virtual void render_for_terminal(StringBuilder& builder) const override;
  37. virtual size_t terminal_length() const override;
  38. virtual RecursionDecision walk(Visitor&) const override;
  39. };
  40. class CodeNode : public Node {
  41. public:
  42. NonnullOwnPtr<Node> code;
  43. CodeNode(NonnullOwnPtr<Node> code)
  44. : code(move(code))
  45. {
  46. }
  47. virtual void render_to_html(StringBuilder& builder) const override;
  48. virtual void render_for_terminal(StringBuilder& builder) const override;
  49. virtual size_t terminal_length() const override;
  50. virtual RecursionDecision walk(Visitor&) const override;
  51. };
  52. class BreakNode : public Node {
  53. public:
  54. virtual void render_to_html(StringBuilder& builder) const override;
  55. virtual void render_for_terminal(StringBuilder& builder) const override;
  56. virtual size_t terminal_length() const override;
  57. virtual RecursionDecision walk(Visitor&) const override;
  58. };
  59. class TextNode : public Node {
  60. public:
  61. String text;
  62. bool collapsible;
  63. TextNode(StringView text)
  64. : text(text)
  65. , collapsible(true)
  66. {
  67. }
  68. TextNode(StringView text, bool collapsible)
  69. : text(text)
  70. , collapsible(collapsible)
  71. {
  72. }
  73. virtual void render_to_html(StringBuilder& builder) const override;
  74. virtual void render_for_terminal(StringBuilder& builder) const override;
  75. virtual size_t terminal_length() const override;
  76. virtual RecursionDecision walk(Visitor&) const override;
  77. };
  78. class LinkNode : public Node {
  79. public:
  80. bool is_image;
  81. NonnullOwnPtr<Node> text;
  82. String href;
  83. LinkNode(bool is_image, NonnullOwnPtr<Node> text, String href)
  84. : is_image(is_image)
  85. , text(move(text))
  86. , href(move(href))
  87. {
  88. }
  89. virtual void render_to_html(StringBuilder& builder) const override;
  90. virtual void render_for_terminal(StringBuilder& builder) const override;
  91. virtual size_t terminal_length() const override;
  92. virtual RecursionDecision walk(Visitor&) const override;
  93. };
  94. class MultiNode : public Node {
  95. public:
  96. NonnullOwnPtrVector<Node> children;
  97. virtual void render_to_html(StringBuilder& builder) const override;
  98. virtual void render_for_terminal(StringBuilder& builder) const override;
  99. virtual size_t terminal_length() const override;
  100. virtual RecursionDecision walk(Visitor&) const override;
  101. };
  102. size_t terminal_length() const;
  103. String render_to_html() const;
  104. String render_for_terminal() const;
  105. RecursionDecision walk(Visitor&) const;
  106. static Text parse(StringView);
  107. private:
  108. struct Token {
  109. String data;
  110. // Flanking basically means that a delimiter run has a non-whitespace,
  111. // non-punctuation character on the corresponding side. For a more exact
  112. // definition, see the CommonMark spec.
  113. bool left_flanking;
  114. bool right_flanking;
  115. bool punct_before;
  116. bool punct_after;
  117. // is_run indicates that this token is a 'delimiter run'. A delimiter
  118. // run occurs when several of the same syntactical character ('`', '_',
  119. // or '*') occur in a row.
  120. bool is_run;
  121. char run_char() const
  122. {
  123. VERIFY(is_run);
  124. return data[0];
  125. }
  126. char run_length() const
  127. {
  128. VERIFY(is_run);
  129. return data.length();
  130. }
  131. bool is_space() const
  132. {
  133. return data[0] == ' ';
  134. }
  135. bool operator==(StringView str) const { return str == data; }
  136. };
  137. static Vector<Token> tokenize(StringView);
  138. static bool can_open(Token const& opening);
  139. static bool can_close_for(Token const& opening, Token const& closing);
  140. static NonnullOwnPtr<MultiNode> parse_sequence(Vector<Token>::ConstIterator& tokens, bool in_link);
  141. static NonnullOwnPtr<Node> parse_break(Vector<Token>::ConstIterator& tokens);
  142. static NonnullOwnPtr<Node> parse_newline(Vector<Token>::ConstIterator& tokens);
  143. static NonnullOwnPtr<Node> parse_emph(Vector<Token>::ConstIterator& tokens, bool in_link);
  144. static NonnullOwnPtr<Node> parse_code(Vector<Token>::ConstIterator& tokens);
  145. static NonnullOwnPtr<Node> parse_link(Vector<Token>::ConstIterator& tokens);
  146. OwnPtr<Node> m_node;
  147. };
  148. }