Coverage Report

Created: 2026-02-16 07:47

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/serenity/Userland/Libraries/LibMarkdown/List.cpp
Line
Count
Source
1
/*
2
 * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
3
 * Copyright (c) 2021, Peter Elliott <pelliott@serenityos.org>
4
 *
5
 * SPDX-License-Identifier: BSD-2-Clause
6
 */
7
8
#include <AK/Forward.h>
9
#include <AK/StringBuilder.h>
10
#include <LibMarkdown/List.h>
11
#include <LibMarkdown/Paragraph.h>
12
#include <LibMarkdown/Visitor.h>
13
14
namespace Markdown {
15
16
ByteString List::render_to_html(bool) const
17
0
{
18
0
    StringBuilder builder;
19
20
0
    char const* tag = m_is_ordered ? "ol" : "ul";
21
0
    builder.appendff("<{}", tag);
22
23
0
    if (m_start_number != 1)
24
0
        builder.appendff(" start=\"{}\"", m_start_number);
25
26
0
    builder.append(">\n"sv);
27
28
0
    for (auto& item : m_items) {
29
0
        builder.append("<li>"sv);
30
0
        if (!m_is_tight || (item->blocks().size() != 0 && !dynamic_cast<Paragraph const*>(item->blocks()[0].ptr())))
31
0
            builder.append('\n');
32
0
        builder.append(item->render_to_html(m_is_tight));
33
0
        builder.append("</li>\n"sv);
34
0
    }
35
36
0
    builder.appendff("</{}>\n", tag);
37
38
0
    return builder.to_byte_string();
39
0
}
40
41
Vector<ByteString> List::render_lines_for_terminal(size_t view_width) const
42
0
{
43
0
    Vector<ByteString> lines;
44
45
0
    int i = 0;
46
0
    for (auto& item : m_items) {
47
0
        auto item_lines = item->render_lines_for_terminal(view_width);
48
0
        auto first_line = item_lines.take_first();
49
50
0
        StringBuilder builder;
51
0
        builder.append("  "sv);
52
0
        if (m_is_ordered)
53
0
            builder.appendff("{}.", ++i);
54
0
        else
55
0
            builder.append('*');
56
0
        auto item_indentation = builder.length();
57
58
0
        builder.append(first_line);
59
60
0
        lines.append(builder.to_byte_string());
61
62
0
        for (auto& line : item_lines) {
63
0
            builder.clear();
64
0
            builder.append(ByteString::repeated(' ', item_indentation));
65
0
            builder.append(line);
66
0
            lines.append(builder.to_byte_string());
67
0
        }
68
0
    }
69
70
0
    return lines;
71
0
}
72
73
RecursionDecision List::walk(Visitor& visitor) const
74
0
{
75
0
    RecursionDecision rd = visitor.visit(*this);
76
0
    if (rd != RecursionDecision::Recurse)
77
0
        return rd;
78
79
0
    for (auto const& block : m_items) {
80
0
        rd = block->walk(visitor);
81
0
        if (rd == RecursionDecision::Break)
82
0
            return rd;
83
0
    }
84
85
0
    return RecursionDecision::Continue;
86
0
}
87
88
OwnPtr<List> List::parse(LineIterator& lines)
89
10.9M
{
90
10.9M
    Vector<OwnPtr<ContainerBlock>> items;
91
92
10.9M
    bool first = true;
93
10.9M
    bool is_ordered = false;
94
95
10.9M
    bool is_tight = true;
96
10.9M
    bool has_trailing_blank_lines = false;
97
10.9M
    size_t start_number = 1;
98
99
11.8M
    while (!lines.is_end()) {
100
101
11.8M
        size_t offset = 0;
102
103
11.8M
        StringView line = *lines;
104
105
11.8M
        bool appears_unordered = false;
106
107
15.4M
        while (offset < line.length() && line[offset] == ' ')
108
3.64M
            ++offset;
109
110
11.8M
        if (offset + 2 <= line.length()) {
111
4.89M
            if (line[offset + 1] == ' ' && (line[offset] == '*' || line[offset] == '-' || line[offset] == '+')) {
112
844k
                appears_unordered = true;
113
844k
                offset++;
114
844k
            }
115
4.89M
        }
116
117
11.8M
        bool appears_ordered = false;
118
11.8M
        for (size_t i = offset; i < 10 && i < line.length(); i++) {
119
11.8M
            char ch = line[i];
120
11.8M
            if ('0' <= ch && ch <= '9')
121
17.0k
                continue;
122
11.8M
            if (ch == '.' || ch == ')')
123
4.16M
                if (i + 1 < line.length() && line[i + 1] == ' ') {
124
143k
                    auto maybe_start_number = line.substring_view(offset, i - offset).to_number<size_t>();
125
143k
                    if (!maybe_start_number.has_value())
126
133k
                        break;
127
10.6k
                    if (first)
128
10.0k
                        start_number = maybe_start_number.value();
129
10.6k
                    appears_ordered = true;
130
10.6k
                    offset = i + 1;
131
10.6k
                }
132
11.6M
            break;
133
11.8M
        }
134
135
11.8M
        VERIFY(!(appears_unordered && appears_ordered));
136
11.8M
        if (!appears_unordered && !appears_ordered) {
137
10.9M
            if (first)
138
10.9M
                return {};
139
140
57.1k
            break;
141
10.9M
        }
142
143
2.97M
        while (offset < line.length() && line[offset] == ' ')
144
2.12M
            offset++;
145
146
855k
        if (first) {
147
86.3k
            is_ordered = appears_ordered;
148
769k
        } else if (appears_ordered != is_ordered) {
149
758
            break;
150
758
        }
151
152
854k
        is_tight = is_tight && !has_trailing_blank_lines;
153
154
854k
        lines.push_context(LineIterator::Context::list_item(offset));
155
156
854k
        auto list_item = ContainerBlock::parse(lines);
157
854k
        is_tight = is_tight && !list_item->has_blank_lines();
158
854k
        has_trailing_blank_lines = has_trailing_blank_lines || list_item->has_trailing_blank_lines();
159
854k
        items.append(move(list_item));
160
161
854k
        lines.pop_context();
162
163
854k
        first = false;
164
854k
    }
165
166
86.3k
    return make<List>(move(items), is_ordered, is_tight, start_number);
167
10.9M
}
168
169
}