Heading.cpp 2.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. /*
  2. * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Slugify.h>
  7. #include <AK/StringBuilder.h>
  8. #include <LibMarkdown/Heading.h>
  9. #include <LibMarkdown/Visitor.h>
  10. #include <LibUnicode/Normalize.h>
  11. namespace Markdown {
  12. ByteString Heading::render_to_html(bool) const
  13. {
  14. auto input = Unicode::normalize(m_text.render_for_raw_print().view(), Unicode::NormalizationForm::NFD);
  15. auto slugified = MUST(AK::slugify(input));
  16. return ByteString::formatted("<h{} id='{}'><a href='#{}'>#</a> {}</h{}>\n", m_level, slugified, slugified, m_text.render_to_html(), m_level);
  17. }
  18. Vector<ByteString> Heading::render_lines_for_terminal(size_t) const
  19. {
  20. StringBuilder builder;
  21. builder.append("\n\033[0;31;1m"sv);
  22. switch (m_level) {
  23. case 1:
  24. case 2:
  25. builder.append(m_text.render_for_terminal().to_uppercase());
  26. builder.append("\033[0m"sv);
  27. break;
  28. default:
  29. builder.append(m_text.render_for_terminal());
  30. builder.append("\033[0m"sv);
  31. break;
  32. }
  33. return Vector<ByteString> { builder.to_byte_string() };
  34. }
  35. RecursionDecision Heading::walk(Visitor& visitor) const
  36. {
  37. RecursionDecision rd = visitor.visit(*this);
  38. if (rd != RecursionDecision::Recurse)
  39. return rd;
  40. return m_text.walk(visitor);
  41. }
  42. OwnPtr<Heading> Heading::parse(LineIterator& lines)
  43. {
  44. if (lines.is_end())
  45. return {};
  46. StringView line = *lines;
  47. size_t indent = 0;
  48. // Allow for up to 3 spaces of indentation.
  49. // https://spec.commonmark.org/0.30/#example-68
  50. for (size_t i = 0; i < 3; ++i) {
  51. if (line[i] != ' ')
  52. break;
  53. ++indent;
  54. }
  55. size_t level;
  56. for (level = 0; indent + level < line.length(); level++) {
  57. if (line[indent + level] != '#')
  58. break;
  59. }
  60. if (!level || indent + level >= line.length() || line[indent + level] != ' ' || level > 6)
  61. return {};
  62. StringView title_view = line.substring_view(indent + level + 1);
  63. auto text = Text::parse(title_view);
  64. auto heading = make<Heading>(move(text), level);
  65. ++lines;
  66. return heading;
  67. }
  68. }